39#include "FGAerodynamics.h"
40#include "input_output/FGXMLElement.h"
53 Name =
"FGAerodynamics";
63 AxisIdx[
"NORMAL"] = 2;
69 forceAxisType = atNone;
70 momentAxisType = atNone;
72 AeroFunctions =
new AeroFunctionArray[6];
73 AeroFunctionsAtCG =
new AeroFunctionArray[6];
75 impending_stall = stall_hyst = 0.0;
76 alphaclmin = alphaclmax = 0.0;
77 alphaclmin0 = alphaclmax0 = 0.0;
78 alphahystmin = alphahystmax = 0.0;
81 bi2vel = ci2vel = 0.0;
83 vDeltaRP.InitMatrix();
97 for (j=0; j<AeroFunctions[i].size(); j++)
98 delete AeroFunctions[i][j];
100 for (j=0; j<AeroFunctionsAtCG[i].size(); j++)
101 delete AeroFunctionsAtCG[i][j];
103 delete[] AeroFunctions;
104 delete[] AeroFunctionsAtCG;
113bool FGAerodynamics::InitModel(
void)
115 if (!FGModel::InitModel())
return false;
117 impending_stall = stall_hyst = 0.0;
118 alphaclmin = alphaclmin0;
119 alphaclmax = alphaclmax0;
120 alphahystmin = alphahystmax = 0.0;
123 bi2vel = ci2vel = 0.0;
125 vDeltaRP.InitMatrix();
126 vForces.InitMatrix();
127 vMoments.InitMatrix();
136 if (Holding)
return false;
138 unsigned int axis_ctr;
139 const double twovel=2*in.Vt;
144 if ( in.Qbar > 1.0) {
148 clsq = vFw(eLift) / (in.Wingarea*in.Qbar);
157 bi2vel = in.Wingspan / twovel;
158 ci2vel = in.Wingchord / twovel;
160 alphaw = in.Alpha + in.Wingincidence;
161 qbar_area = in.Wingarea * in.Qbar;
163 if (alphaclmax != 0) {
164 if (in.Alpha > 0.85*alphaclmax) {
165 impending_stall = 10*(in.Alpha/alphaclmax - 0.85);
171 if (alphahystmax != 0.0 && alphahystmin != 0.0) {
172 if (in.Alpha > alphahystmax) {
174 }
else if (in.Alpha < alphahystmin) {
180 vFnative.InitMatrix();
181 vFnativeAtCG.InitMatrix();
183 BuildStabilityTransformMatrices();
185 for (axis_ctr = 0; axis_ctr < 3; ++axis_ctr) {
186 AeroFunctionArray::iterator f;
188 AeroFunctionArray* array = &AeroFunctions[axis_ctr];
189 for (f=array->begin(); f != array->end(); ++f) {
194 (*f)->cacheValue(
true);
195 vFnative(axis_ctr+1) += (*f)->GetValue();
198 array = &AeroFunctionsAtCG[axis_ctr];
199 for (f=array->begin(); f != array->end(); ++f) {
200 (*f)->cacheValue(
true);
201 vFnativeAtCG(axis_ctr+1) += (*f)->GetValue();
205 switch (forceAxisType) {
208 vForcesAtCG = vFnativeAtCG;
211 vFnative(eDrag)*=-1; vFnative(eLift)*=-1;
212 vForces = in.Tw2b*vFnative;
214 vFnativeAtCG(eDrag)*=-1; vFnativeAtCG(eLift)*=-1;
215 vForcesAtCG = in.Tw2b*vFnativeAtCG;
217 case atBodyAxialNormal:
218 vFnative(eX)*=-1; vFnative(eZ)*=-1;
221 vFnativeAtCG(eX)*=-1; vFnativeAtCG(eZ)*=-1;
222 vForcesAtCG = vFnativeAtCG;
225 vFnative(eDrag) *= -1; vFnative(eLift) *= -1;
226 vForces = Ts2b*vFnative;
228 vFnativeAtCG(eDrag) *= -1; vFnativeAtCG(eLift) *= -1;
229 vForcesAtCG = Ts2b*vFnativeAtCG;
234 s <<
" A proper axis type has NOT been selected. Check "
235 <<
"your aerodynamics definition.";
236 cerr << endl << s.str() << endl;
245 if (AeroRPShift) vDeltaRP(eX) = AeroRPShift->
GetValue()*in.Wingchord;
247 vDXYZcg(eX) = in.RPBody(eX) - vDeltaRP(eX);
248 vDXYZcg(eY) = in.RPBody(eY) + vDeltaRP(eY);
249 vDXYZcg(eZ) = in.RPBody(eZ) - vDeltaRP(eZ);
251 vMomentsMRC.InitMatrix();
253 for (axis_ctr = 0; axis_ctr < 3; axis_ctr++) {
254 AeroFunctionArray* array = &AeroFunctions[axis_ctr+3];
255 for (AeroFunctionArray::iterator f=array->begin(); f != array->end(); ++f) {
260 (*f)->cacheValue(
true);
261 vMomentsMRC(axis_ctr+1) += (*f)->GetValue();
267 vMomentsMRCBodyXYZ.InitMatrix();
268 switch (momentAxisType) {
270 vMomentsMRCBodyXYZ = vMomentsMRC;
273 vMomentsMRCBodyXYZ = Ts2b*vMomentsMRC;
276 vMomentsMRCBodyXYZ = in.Tw2b*vMomentsMRC;
281 s <<
" A proper axis type has NOT been selected. Check "
282 <<
"your aerodynamics definition.";
283 cerr << endl << s.str() << endl;
288 vMoments = vMomentsMRCBodyXYZ + vDXYZcg*vForces;
292 vForces += vForcesAtCG;
306 vFw = in.Tb2w * vForces;
307 vFw(eDrag) *= -1; vFw(eLift) *= -1;
310 if ( fabs(vFw(eDrag)) > 0.0)
311 lod = fabs( vFw(eLift) / vFw(eDrag));
324 vFs(eDrag) *= -1; vFs(eLift) *= -1;
334 string scratch_unit=
"";
335 Element *temp_element, *axis_element, *function_element;
343 DetermineAxisSystem(document);
347 if ((temp_element = document->
FindElement(
"alphalimits"))) {
349 if (scratch_unit.empty()) scratch_unit =
"RAD";
352 alphaclmin = alphaclmin0;
353 alphaclmax = alphaclmax0;
356 if ((temp_element = document->
FindElement(
"hysteresis_limits"))) {
358 if (scratch_unit.empty()) scratch_unit =
"RAD";
363 if ((temp_element = document->
FindElement(
"aero_ref_pt_shift_x"))) {
364 function_element = temp_element->
FindElement(
"function");
365 AeroRPShift =
new FGFunction(FDMExec, function_element);
369 while (axis_element) {
370 AeroFunctionArray ca;
371 AeroFunctionArray ca_atCG;
373 function_element = axis_element->
FindElement(
"function");
374 while (function_element) {
378 ca_atCG.push_back(
new FGFunction(FDMExec, function_element));
380 ca.push_back(
new FGFunction(FDMExec, function_element));
383 cerr << endl << axis_element->
ReadFrom()
384 << endl <<
fgred <<
"Error loading aerodynamic function in "
385 << current_func_name <<
":" << e.what() <<
" Aborting." <<
reset << endl;
390 AeroFunctions[AxisIdx[axis]] = ca;
391 AeroFunctionsAtCG[AxisIdx[axis]] = ca_atCG;
395 PostLoad(document, FDMExec);
440void FGAerodynamics::DetermineAxisSystem(
Element* document)
444 while (axis_element) {
447 if (axis ==
"X" || axis ==
"Y" || axis ==
"Z") {
448 ProcessAxesNameAndFrame(forceAxisType, axis, frame, axis_element,
450 }
else if (axis ==
"ROLL" || axis ==
"PITCH" || axis ==
"YAW") {
451 ProcessAxesNameAndFrame(momentAxisType, axis, frame, axis_element,
453 }
else if (axis ==
"LIFT" || axis ==
"DRAG") {
454 if (forceAxisType == atNone) forceAxisType = atWind;
455 else if (forceAxisType != atWind) {
456 cerr << endl << axis_element->
ReadFrom()
457 << endl <<
" Mixed aerodynamic axis systems have been used in the"
458 <<
" aircraft config file. (LIFT DRAG)" << endl;
460 }
else if (axis ==
"SIDE") {
461 if (forceAxisType != atNone && forceAxisType != atWind && forceAxisType != atBodyAxialNormal) {
462 cerr << endl << axis_element->
ReadFrom()
463 << endl <<
" Mixed aerodynamic axis systems have been used in the"
464 <<
" aircraft config file. (SIDE)" << endl;
466 }
else if (axis ==
"AXIAL" || axis ==
"NORMAL") {
467 if (forceAxisType == atNone) forceAxisType = atBodyAxialNormal;
468 else if (forceAxisType != atBodyAxialNormal) {
469 cerr << endl << axis_element->
ReadFrom()
470 << endl <<
" Mixed aerodynamic axis systems have been used in the"
471 <<
" aircraft config file. (NORMAL AXIAL)" << endl;
476 << endl <<
" An unknown axis type, " << axis <<
" has been specified"
477 <<
" in the aircraft configuration file.";
478 cerr << endl << s.str() << endl;
479 throw BaseException(s.str());
484 if (forceAxisType == atNone) {
485 forceAxisType = atWind;
486 cerr << endl <<
" The aerodynamic axis system has been set by default"
487 <<
" to the Lift/Side/Drag system." << endl;
489 if (momentAxisType == atNone) {
490 momentAxisType = atBodyXYZ;
491 cerr << endl <<
" The aerodynamic moment axis system has been set by default"
492 <<
" to the bodyXYZ system." << endl;
498void FGAerodynamics::ProcessAxesNameAndFrame(eAxisType& axisType,
const string& name,
499 const string& frame, Element* el,
500 const string& validNames)
502 if (frame ==
"BODY" || frame.empty()) {
503 if (axisType == atNone) axisType = atBodyXYZ;
504 else if (axisType != atBodyXYZ)
505 cerr << endl << el->ReadFrom()
506 << endl <<
" Mixed aerodynamic axis systems have been used in the "
507 <<
" aircraft config file." << validNames <<
" - BODY" << endl;
509 else if (frame ==
"STABILITY") {
510 if (axisType == atNone) axisType = atStability;
511 else if (axisType != atStability)
512 cerr << endl << el->ReadFrom()
513 << endl <<
" Mixed aerodynamic axis systems have been used in the "
514 <<
" aircraft config file." << validNames <<
" - STABILITY" << endl;
516 else if (frame ==
"WIND") {
517 if (axisType == atNone) axisType = atWind;
518 else if (axisType != atWind)
519 cerr << endl << el->ReadFrom()
520 << endl <<
" Mixed aerodynamic axis systems have been used in the "
521 <<
" aircraft config file." << validNames <<
" - WIND" << endl;
525 s <<
" Unknown axis frame type of - " << frame;
526 cerr << endl << s.str() << endl;
527 throw BaseException(s.str());
535 string AeroFunctionStrings =
"";
536 bool firstime =
true;
537 unsigned int axis, sd;
539 for (axis = 0; axis < 6; axis++) {
540 for (sd = 0; sd < AeroFunctions[axis].size(); sd++) {
544 AeroFunctionStrings += delimeter;
546 AeroFunctionStrings += AeroFunctions[axis][sd]->GetName();
552 if (!FunctionStrings.empty()) {
553 if (!AeroFunctionStrings.empty()) {
554 AeroFunctionStrings += delimeter + FunctionStrings;
556 AeroFunctionStrings = FunctionStrings;
560 return AeroFunctionStrings;
569 for (
unsigned int axis = 0; axis < 6; axis++) {
570 for (
unsigned int sd = 0; sd < AeroFunctions[axis].size(); sd++) {
571 if (buf.tellp() > 0) buf << delimeter;
572 buf << AeroFunctions[axis][sd]->GetValue();
578 if (!FunctionValues.empty()) {
579 if (!buf.str().empty()) {
580 buf << delimeter << FunctionValues;
582 buf << FunctionValues;
591void FGAerodynamics::bind(
void)
615 PropertyManager->Tie(
"aero/qbar-area", &qbar_area);
616 PropertyManager->Tie(
"aero/alpha-max-rad",
this, &FGAerodynamics::GetAlphaCLMax, &FGAerodynamics::SetAlphaCLMax);
617 PropertyManager->Tie(
"aero/alpha-min-rad",
this, &FGAerodynamics::GetAlphaCLMin, &FGAerodynamics::SetAlphaCLMin);
618 PropertyManager->Tie(
"aero/bi2vel",
this, &FGAerodynamics::GetBI2Vel);
619 PropertyManager->Tie(
"aero/ci2vel",
this, &FGAerodynamics::GetCI2Vel);
620 PropertyManager->Tie(
"aero/alpha-wing-rad",
this, &FGAerodynamics::GetAlphaW);
621 PropertyManager->Tie(
"systems/stall-warn-norm",
this, &FGAerodynamics::GetStallWarn);
622 PropertyManager->Tie(
"aero/stall-hyst-norm",
this, &FGAerodynamics::GetHysteresisParm);
644void FGAerodynamics::BuildStabilityTransformMatrices(
void)
646 double ca = cos(in.Alpha);
647 double sa = sin(in.Alpha);
682void FGAerodynamics::Debug(
int from)
684 if (debug_lvl <= 0)
return;
688 switch (forceAxisType) {
690 cout << endl <<
" Aerodynamics (Lift|Side|Drag axes):" << endl << endl;
692 case (atBodyAxialNormal):
693 cout << endl <<
" Aerodynamics (Axial|Side|Normal axes):" << endl << endl;
696 cout << endl <<
" Aerodynamics (Body X|Y|Z axes):" << endl << endl;
699 cout << endl <<
" Aerodynamics (Stability X|Y|Z axes):" << endl << endl;
702 cout << endl <<
" Aerodynamics (undefined axes):" << endl << endl;
707 if (debug_lvl & 2 ) {
708 if (from == 0) cout <<
"Instantiated: FGAerodynamics" << endl;
709 if (from == 1) cout <<
"Destroyed: FGAerodynamics" << endl;
711 if (debug_lvl & 4 ) {
713 if (debug_lvl & 8 ) {
715 if (debug_lvl & 16) {
717 if (debug_lvl & 64) {
Element * FindElement(const std::string &el="")
Searches for a specified element.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
std::string ReadFrom(void) const
Return a string that contains a description of the location where the current XML element was read fr...
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
bool HasAttribute(const std::string &key)
Determines if an element has the supplied attribute.
double FindElementValueAsNumberConvertFromTo(const std::string &el, const std::string &supplied_units, const std::string &target_units)
Searches for the named element and converts and returns the data belonging to it.
Encapsulates the aerodynamic calculations.
std::string GetAeroFunctionStrings(const std::string &delimeter) const
Gets the strings for the current set of aero functions.
const FGColumnVector3 & GetMoments(void) const
Gets the total aerodynamic moment vector about the CG.
bool Load(Element *element) override
Loads the Aerodynamics model.
FGColumnVector3 GetForcesInStabilityAxes(void) const
Retrieves the aerodynamic forces in the stability axes.
FGColumnVector3 GetMomentsInWindAxes(void) const
Gets the total aerodynamic moment vector about the CG in the wind axes.
~FGAerodynamics() override
Destructor.
FGColumnVector3 GetMomentsInStabilityAxes(void) const
Gets the total aerodynamic moment vector about the CG in the stability axes.
double GetLoD(void) const
Retrieves the lift over drag ratio.
FGAerodynamics(FGFDMExec *Executive)
Constructor.
const FGColumnVector3 & GetvFw(void) const
Retrieves the aerodynamic forces in the wind axes.
bool Run(bool Holding) override
Runs the Aerodynamics model; called by the Executive Can pass in a value indicating if the executive ...
std::string GetAeroFunctionValues(const std::string &delimeter) const
Gets the aero function values.
double GetClSquared(void) const
Retrieves the square of the lift coefficient.
const FGColumnVector3 & GetForces(void) const
Gets the total aerodynamic force vector.
This class implements a 3 element column vector.
Encapsulates the JSBSim simulation executive.
Represents a mathematical function.
double GetValue(void) const override
Retrieves the value of the function object.
static char fgred[6]
red text
static char reset[5]
resets text properties
FGMatrix33 Transposed(void) const
Transposed matrix.
std::string GetFunctionStrings(const std::string &delimeter) const
Gets the strings for the current set of functions.
std::string GetFunctionValues(const std::string &delimeter) const
Gets the function values.
Base class for all scheduled JSBSim models.
virtual bool Run(bool Holding)
Runs the model; called by the Executive.
bool Upload(Element *el, bool preLoad)
Uploads this model in memory.