2022-09-20 13:43:26 +02:00
|
|
|
/*
|
|
|
|
* PtgCtrl.cpp
|
|
|
|
*
|
|
|
|
* Created on: 17 Jul 2022
|
|
|
|
* Author: Robin Marquardt
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "PtgCtrl.h"
|
2022-12-01 15:56:55 +01:00
|
|
|
|
2022-09-27 11:06:11 +02:00
|
|
|
#include <fsfw/globalfunctions/constants.h>
|
|
|
|
#include <fsfw/globalfunctions/math/MatrixOperations.h>
|
|
|
|
#include <fsfw/globalfunctions/math/QuaternionOperations.h>
|
|
|
|
#include <fsfw/globalfunctions/math/VectorOperations.h>
|
|
|
|
#include <fsfw/globalfunctions/sign.h>
|
2022-09-20 13:43:26 +02:00
|
|
|
#include <math.h>
|
|
|
|
|
2022-12-01 15:56:55 +01:00
|
|
|
#include "../util/MathOperations.h"
|
2022-09-20 13:43:26 +02:00
|
|
|
|
2022-12-13 11:51:03 +01:00
|
|
|
PtgCtrl::PtgCtrl(AcsParameters *acsParameters_) { loadAcsParameters(acsParameters_); }
|
2022-09-20 13:43:26 +02:00
|
|
|
|
2022-12-01 15:56:55 +01:00
|
|
|
PtgCtrl::~PtgCtrl() {}
|
2022-09-20 13:43:26 +02:00
|
|
|
|
2022-12-01 15:56:55 +01:00
|
|
|
void PtgCtrl::loadAcsParameters(AcsParameters *acsParameters_) {
|
|
|
|
inertiaEIVE = &(acsParameters_->inertiaEIVE);
|
|
|
|
rwHandlingParameters = &(acsParameters_->rwHandlingParameters);
|
|
|
|
rwMatrices = &(acsParameters_->rwMatrices);
|
2022-09-20 13:43:26 +02:00
|
|
|
}
|
|
|
|
|
2023-01-23 16:34:52 +01:00
|
|
|
void PtgCtrl::ptgLaw(AcsParameters::PointingLawParameters *pointingLawParameters,
|
2023-02-17 15:57:07 +01:00
|
|
|
const double *errorQuat, const double *deltaRate, const double *rwPseudoInv,
|
2023-01-23 16:34:52 +01:00
|
|
|
double *torqueRws) {
|
2022-12-01 15:56:55 +01:00
|
|
|
//------------------------------------------------------------------------------------------------
|
|
|
|
// Compute gain matrix K and P matrix
|
|
|
|
//------------------------------------------------------------------------------------------------
|
2023-01-12 15:19:21 +01:00
|
|
|
double om = pointingLawParameters->om;
|
|
|
|
double zeta = pointingLawParameters->zeta;
|
|
|
|
double qErrorMin = pointingLawParameters->qiMin;
|
|
|
|
double omMax = pointingLawParameters->omMax;
|
2022-12-01 15:56:55 +01:00
|
|
|
|
2023-02-17 15:57:07 +01:00
|
|
|
double qError[3] = {errorQuat[0], errorQuat[1], errorQuat[2]};
|
|
|
|
|
2022-12-01 15:56:55 +01:00
|
|
|
double cInt = 2 * om * zeta;
|
|
|
|
double kInt = 2 * pow(om, 2);
|
|
|
|
|
|
|
|
double qErrorLaw[3] = {0, 0, 0};
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
if (abs(qError[i]) < qErrorMin) {
|
|
|
|
qErrorLaw[i] = qErrorMin;
|
|
|
|
} else {
|
|
|
|
qErrorLaw[i] = abs(qError[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
double qErrorLawNorm = VectorOperations<double>::norm(qErrorLaw, 3);
|
|
|
|
|
|
|
|
double gain1 = cInt * omMax / qErrorLawNorm;
|
|
|
|
double gainVector[3] = {0, 0, 0};
|
|
|
|
VectorOperations<double>::mulScalar(qErrorLaw, gain1, gainVector, 3);
|
|
|
|
|
|
|
|
double gainMatrixDiagonal[3][3] = {{0, 0, 0}, {0, 0, 0}, {0, 0, 0}};
|
|
|
|
double gainMatrix[3][3] = {{0, 0, 0}, {0, 0, 0}, {0, 0, 0}};
|
|
|
|
gainMatrixDiagonal[0][0] = gainVector[0];
|
|
|
|
gainMatrixDiagonal[1][1] = gainVector[1];
|
|
|
|
gainMatrixDiagonal[2][2] = gainVector[2];
|
|
|
|
MatrixOperations<double>::multiply(*gainMatrixDiagonal, *(inertiaEIVE->inertiaMatrix),
|
|
|
|
*gainMatrix, 3, 3, 3);
|
|
|
|
|
|
|
|
// Inverse of gainMatrix
|
|
|
|
double gainMatrixInverse[3][3] = {{0, 0, 0}, {0, 0, 0}, {0, 0, 0}};
|
|
|
|
gainMatrixInverse[0][0] = 1 / gainMatrix[0][0];
|
|
|
|
gainMatrixInverse[1][1] = 1 / gainMatrix[1][1];
|
|
|
|
gainMatrixInverse[2][2] = 1 / gainMatrix[2][2];
|
|
|
|
|
|
|
|
double pMatrix[3][3] = {{0, 0, 0}, {0, 0, 0}, {0, 0, 0}};
|
|
|
|
MatrixOperations<double>::multiply(*gainMatrixInverse, *(inertiaEIVE->inertiaMatrix), *pMatrix, 3,
|
|
|
|
3, 3);
|
|
|
|
MatrixOperations<double>::multiplyScalar(*pMatrix, kInt, *pMatrix, 3, 3);
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------------------------
|
|
|
|
// Torque Calculations for the reaction wheels
|
|
|
|
//------------------------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
double pError[3] = {0, 0, 0};
|
|
|
|
MatrixOperations<double>::multiply(*pMatrix, qError, pError, 3, 3, 1);
|
|
|
|
double pErrorSign[3] = {0, 0, 0};
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; i++) {
|
2022-12-13 11:51:03 +01:00
|
|
|
if (abs(pError[i]) > 1) {
|
|
|
|
pErrorSign[i] = sign(pError[i]);
|
|
|
|
} else {
|
|
|
|
pErrorSign[i] = pError[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Torque for quaternion error
|
|
|
|
double torqueQuat[3] = {0, 0, 0};
|
|
|
|
MatrixOperations<double>::multiply(*gainMatrix, pErrorSign, torqueQuat, 3, 3, 1);
|
|
|
|
VectorOperations<double>::mulScalar(torqueQuat, -1, torqueQuat, 3);
|
|
|
|
|
|
|
|
// Torque for rate error
|
|
|
|
double torqueRate[3] = {0, 0, 0};
|
|
|
|
MatrixOperations<double>::multiply(*(inertiaEIVE->inertiaMatrix), deltaRate, torqueRate, 3, 3, 1);
|
|
|
|
VectorOperations<double>::mulScalar(torqueRate, cInt, torqueRate, 3);
|
|
|
|
VectorOperations<double>::mulScalar(torqueRate, -1, torqueRate, 3);
|
|
|
|
|
|
|
|
// Final commanded Torque for every reaction wheel
|
|
|
|
double torque[3] = {0, 0, 0};
|
|
|
|
VectorOperations<double>::add(torqueRate, torqueQuat, torque, 3);
|
|
|
|
MatrixOperations<double>::multiply(rwPseudoInv, torque, torqueRws, 4, 3, 1);
|
|
|
|
VectorOperations<double>::mulScalar(torqueRws, -1, torqueRws, 4);
|
2022-09-20 13:43:26 +02:00
|
|
|
}
|
|
|
|
|
2023-01-23 16:34:52 +01:00
|
|
|
void PtgCtrl::ptgDesaturation(AcsParameters::PointingLawParameters *pointingLawParameters,
|
|
|
|
double *magFieldEst, bool magFieldEstValid, double *satRate,
|
2022-10-12 15:06:24 +02:00
|
|
|
int32_t *speedRw0, int32_t *speedRw1, int32_t *speedRw2,
|
|
|
|
int32_t *speedRw3, double *mgtDpDes) {
|
2023-01-12 15:19:21 +01:00
|
|
|
if (!(magFieldEstValid) || !(pointingLawParameters->desatOn)) {
|
2022-10-12 15:06:24 +02:00
|
|
|
mgtDpDes[0] = 0;
|
|
|
|
mgtDpDes[1] = 0;
|
|
|
|
mgtDpDes[2] = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// calculating momentum of satellite and momentum of reaction wheels
|
2022-10-24 10:29:57 +02:00
|
|
|
double speedRws[4] = {(double)*speedRw0, (double)*speedRw1, (double)*speedRw2, (double)*speedRw3};
|
2022-10-12 15:06:24 +02:00
|
|
|
double momentumRwU[4] = {0, 0, 0, 0}, momentumRw[3] = {0, 0, 0};
|
|
|
|
VectorOperations<double>::mulScalar(speedRws, rwHandlingParameters->inertiaWheel, momentumRwU, 4);
|
|
|
|
MatrixOperations<double>::multiply(*(rwMatrices->alignmentMatrix), momentumRwU, momentumRw, 3, 4,
|
|
|
|
1);
|
|
|
|
double momentumSat[3] = {0, 0, 0}, momentumTotal[3] = {0, 0, 0};
|
|
|
|
MatrixOperations<double>::multiply(*(inertiaEIVE->inertiaMatrix), satRate, momentumSat, 3, 3, 1);
|
|
|
|
VectorOperations<double>::add(momentumSat, momentumRw, momentumTotal, 3);
|
|
|
|
// calculating momentum error
|
|
|
|
double deltaMomentum[3] = {0, 0, 0};
|
2023-01-23 16:34:52 +01:00
|
|
|
VectorOperations<double>::subtract(momentumTotal, pointingLawParameters->desatMomentumRef,
|
|
|
|
deltaMomentum, 3);
|
2022-10-12 15:06:24 +02:00
|
|
|
// resulting magnetic dipole command
|
|
|
|
double crossMomentumMagField[3] = {0, 0, 0};
|
|
|
|
VectorOperations<double>::cross(deltaMomentum, magFieldEst, crossMomentumMagField);
|
|
|
|
double normMag = VectorOperations<double>::norm(magFieldEst, 3), factor = 0;
|
2023-01-12 15:19:21 +01:00
|
|
|
factor = (pointingLawParameters->deSatGainFactor) / normMag;
|
2022-10-12 15:06:24 +02:00
|
|
|
VectorOperations<double>::mulScalar(crossMomentumMagField, factor, mgtDpDes, 3);
|
2022-09-20 13:43:26 +02:00
|
|
|
}
|
|
|
|
|
2023-01-23 16:34:52 +01:00
|
|
|
void PtgCtrl::ptgNullspace(AcsParameters::PointingLawParameters *pointingLawParameters,
|
|
|
|
const int32_t *speedRw0, const int32_t *speedRw1,
|
2022-10-12 15:06:24 +02:00
|
|
|
const int32_t *speedRw2, const int32_t *speedRw3, double *rwTrqNs) {
|
2022-10-24 10:29:57 +02:00
|
|
|
double speedRws[4] = {(double)*speedRw0, (double)*speedRw1, (double)*speedRw2, (double)*speedRw3};
|
2022-10-12 15:06:24 +02:00
|
|
|
double wheelMomentum[4] = {0, 0, 0, 0};
|
|
|
|
double rpmOffset[4] = {1, 1, 1, -1}, factor = 350 * 2 * Math::PI / 60;
|
|
|
|
// Conversion to [rad/s] for further calculations
|
|
|
|
VectorOperations<double>::mulScalar(rpmOffset, factor, rpmOffset, 4);
|
|
|
|
VectorOperations<double>::mulScalar(speedRws, 2 * Math::PI / 60, speedRws, 4);
|
|
|
|
double diffRwSpeed[4] = {0, 0, 0, 0};
|
|
|
|
VectorOperations<double>::subtract(speedRws, rpmOffset, diffRwSpeed, 4);
|
|
|
|
VectorOperations<double>::mulScalar(diffRwSpeed, rwHandlingParameters->inertiaWheel,
|
|
|
|
wheelMomentum, 4);
|
2023-01-12 15:19:21 +01:00
|
|
|
double gainNs = pointingLawParameters->gainNullspace;
|
2022-10-12 15:06:24 +02:00
|
|
|
double nullSpaceMatrix[4][4] = {{0, 0, 0}, {0, 0, 0}, {0, 0, 0}, {0, 0, 0}};
|
|
|
|
MathOperations<double>::vecTransposeVecMatrix(rwMatrices->nullspace, rwMatrices->nullspace,
|
|
|
|
*nullSpaceMatrix, 4);
|
|
|
|
MatrixOperations<double>::multiply(*nullSpaceMatrix, wheelMomentum, rwTrqNs, 4, 4, 1);
|
|
|
|
VectorOperations<double>::mulScalar(rwTrqNs, gainNs, rwTrqNs, 4);
|
|
|
|
VectorOperations<double>::mulScalar(rwTrqNs, -1, rwTrqNs, 4);
|
2022-09-20 13:43:26 +02:00
|
|
|
}
|
2022-11-08 13:48:50 +01:00
|
|
|
|
2023-02-17 09:21:55 +01:00
|
|
|
void PtgCtrl::rwAntistiction(ACS::SensorValues *sensorValues, double *torqueCommand) {
|
|
|
|
bool rwAvailable[4] = {
|
|
|
|
(sensorValues->rw1Set.state.value && sensorValues->rw1Set.state.isValid()),
|
|
|
|
(sensorValues->rw2Set.state.value && sensorValues->rw2Set.state.isValid()),
|
|
|
|
(sensorValues->rw3Set.state.value && sensorValues->rw3Set.state.isValid()),
|
|
|
|
(sensorValues->rw4Set.state.value && sensorValues->rw4Set.state.isValid())};
|
|
|
|
int32_t omegaRw[4] = {sensorValues->rw1Set.currSpeed.value, sensorValues->rw2Set.currSpeed.value,
|
|
|
|
sensorValues->rw3Set.currSpeed.value, sensorValues->rw4Set.currSpeed.value};
|
2022-12-13 11:51:03 +01:00
|
|
|
for (uint8_t i = 0; i < 4; i++) {
|
|
|
|
if (rwAvailable[i]) {
|
|
|
|
if (torqueMemory[i] != 0) {
|
|
|
|
if ((omegaRw[i] * torqueMemory[i]) > rwHandlingParameters->stictionReleaseSpeed) {
|
|
|
|
torqueMemory[i] = 0;
|
|
|
|
} else {
|
|
|
|
torqueCommand[i] = torqueMemory[i] * rwHandlingParameters->stictionTorque;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((omegaRw[i] < rwHandlingParameters->stictionSpeed) &&
|
|
|
|
(omegaRw[i] > -rwHandlingParameters->stictionSpeed)) {
|
|
|
|
if (omegaRw[i] < omegaMemory[i]) {
|
|
|
|
torqueMemory[i] = -1;
|
|
|
|
} else {
|
|
|
|
torqueMemory[i] = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
torqueCommand[i] = torqueMemory[i] * rwHandlingParameters->stictionTorque;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
torqueMemory[i] = 0;
|
|
|
|
}
|
|
|
|
omegaMemory[i] = omegaRw[i];
|
|
|
|
}
|
|
|
|
}
|