36# pragma warning (disable : 4786)
43#include "models/FGAtmosphere.h"
44#include "FGInitialCondition.h"
45#include "FGTrimAxis.h"
46#include "models/FGPropulsion.h"
47#include "models/FGAerodynamics.h"
48#include "models/FGFCS.h"
49#include "models/propulsion/FGEngine.h"
50#include "models/FGAuxiliary.h"
51#include "models/FGGroundReactions.h"
52#include "models/FGPropagate.h"
53#include "models/FGAccelerations.h"
70 its_to_stable_value=0;
72 total_stability_iterations=0;
78 case tUdot: tolerance = DEFAULT_TOLERANCE;
break;
79 case tVdot: tolerance = DEFAULT_TOLERANCE;
break;
80 case tWdot: tolerance = DEFAULT_TOLERANCE;
break;
81 case tQdot: tolerance = DEFAULT_TOLERANCE / 10;
break;
82 case tPdot: tolerance = DEFAULT_TOLERANCE / 10;
break;
83 case tRdot: tolerance = DEFAULT_TOLERANCE / 10;
break;
84 case tHmgt: tolerance = 0.01;
break;
85 case tNlf: state_target=1.0; tolerance = 1E-5;
break;
97 control_min=-30*degtorad;
98 control_max=30*degtorad;
99 control_convert=radtodeg;
104 if(control_max <= control_min) {
105 control_max=20*degtorad;
106 control_min=-5*degtorad;
108 control_value= (control_min+control_max)/2;
109 control_convert=radtodeg;
110 solver_eps=tolerance/100;
120 state_convert=radtodeg;
121 solver_eps=tolerance/100;
127 solver_eps=tolerance/100;
132 state_convert=radtodeg;
137 state_convert=radtodeg;
138 control_convert=radtodeg;
141 solver_eps=tolerance/100;
142 control_min=-80*degtorad;
143 control_max=80*degtorad;
144 control_convert=radtodeg;
149 state_convert=radtodeg;
166void FGTrimAxis::getState(
void) {
168 case tUdot: state_value=fdmex->
GetAccelerations()->GetUVWdot(1)-state_target;
break;
169 case tVdot: state_value=fdmex->
GetAccelerations()->GetUVWdot(2)-state_target;
break;
170 case tWdot: state_value=fdmex->
GetAccelerations()->GetUVWdot(3)-state_target;
break;
171 case tQdot: state_value=fdmex->
GetAccelerations()->GetPQRdot(2)-state_target;
break;
172 case tPdot: state_value=fdmex->
GetAccelerations()->GetPQRdot(1)-state_target;
break;
173 case tRdot: state_value=fdmex->
GetAccelerations()->GetPQRdot(3)-state_target;
break;
174 case tHmgt: state_value=computeHmgt()-state_target;
break;
175 case tNlf: state_value=fdmex->
GetAuxiliary()->GetNlf()-state_target;
break;
184void FGTrimAxis::getControl(
void) {
186 case tThrottle: control_value=fdmex->
GetFCS()->GetThrottleCmd(0);
break;
187 case tBeta: control_value=fdmex->
GetAuxiliary()->Getbeta();
break;
188 case tAlpha: control_value=fdmex->
GetAuxiliary()->Getalpha();
break;
189 case tPitchTrim: control_value=fdmex->
GetFCS() -> GetPitchTrimCmd();
break;
190 case tElevator: control_value=fdmex->
GetFCS() -> GetDeCmd();
break;
192 case tAileron: control_value=fdmex->
GetFCS() -> GetDaCmd();
break;
194 case tRudder: control_value=fdmex->
GetFCS() -> GetDrCmd();
break;
195 case tAltAGL: control_value=fdmex->
GetPropagate()->GetDistanceAGL();
break;
196 case tTheta: control_value=fdmex->
GetPropagate()->GetEuler(eTht);
break;
197 case tPhi: control_value=fdmex->
GetPropagate()->GetEuler(ePhi);
break;
198 case tGamma: control_value=fdmex->
GetAuxiliary()->GetGamma();
break;
199 case tHeading: control_value=fdmex->
GetPropagate()->GetEuler(ePsi);
break;
205double FGTrimAxis::computeHmgt(
void) {
212 return (diff + 2*M_PI);
213 }
else if( diff > M_PI ) {
214 return (diff - 2*M_PI);
224void FGTrimAxis::setControl(
void) {
226 case tThrottle: setThrottlesPct();
break;
229 case tPitchTrim: fdmex->
GetFCS()->SetPitchTrimCmd(control_value);
break;
230 case tElevator: fdmex->
GetFCS()->SetDeCmd(control_value);
break;
232 case tAileron: fdmex->
GetFCS()->SetDaCmd(control_value);
break;
234 case tRudder: fdmex->
GetFCS()->SetDrCmd(control_value);
break;
237 case tPhi: fgic->
SetPhiRadIC(control_value);
break;
239 case tHeading: fgic->
SetPsiRadIC(control_value);
break;
247 double last_state_value;
255 last_state_value=state_value;
260 if((fabs(last_state_value - state_value) < tolerance) || (i >= 100) )
265 its_to_stable_value=i;
266 total_stability_iterations+=its_to_stable_value;
272void FGTrimAxis::setThrottlesPct(
void) {
274 for(
unsigned i=0;i<fdmex->
GetPropulsion()->GetNumEngines();i++) {
280 fdmex->
GetFCS()->SetThrottleCmd(i,tMin+control_value*(tMax-tMin));
281 fdmex->
GetPropulsion()->in.ThrottlePos[i] = tMin +control_value*(tMax - tMin);
291void FGTrimAxis::AxisReport(
void) {
293 std::ios_base::fmtflags originalFormat = cout.flags();
294 std::streamsize originalPrecision = cout.precision();
295 std::streamsize originalWidth = cout.width();
296 cout <<
" " << setw(20) << GetControlName() <<
": ";
297 cout << setw(6) << setprecision(2) << GetControl()*control_convert <<
' ';
298 cout << setw(5) << GetStateName() <<
": ";
299 cout << setw(9) << setprecision(2) << scientific << GetState()+state_target;
300 cout <<
" Tolerance: " << setw(3) << setprecision(0) << scientific << GetTolerance();
302 if( fabs(GetState()+state_target) < fabs(GetTolerance()) )
303 cout <<
" Passed" << endl;
305 cout <<
" Failed" << endl;
307 cout.flags(originalFormat);
308 cout.precision(originalPrecision);
309 cout.width(originalWidth);
314double FGTrimAxis::GetAvgStability(
void ) {
315 if(total_iterations > 0) {
316 return double(total_stability_iterations)/double(total_iterations);
340void FGTrimAxis::Debug(
int from)
343 if (debug_lvl <= 0)
return;
344 if (debug_lvl & 1 ) {
349 if (debug_lvl & 2 ) {
350 if (from == 0) cout <<
"Instantiated: FGTrimAxis" << endl;
351 if (from == 1) cout <<
"Destroyed: FGTrimAxis" << endl;
353 if (debug_lvl & 4 ) {
355 if (debug_lvl & 8 ) {
357 if (debug_lvl & 16) {
359 if (debug_lvl & 64) {
Encapsulates the JSBSim simulation executive.
std::shared_ptr< FGFCS > GetFCS(void) const
Returns the FGFCS pointer.
std::shared_ptr< FGPropagate > GetPropagate(void) const
Returns the FGPropagate pointer.
std::shared_ptr< FGAerodynamics > GetAerodynamics(void) const
Returns the FGAerodynamics pointer.
bool Run(void)
This function executes each scheduled model in succession.
std::shared_ptr< FGPropulsion > GetPropulsion(void) const
Returns the FGPropulsion pointer.
void Initialize(const FGInitialCondition *FGIC)
Initializes the simulation with initial conditions.
std::shared_ptr< FGAuxiliary > GetAuxiliary(void) const
Returns the FGAuxiliary pointer.
std::shared_ptr< FGAccelerations > GetAccelerations(void) const
Returns the FGAccelerations pointer.
Initializes the simulation run.
void SetPsiRadIC(double psi)
Sets the initial heading angle.
double GetPhiRadIC(void) const
Gets the initial roll angle.
void SetFlightPathAngleRadIC(double gamma)
Sets the initial flight path angle.
void SetBetaRadIC(double beta)
Sets the initial sideslip angle.
void SetAltitudeAGLFtIC(double agl)
Sets the initial Altitude above ground level.
double GetAltitudeAGLFtIC(void) const
Gets the initial altitude above ground level.
void SetThetaRadIC(double theta)
Sets the initial pitch angle.
void SetAlphaRadIC(double alpha)
Sets the initial angle of attack.
double GetPsiRadIC(void) const
Gets the initial heading angle.
double GetThetaRadIC(void) const
Gets the initial pitch angle.
void SetPhiRadIC(double phi)
Sets the initial roll angle.
void Run(void)
This function iterates through a call to the FGFDMExec::RunIC() function until the desired trimming c...
FGTrimAxis(FGFDMExec *fdmex, FGInitialCondition *IC, State state, Control control)
Constructor for Trim Axis class.