JSBSim Flight Dynamics Model 1.2.2 (22 Mar 2025)
An Open Source Flight Dynamics and Control Software Library in C++
Loading...
Searching...
No Matches
FGAerodynamics.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGAerodynamics.cpp
4 Author: Jon S. Berndt
5 Date started: 09/13/00
6 Purpose: Encapsulates the aerodynamic forces
7
8 ------------- Copyright (C) 2000 Jon S. Berndt (jon@jsbsim.org) -------------
9
10 This program is free software; you can redistribute it and/or modify it under
11 the terms of the GNU Lesser General Public License as published by the Free
12 Software Foundation; either version 2 of the License, or (at your option) any
13 later version.
14
15 This program is distributed in the hope that it will be useful, but WITHOUT
16 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
17 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
18 details.
19
20 You should have received a copy of the GNU Lesser General Public License along
21 with this program; if not, write to the Free Software Foundation, Inc., 59
22 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23
24 Further information about the GNU Lesser General Public License can also be
25 found on the world wide web at http://www.gnu.org.
26
27FUNCTIONAL DESCRIPTION
28--------------------------------------------------------------------------------
29
30HISTORY
31--------------------------------------------------------------------------------
3209/13/00 JSB Created
3304/22/01 JSB Moved code into here from FGAircraft
34
35%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
36INCLUDES
37%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
38
39#include "FGAerodynamics.h"
40#include "input_output/FGXMLElement.h"
41
42using namespace std;
43
44namespace JSBSim {
45
46/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
47CLASS IMPLEMENTATION
48%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
49
50
52{
53 Name = "FGAerodynamics";
54
55 AxisIdx["DRAG"] = 0;
56 AxisIdx["SIDE"] = 1;
57 AxisIdx["LIFT"] = 2;
58 AxisIdx["ROLL"] = 3;
59 AxisIdx["PITCH"] = 4;
60 AxisIdx["YAW"] = 5;
61
62 AxisIdx["AXIAL"] = 0;
63 AxisIdx["NORMAL"] = 2;
64
65 AxisIdx["X"] = 0;
66 AxisIdx["Y"] = 1;
67 AxisIdx["Z"] = 2;
68
69 forceAxisType = atNone;
70 momentAxisType = atNone;
71
72 AeroFunctions = new AeroFunctionArray[6];
73 AeroFunctionsAtCG = new AeroFunctionArray[6];
74
75 impending_stall = stall_hyst = 0.0;
76 alphaclmin = alphaclmax = 0.0;
77 alphaclmin0 = alphaclmax0 = 0.0;
78 alphahystmin = alphahystmax = 0.0;
79 clsq = lod = 0.0;
80 alphaw = 0.0;
81 bi2vel = ci2vel = 0.0;
82 AeroRPShift = 0;
83 vDeltaRP.InitMatrix();
84
85 bind();
86
87 Debug(0);
88}
89
90//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
91
93{
94 unsigned int i,j;
95
96 for (i=0; i<6; i++)
97 for (j=0; j<AeroFunctions[i].size(); j++)
98 delete AeroFunctions[i][j];
99 for (i=0; i<6; i++)
100 for (j=0; j<AeroFunctionsAtCG[i].size(); j++)
101 delete AeroFunctionsAtCG[i][j];
102
103 delete[] AeroFunctions;
104 delete[] AeroFunctionsAtCG;
105
106 delete AeroRPShift;
107
108 Debug(1);
109}
110
111//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
112
113bool FGAerodynamics::InitModel(void)
114{
115 if (!FGModel::InitModel()) return false;
116
117 impending_stall = stall_hyst = 0.0;
118 alphaclmin = alphaclmin0;
119 alphaclmax = alphaclmax0;
120 alphahystmin = alphahystmax = 0.0;
121 clsq = lod = 0.0;
122 alphaw = 0.0;
123 bi2vel = ci2vel = 0.0;
124 AeroRPShift = 0;
125 vDeltaRP.InitMatrix();
126 vForces.InitMatrix();
127 vMoments.InitMatrix();
128 return true;
129}
130//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
131
132bool FGAerodynamics::Run(bool Holding)
133{
134
135 if (FGModel::Run(Holding)) return true;
136 if (Holding) return false; // if paused don't execute
137
138 unsigned int axis_ctr;
139 const double twovel=2*in.Vt;
140
141 // The lift coefficient squared (property aero/cl-squared) is computed before
142 // the aero functions are called to make sure that they use the same value for
143 // qbar.
144 if ( in.Qbar > 1.0) {
145 // Skip the computation if qbar is close to zero to avoid huge values for
146 // aero/cl-squared when a non-null lift coincides with a very small aero
147 // velocity (i.e. when qbar is close to zero).
148 clsq = vFw(eLift) / (in.Wingarea*in.Qbar);
149 clsq *= clsq;
150 }
151
152 RunPreFunctions();
153
154 // calculate some oft-used quantities for speed
155
156 if (twovel != 0) {
157 bi2vel = in.Wingspan / twovel;
158 ci2vel = in.Wingchord / twovel;
159 }
160 alphaw = in.Alpha + in.Wingincidence;
161 qbar_area = in.Wingarea * in.Qbar;
162
163 if (alphaclmax != 0) {
164 if (in.Alpha > 0.85*alphaclmax) {
165 impending_stall = 10*(in.Alpha/alphaclmax - 0.85);
166 } else {
167 impending_stall = 0;
168 }
169 }
170
171 if (alphahystmax != 0.0 && alphahystmin != 0.0) {
172 if (in.Alpha > alphahystmax) {
173 stall_hyst = 1;
174 } else if (in.Alpha < alphahystmin) {
175 stall_hyst = 0;
176 }
177 }
178
179 vFw.InitMatrix();
180 vFnative.InitMatrix();
181 vFnativeAtCG.InitMatrix();
182
183 BuildStabilityTransformMatrices();
184
185 for (axis_ctr = 0; axis_ctr < 3; ++axis_ctr) {
186 AeroFunctionArray::iterator f;
187
188 AeroFunctionArray* array = &AeroFunctions[axis_ctr];
189 for (f=array->begin(); f != array->end(); ++f) {
190 // Tell the Functions to cache values, so when the function values are
191 // being requested for output, the functions do not get calculated again
192 // in a context that might have changed, but instead use the values that
193 // have already been calculated for this frame.
194 (*f)->cacheValue(true);
195 vFnative(axis_ctr+1) += (*f)->GetValue();
196 }
197
198 array = &AeroFunctionsAtCG[axis_ctr];
199 for (f=array->begin(); f != array->end(); ++f) {
200 (*f)->cacheValue(true); // Same as above
201 vFnativeAtCG(axis_ctr+1) += (*f)->GetValue();
202 }
203 }
204
205 switch (forceAxisType) {
206 case atBodyXYZ: // Forces already in body axes; no manipulation needed
207 vForces = vFnative;
208 vForcesAtCG = vFnativeAtCG;
209 break;
210 case atWind: // Copy forces into wind axes
211 vFnative(eDrag)*=-1; vFnative(eLift)*=-1;
212 vForces = in.Tw2b*vFnative;
213
214 vFnativeAtCG(eDrag)*=-1; vFnativeAtCG(eLift)*=-1;
215 vForcesAtCG = in.Tw2b*vFnativeAtCG;
216 break;
217 case atBodyAxialNormal: // Convert native forces into Axial|Normal|Side system
218 vFnative(eX)*=-1; vFnative(eZ)*=-1;
219 vForces = vFnative;
220
221 vFnativeAtCG(eX)*=-1; vFnativeAtCG(eZ)*=-1;
222 vForcesAtCG = vFnativeAtCG;
223 break;
224 case atStability: // Convert from stability axes to both body and wind axes
225 vFnative(eDrag) *= -1; vFnative(eLift) *= -1;
226 vForces = Ts2b*vFnative;
227
228 vFnativeAtCG(eDrag) *= -1; vFnativeAtCG(eLift) *= -1;
229 vForcesAtCG = Ts2b*vFnativeAtCG;
230 break;
231 default:
232 {
233 stringstream s;
234 s << " A proper axis type has NOT been selected. Check "
235 << "your aerodynamics definition.";
236 cerr << endl << s.str() << endl;
237 throw BaseException(s.str());
238 }
239 }
240 // Calculate aerodynamic reference point shift, if any. The shift takes place
241 // in the structual axis. That is, if the shift is positive, it is towards the
242 // back (tail) of the vehicle. The AeroRPShift function should be
243 // non-dimensionalized by the wing chord. The calculated vDeltaRP will be in
244 // feet.
245 if (AeroRPShift) vDeltaRP(eX) = AeroRPShift->GetValue()*in.Wingchord;
246
247 vDXYZcg(eX) = in.RPBody(eX) - vDeltaRP(eX); // vDeltaRP is given in the
248 vDXYZcg(eY) = in.RPBody(eY) + vDeltaRP(eY); // structural frame.
249 vDXYZcg(eZ) = in.RPBody(eZ) - vDeltaRP(eZ);
250
251 vMomentsMRC.InitMatrix();
252
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) {
256 // Tell the Functions to cache values, so when the function values are
257 // being requested for output, the functions do not get calculated again
258 // in a context that might have changed, but instead use the values that
259 // have already been calculated for this frame.
260 (*f)->cacheValue(true);
261 vMomentsMRC(axis_ctr+1) += (*f)->GetValue();
262 }
263 }
264
265 // Transform moments to bodyXYZ if the moments are specified in stability or
266 // wind axes
267 vMomentsMRCBodyXYZ.InitMatrix();
268 switch (momentAxisType) {
269 case atBodyXYZ:
270 vMomentsMRCBodyXYZ = vMomentsMRC;
271 break;
272 case atStability:
273 vMomentsMRCBodyXYZ = Ts2b*vMomentsMRC;
274 break;
275 case atWind:
276 vMomentsMRCBodyXYZ = in.Tw2b*vMomentsMRC;
277 break;
278 default:
279 {
280 stringstream s;
281 s << " A proper axis type has NOT been selected. Check "
282 << "your aerodynamics definition.";
283 cerr << endl << s.str() << endl;
284 throw BaseException(s.str());
285 }
286 }
287
288 vMoments = vMomentsMRCBodyXYZ + vDXYZcg*vForces; // M = r X F
289
290 // Now add the "at CG" values to base forces - after the moments have been
291 // transferred.
292 vForces += vForcesAtCG;
293
294 // Note that we still need to convert to wind axes here, because it is used in
295 // the L/D calculation, and we still may want to look at Lift and Drag.
296 //
297 // JSB 4/27/12 - After use, convert wind axes to produce normal lift and drag
298 // values - not negative ones!
299 //
300 // As a clarification, JSBSim assumes that drag and lift values are defined in
301 // wind axes - BUT with a 180 rotation about the Y axis. That is, lift and
302 // drag will be positive up and aft, respectively, so that they are reported
303 // as positive numbers. However, the wind axes themselves assume that the X
304 // and Z forces are positive forward and down. Same applies to the stability
305 // axes.
306 vFw = in.Tb2w * vForces;
307 vFw(eDrag) *= -1; vFw(eLift) *= -1;
308
309 // Calculate Lift over Drag
310 if ( fabs(vFw(eDrag)) > 0.0)
311 lod = fabs( vFw(eLift) / vFw(eDrag));
312
313 RunPostFunctions();
314
315 return false;
316}
317
318//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
319
321{
322 FGColumnVector3 vFs = Tb2s*vForces;
323 // Need sign flips since drag is positive and lift is positive in stability axes
324 vFs(eDrag) *= -1; vFs(eLift) *= -1;
325
326 return vFs;
327}
328
329//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
330
332{
333 string axis;
334 string scratch_unit="";
335 Element *temp_element, *axis_element, *function_element;
336
337 Name = "Aerodynamics Model: " + document->GetAttributeValue("name");
338
339 // Perform base class Pre-Load
340 if (!FGModel::Upload(document, true))
341 return false;
342
343 DetermineAxisSystem(document); // Determine if Lift/Side/Drag, etc. is used.
344
345 Debug(2);
346
347 if ((temp_element = document->FindElement("alphalimits"))) {
348 scratch_unit = temp_element->GetAttributeValue("unit");
349 if (scratch_unit.empty()) scratch_unit = "RAD";
350 alphaclmin0 = temp_element->FindElementValueAsNumberConvertFromTo("min", scratch_unit, "RAD");
351 alphaclmax0 = temp_element->FindElementValueAsNumberConvertFromTo("max", scratch_unit, "RAD");
352 alphaclmin = alphaclmin0;
353 alphaclmax = alphaclmax0;
354 }
355
356 if ((temp_element = document->FindElement("hysteresis_limits"))) {
357 scratch_unit = temp_element->GetAttributeValue("unit");
358 if (scratch_unit.empty()) scratch_unit = "RAD";
359 alphahystmin = temp_element->FindElementValueAsNumberConvertFromTo("min", scratch_unit, "RAD");
360 alphahystmax = temp_element->FindElementValueAsNumberConvertFromTo("max", scratch_unit, "RAD");
361 }
362
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);
366 }
367
368 axis_element = document->FindElement("axis");
369 while (axis_element) {
370 AeroFunctionArray ca;
371 AeroFunctionArray ca_atCG;
372 axis = axis_element->GetAttributeValue("name");
373 function_element = axis_element->FindElement("function");
374 while (function_element) {
375 try {
376 if (function_element->HasAttribute("apply_at_cg") &&
377 function_element->GetAttributeValue("apply_at_cg") == "true")
378 ca_atCG.push_back(new FGFunction(FDMExec, function_element));
379 else
380 ca.push_back(new FGFunction(FDMExec, function_element));
381 } catch (BaseException& e) {
382 string current_func_name = function_element->GetAttributeValue("name");
383 cerr << endl << axis_element->ReadFrom()
384 << endl << fgred << "Error loading aerodynamic function in "
385 << current_func_name << ":" << e.what() << " Aborting." << reset << endl;
386 return false;
387 }
388 function_element = axis_element->FindNextElement("function");
389 }
390 AeroFunctions[AxisIdx[axis]] = ca;
391 AeroFunctionsAtCG[AxisIdx[axis]] = ca_atCG;
392 axis_element = document->FindNextElement("axis");
393 }
394
395 PostLoad(document, FDMExec); // Perform base class Post-Load
396
397 return true;
398}
399
400//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
401//
402// This private class function checks to verify consistency in the choice of
403// aerodynamic axes used in the config file. One set of LIFT|DRAG|SIDE, or
404// X|Y|Z, or AXIAL|NORMAL|SIDE must be chosen; mixed system axes are not allowed.
405// Note that if the "SIDE" axis specifier is entered first in a config file,
406// a warning message will be given IF the AXIAL|NORMAL specifiers are also given.
407// This is OK, and the warning is due to the SIDE specifier used for both
408// the Lift/Drag and Axial/Normal axis systems.
409// Alternatively the axis name 'X|Y|Z or ROLL|PITCH|YAW' can be specified in
410// conjunction with a frame 'BODY|STABILITY|WIND', for example:
411// <axis name="X" frame="STABILITY"/>
412//
413// In summary, possible combinations:
414//
415// FORCES
416// Body
417// <axis name="AXIAL|SIDE|NORMAL" />
418// <axis name="X|Y|Z" />
419// <axis name="X|Y|Z" frame="BODY" />
420//
421// Wind
422// <axis name="DRAG|SIDE|LIFT" / >
423// <axis name="X|Y|Z" frame="WIND" / >
424//
425// Stability
426// <axis name="X|Y|Z" frame="STABILITY" />
427//
428// MOMENTS
429// Body
430// <axis name="ROLL|PITCH|YAW" />
431// <axis name="ROLL|PITCH|YAW" frame="BODY" / >
432//
433// Wind
434// <axis name="ROLL|PITCH|YAW" frame="WIND" />
435//
436// Stability
437// <axis name="ROLL|PITCH|YAW" frame="STABILITY" />
438//
439
440void FGAerodynamics::DetermineAxisSystem(Element* document)
441{
442 Element* axis_element = document->FindElement("axis");
443 string axis;
444 while (axis_element) {
445 axis = axis_element->GetAttributeValue("name");
446 string frame = axis_element->GetAttributeValue("frame");
447 if (axis == "X" || axis == "Y" || axis == "Z") {
448 ProcessAxesNameAndFrame(forceAxisType, axis, frame, axis_element,
449 "(X Y Z)");
450 } else if (axis == "ROLL" || axis == "PITCH" || axis == "YAW") {
451 ProcessAxesNameAndFrame(momentAxisType, axis, frame, axis_element,
452 "(ROLL PITCH YAW)");
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;
459 }
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;
465 }
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;
472 }
473 } else { // error
474 stringstream s;
475 s << axis_element->ReadFrom()
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());
480 }
481 axis_element = document->FindNextElement("axis");
482 }
483
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;
488 }
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;
493 }
494}
495
496//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
497
498void FGAerodynamics::ProcessAxesNameAndFrame(eAxisType& axisType, const string& name,
499 const string& frame, Element* el,
500 const string& validNames)
501{
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;
508 }
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;
515 }
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;
522 }
523 else {
524 stringstream s;
525 s << " Unknown axis frame type of - " << frame;
526 cerr << endl << s.str() << endl;
527 throw BaseException(s.str());
528 }
529}
530
531//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
532
533string FGAerodynamics::GetAeroFunctionStrings(const string& delimeter) const
534{
535 string AeroFunctionStrings = "";
536 bool firstime = true;
537 unsigned int axis, sd;
538
539 for (axis = 0; axis < 6; axis++) {
540 for (sd = 0; sd < AeroFunctions[axis].size(); sd++) {
541 if (firstime) {
542 firstime = false;
543 } else {
544 AeroFunctionStrings += delimeter;
545 }
546 AeroFunctionStrings += AeroFunctions[axis][sd]->GetName();
547 }
548 }
549
550 string FunctionStrings = FGModelFunctions::GetFunctionStrings(delimeter);
551
552 if (!FunctionStrings.empty()) {
553 if (!AeroFunctionStrings.empty()) {
554 AeroFunctionStrings += delimeter + FunctionStrings;
555 } else {
556 AeroFunctionStrings = FunctionStrings;
557 }
558 }
559
560 return AeroFunctionStrings;
561}
562
563//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
564
565string FGAerodynamics::GetAeroFunctionValues(const string& delimeter) const
566{
567 ostringstream buf;
568
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();
573 }
574 }
575
576 string FunctionValues = FGModelFunctions::GetFunctionValues(delimeter);
577
578 if (!FunctionValues.empty()) {
579 if (!buf.str().empty()) {
580 buf << delimeter << FunctionValues;
581 } else {
582 buf << FunctionValues;
583 }
584 }
585
586 return buf.str();
587}
588
589//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
590
591void FGAerodynamics::bind(void)
592{
593 typedef double (FGAerodynamics::*PMF)(int) const;
594
595 PropertyManager->Tie("forces/fbx-aero-lbs", this, eX, (PMF)&FGAerodynamics::GetForces);
596 PropertyManager->Tie("forces/fby-aero-lbs", this, eY, (PMF)&FGAerodynamics::GetForces);
597 PropertyManager->Tie("forces/fbz-aero-lbs", this, eZ, (PMF)&FGAerodynamics::GetForces);
598 PropertyManager->Tie("moments/l-aero-lbsft", this, eL, (PMF)&FGAerodynamics::GetMoments);
599 PropertyManager->Tie("moments/m-aero-lbsft", this, eM, (PMF)&FGAerodynamics::GetMoments);
600 PropertyManager->Tie("moments/n-aero-lbsft", this, eN, (PMF)&FGAerodynamics::GetMoments);
601 PropertyManager->Tie("forces/fwx-aero-lbs", this, eDrag, (PMF)&FGAerodynamics::GetvFw);
602 PropertyManager->Tie("forces/fwy-aero-lbs", this, eSide, (PMF)&FGAerodynamics::GetvFw);
603 PropertyManager->Tie("forces/fwz-aero-lbs", this, eLift, (PMF)&FGAerodynamics::GetvFw);
604 PropertyManager->Tie("forces/fsx-aero-lbs", this, eX, (PMF)&FGAerodynamics::GetForcesInStabilityAxes);
605 PropertyManager->Tie("forces/fsy-aero-lbs", this, eY, (PMF)&FGAerodynamics::GetForcesInStabilityAxes);
606 PropertyManager->Tie("forces/fsz-aero-lbs", this, eZ, (PMF)&FGAerodynamics::GetForcesInStabilityAxes);
607 PropertyManager->Tie("moments/roll-stab-aero-lbsft", this, eRoll, (PMF)&FGAerodynamics::GetMomentsInStabilityAxes);
608 PropertyManager->Tie("moments/pitch-stab-aero-lbsft", this, ePitch, (PMF)&FGAerodynamics::GetMomentsInStabilityAxes);
609 PropertyManager->Tie("moments/yaw-stab-aero-lbsft", this, eYaw, (PMF)&FGAerodynamics::GetMomentsInStabilityAxes);
610 PropertyManager->Tie("moments/roll-wind-aero-lbsft", this, eRoll, (PMF)&FGAerodynamics::GetMomentsInWindAxes);
611 PropertyManager->Tie("moments/pitch-wind-aero-lbsft", this, ePitch, (PMF)&FGAerodynamics::GetMomentsInWindAxes);
612 PropertyManager->Tie("moments/yaw-wind-aero-lbsft", this, eYaw, (PMF)&FGAerodynamics::GetMomentsInWindAxes);
613 PropertyManager->Tie("forces/lod-norm", this, &FGAerodynamics::GetLoD);
614 PropertyManager->Tie("aero/cl-squared", this, &FGAerodynamics::GetClSquared);
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);
623}
624
625//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
626//
627// Build transformation matrices for transforming from stability axes to
628// body axes and to wind axes. Where "a" is alpha and "B" is beta:
629//
630// The transform from body to stability axes is:
631//
632// cos(a) 0 sin(a)
633// 0 1 0
634// -sin(a) 0 cos(a)
635//
636// The transform from stability to body axes is:
637//
638// cos(a) 0 -sin(a)
639// 0 1 0
640// sin(a) 0 cos(a)
641//
642//
643
644void FGAerodynamics::BuildStabilityTransformMatrices(void)
645{
646 double ca = cos(in.Alpha);
647 double sa = sin(in.Alpha);
648
649 // Stability-to-body
650 Ts2b(1, 1) = ca;
651 Ts2b(1, 2) = 0.0;
652 Ts2b(1, 3) = -sa;
653 Ts2b(2, 1) = 0.0;
654 Ts2b(2, 2) = 1.0;
655 Ts2b(2, 3) = 0.0;
656 Ts2b(3, 1) = sa;
657 Ts2b(3, 2) = 0.0;
658 Ts2b(3, 3) = ca;
659
660 Tb2s = Ts2b.Transposed();
661}
662
663//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
664// The bitmasked value choices are as follows:
665// unset: In this case (the default) JSBSim would only print
666// out the normally expected messages, essentially echoing
667// the config files as they are read. If the environment
668// variable is not set, debug_lvl is set to 1 internally
669// 0: This requests JSBSim not to output any messages
670// whatsoever.
671// 1: This value explicity requests the normal JSBSim
672// startup messages
673// 2: This value asks for a message to be printed out when
674// a class is instantiated
675// 4: When this value is set, a message is displayed when a
676// FGModel object executes its Run() method
677// 8: When this value is set, various runtime state variables
678// are printed out periodically
679// 16: When set various parameters are sanity checked and
680// a message is printed out when they go out of bounds
681
682void FGAerodynamics::Debug(int from)
683{
684 if (debug_lvl <= 0) return;
685
686 if (debug_lvl & 1) { // Standard console startup message output
687 if (from == 2) { // Loader
688 switch (forceAxisType) {
689 case (atWind):
690 cout << endl << " Aerodynamics (Lift|Side|Drag axes):" << endl << endl;
691 break;
692 case (atBodyAxialNormal):
693 cout << endl << " Aerodynamics (Axial|Side|Normal axes):" << endl << endl;
694 break;
695 case (atBodyXYZ):
696 cout << endl << " Aerodynamics (Body X|Y|Z axes):" << endl << endl;
697 break;
698 case (atStability):
699 cout << endl << " Aerodynamics (Stability X|Y|Z axes):" << endl << endl;
700 break;
701 case (atNone):
702 cout << endl << " Aerodynamics (undefined axes):" << endl << endl;
703 break;
704 }
705 }
706 }
707 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
708 if (from == 0) cout << "Instantiated: FGAerodynamics" << endl;
709 if (from == 1) cout << "Destroyed: FGAerodynamics" << endl;
710 }
711 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
712 }
713 if (debug_lvl & 8 ) { // Runtime state variables
714 }
715 if (debug_lvl & 16) { // Sanity checking
716 }
717 if (debug_lvl & 64) {
718 if (from == 0) { // Constructor
719 }
720 }
721}
722
723} // namespace JSBSim
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.
Definition FGFDMExec.h:184
Represents a mathematical function.
Definition FGFunction.h:765
double GetValue(void) const override
Retrieves the value of the function object.
static char fgred[6]
red text
Definition FGJSBBase.h:166
static char reset[5]
resets text properties
Definition FGJSBBase.h:156
FGMatrix33 Transposed(void) const
Transposed matrix.
Definition FGMatrix33.h:221
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.
Definition FGModel.h:70
virtual bool Run(bool Holding)
Runs the model; called by the Executive.
Definition FGModel.cpp:89
bool Upload(Element *el, bool preLoad)
Uploads this model in memory.
Definition FGModel.cpp:110