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
FGBallonet Class Reference

Detailed Description

Models a ballonet inside a gas cell.

Not intended to be used outside FGGasCell. See FGGasCell for the configuration file format.

Author
Anders Gidenstam

Definition at line 302 of file FGGasCell.h.

#include <FGGasCell.h>

+ Inheritance diagram for FGBallonet:
+ Collaboration diagram for FGBallonet:

Public Member Functions

 FGBallonet (FGFDMExec *exec, Element *el, unsigned int num, FGGasCell *parent, const struct FGGasCell::Inputs &input)
 
void Calculate (double dt)
 Runs the ballonet model; called by FGGasCell.
 
double GetHeatFlow (void) const
 Get the current heat flow into the ballonet.
 
const FGMatrix33GetInertia (void) const
 Get the moments of inertia of the ballonet.
 
double GetMass (void) const
 Get the current mass of the ballonets.
 
double GetVolume (void) const
 Get the current volume of the ballonet.
 
double GetXYZ (int idx) const
 Get the center of gravity location of the ballonet.
 
const FGColumnVector3GetXYZ (void) const
 Get the center of gravity location of the ballonet.
 
- Public Member Functions inherited from FGJSBBase
 FGJSBBase ()
 Constructor for FGJSBBase.
 
virtual ~FGJSBBase ()
 Destructor for FGJSBBase.
 
void disableHighLighting (void)
 Disables highlighting in the console output.
 

Public Attributes

const struct FGGasCell::Inputsin
 

Additional Inherited Members

- Public Types inherited from FGJSBBase
enum  { eL = 1 , eM , eN }
 Moments L, M, N. More...
 
enum  { eP = 1 , eQ , eR }
 Rates P, Q, R. More...
 
enum  { eU = 1 , eV , eW }
 Velocities U, V, W. More...
 
enum  { eX = 1 , eY , eZ }
 Positions X, Y, Z. More...
 
enum  { ePhi = 1 , eTht , ePsi }
 Euler angles Phi, Theta, Psi. More...
 
enum  { eDrag = 1 , eSide , eLift }
 Stability axis forces, Drag, Side force, Lift. More...
 
enum  { eRoll = 1 , ePitch , eYaw }
 Local frame orientation Roll, Pitch, Yaw. More...
 
enum  { eNorth = 1 , eEast , eDown }
 Local frame position North, East, Down. More...
 
enum  { eLat = 1 , eLong , eRad }
 Locations Radius, Latitude, Longitude. More...
 
enum  {
  inNone = 0 , inDegrees , inRadians , inMeters ,
  inFeet
}
 Conversion specifiers. More...
 
- Static Public Member Functions inherited from FGJSBBase
static const std::string & GetVersion (void)
 Returns the version number of JSBSim.
 
static constexpr double KelvinToFahrenheit (double kelvin)
 Converts from degrees Kelvin to degrees Fahrenheit.
 
static constexpr double CelsiusToRankine (double celsius)
 Converts from degrees Celsius to degrees Rankine.
 
static constexpr double RankineToCelsius (double rankine)
 Converts from degrees Rankine to degrees Celsius.
 
static constexpr double KelvinToRankine (double kelvin)
 Converts from degrees Kelvin to degrees Rankine.
 
static constexpr double RankineToKelvin (double rankine)
 Converts from degrees Rankine to degrees Kelvin.
 
static constexpr double FahrenheitToCelsius (double fahrenheit)
 Converts from degrees Fahrenheit to degrees Celsius.
 
static constexpr double CelsiusToFahrenheit (double celsius)
 Converts from degrees Celsius to degrees Fahrenheit.
 
static constexpr double CelsiusToKelvin (double celsius)
 Converts from degrees Celsius to degrees Kelvin.
 
static constexpr double KelvinToCelsius (double kelvin)
 Converts from degrees Kelvin to degrees Celsius.
 
static constexpr double FeetToMeters (double measure)
 Converts from feet to meters.
 
static bool EqualToRoundoff (double a, double b)
 Finite precision comparison.
 
static bool EqualToRoundoff (float a, float b)
 Finite precision comparison.
 
static bool EqualToRoundoff (float a, double b)
 Finite precision comparison.
 
static bool EqualToRoundoff (double a, float b)
 Finite precision comparison.
 
static constexpr double Constrain (double min, double value, double max)
 Constrain a value between a minimum and a maximum value.
 
static constexpr double sign (double num)
 
- Static Public Attributes inherited from FGJSBBase
static char highint [5] = {27, '[', '1', 'm', '\0' }
 highlights text
 
static char halfint [5] = {27, '[', '2', 'm', '\0' }
 low intensity text
 
static char normint [6] = {27, '[', '2', '2', 'm', '\0' }
 normal intensity text
 
static char reset [5] = {27, '[', '0', 'm', '\0' }
 resets text properties
 
static char underon [5] = {27, '[', '4', 'm', '\0' }
 underlines text
 
static char underoff [6] = {27, '[', '2', '4', 'm', '\0' }
 underline off
 
static char fgblue [6] = {27, '[', '3', '4', 'm', '\0' }
 blue text
 
static char fgcyan [6] = {27, '[', '3', '6', 'm', '\0' }
 cyan text
 
static char fgred [6] = {27, '[', '3', '1', 'm', '\0' }
 red text
 
static char fggreen [6] = {27, '[', '3', '2', 'm', '\0' }
 green text
 
static char fgdef [6] = {27, '[', '3', '9', 'm', '\0' }
 default text
 
static short debug_lvl = 1
 
- Static Protected Member Functions inherited from FGJSBBase
static std::string CreateIndexedPropertyName (const std::string &Property, int index)
 
- Static Protected Attributes inherited from FGJSBBase
static constexpr double radtodeg = 180. / M_PI
 
static constexpr double degtorad = M_PI / 180.
 
static constexpr double hptoftlbssec = 550.0
 
static constexpr double psftoinhg = 0.014138
 
static constexpr double psftopa = 47.88
 
static constexpr double fttom = 0.3048
 
static constexpr double ktstofps = 1852./(3600*fttom)
 
static constexpr double fpstokts = 1.0 / ktstofps
 
static constexpr double inchtoft = 1.0/12.0
 
static constexpr double m3toft3 = 1.0/(fttom*fttom*fttom)
 
static constexpr double in3tom3 = inchtoft*inchtoft*inchtoft/m3toft3
 
static constexpr double inhgtopa = 3386.38
 
static constexpr double slugtolb = 32.174049
 Note that definition of lbtoslug by the inverse of slugtolb and not to a different constant you can also get from some tables will make lbtoslug*slugtolb == 1 up to the magnitude of roundoff.
 
static constexpr double lbtoslug = 1.0/slugtolb
 
static constexpr double kgtolb = 2.20462
 
static constexpr double kgtoslug = 0.06852168
 
static const std::string needed_cfg_version = "2.0"
 
static const std::string JSBSim_version = JSBSIM_VERSION " " __DATE__ " " __TIME__
 

Constructor & Destructor Documentation

◆ FGBallonet()

FGBallonet ( FGFDMExec exec,
Element el,
unsigned int  num,
FGGasCell parent,
const struct FGGasCell::Inputs input 
)

Definition at line 499 of file FGGasCell.cpp.

501 : in(input)
502{
503 string token;
504 Element* element;
505
506 auto PropertyManager = exec->GetPropertyManager();
507 MassBalance = exec->GetMassBalance();
508
509 MaxVolume = MaxOverpressure = Temperature = Pressure =
510 Contents = Volume = dVolumeIdeal = dU = 0.0;
511 Xradius = Yradius = Zradius = Xwidth = Ywidth = Zwidth = 0.0;
512 ValveCoefficient = ValveOpen = 0.0;
513 BlowerInput = NULL;
514 CellNum = num;
515 Parent = parent;
516
517 // NOTE: In the local system X points north, Y points east and Z points down.
518 element = el->FindElement("location");
519 if (element) {
520 vXYZ = element->FindElementTripletConvertTo("IN");
521 } else {
522 const string s("Fatal Error: No location found for this ballonet.");
523 cerr << el->ReadFrom() << endl << s << endl;
524 throw BaseException(s);
525 }
526 if ((el->FindElement("x_radius") || el->FindElement("x_width")) &&
527 (el->FindElement("y_radius") || el->FindElement("y_width")) &&
528 (el->FindElement("z_radius") || el->FindElement("z_width"))) {
529
530 if (el->FindElement("x_radius")) {
531 Xradius = el->FindElementValueAsNumberConvertTo("x_radius", "FT");
532 }
533 if (el->FindElement("y_radius")) {
534 Yradius = el->FindElementValueAsNumberConvertTo("y_radius", "FT");
535 }
536 if (el->FindElement("z_radius")) {
537 Zradius = el->FindElementValueAsNumberConvertTo("z_radius", "FT");
538 }
539
540 if (el->FindElement("x_width")) {
541 Xwidth = el->FindElementValueAsNumberConvertTo("x_width", "FT");
542 }
543 if (el->FindElement("y_width")) {
544 Ywidth = el->FindElementValueAsNumberConvertTo("y_width", "FT");
545 }
546 if (el->FindElement("z_width")) {
547 Zwidth = el->FindElementValueAsNumberConvertTo("z_width", "FT");
548 }
549
550 // The volume is a (potentially) extruded ellipsoid.
551 // FIXME: However, currently only a few combinations of radius and
552 // width are fully supported.
553 if ((Xradius != 0.0) && (Yradius != 0.0) && (Zradius != 0.0) &&
554 (Xwidth == 0.0) && (Ywidth == 0.0) && (Zwidth == 0.0)) {
555 // Ellipsoid volume.
556 MaxVolume = 4.0 * M_PI * Xradius * Yradius * Zradius / 3.0;
557 } else if ((Xradius == 0.0) && (Yradius != 0.0) && (Zradius != 0.0) &&
558 (Xwidth != 0.0) && (Ywidth == 0.0) && (Zwidth == 0.0)) {
559 // Cylindrical volume.
560 MaxVolume = M_PI * Yradius * Zradius * Xwidth;
561 } else {
562 cerr << "Warning: Unsupported ballonet shape." << endl;
563 MaxVolume =
564 (4.0 * M_PI * Xradius * Yradius * Zradius / 3.0 +
565 M_PI * Yradius * Zradius * Xwidth +
566 M_PI * Xradius * Zradius * Ywidth +
567 M_PI * Xradius * Yradius * Zwidth +
568 2.0 * Xradius * Ywidth * Zwidth +
569 2.0 * Yradius * Xwidth * Zwidth +
570 2.0 * Zradius * Xwidth * Ywidth +
571 Xwidth * Ywidth * Zwidth);
572 }
573 } else {
574 const string s("Fatal Error: Ballonet shape must be given.");
575 cerr << el->ReadFrom() << endl << s << endl;
576 throw BaseException(s);
577 }
578 if (el->FindElement("max_overpressure")) {
579 MaxOverpressure = el->FindElementValueAsNumberConvertTo("max_overpressure",
580 "LBS/FT2");
581 }
582 if (el->FindElement("fullness")) {
583 const double Fullness = el->FindElementValueAsNumber("fullness");
584 if (0 <= Fullness) {
585 Volume = Fullness * MaxVolume;
586 } else {
587 cerr << "Warning: Invalid initial ballonet fullness value." << endl;
588 }
589 }
590 if (el->FindElement("valve_coefficient")) {
591 ValveCoefficient =
592 el->FindElementValueAsNumberConvertTo("valve_coefficient",
593 "FT4*SEC/SLUG");
594 ValveCoefficient = max(ValveCoefficient, 0.0);
595 }
596
597 // Initialize state
598 if (Temperature == 0.0) {
599 Temperature = Parent->GetTemperature();
600 }
601 if (Pressure == 0.0) {
602 Pressure = Parent->GetPressure();
603 }
604 if (Volume != 0.0) {
605 // Calculate initial air content.
606 Contents = Pressure * Volume / (R * Temperature);
607
608 // Clip to max allowed value.
609 const double IdealPressure = Contents * R * Temperature / MaxVolume;
610 if (IdealPressure > Pressure + MaxOverpressure) {
611 Contents = (Pressure + MaxOverpressure) * MaxVolume / (R * Temperature);
612 Pressure = Pressure + MaxOverpressure;
613 } else {
614 Pressure = max(IdealPressure, Pressure);
615 }
616 } else {
617 // Calculate initial air content.
618 Contents = Pressure * MaxVolume / (R * Temperature);
619 }
620
621 Volume = Contents * R * Temperature / Pressure;
622
623 // Bind relevant properties
624 string property_name, base_property_name;
625 base_property_name = CreateIndexedPropertyName("buoyant_forces/gas-cell", Parent->GetIndex());
626 base_property_name = CreateIndexedPropertyName(base_property_name + "/ballonet", CellNum);
627
628 property_name = base_property_name + "/max_volume-ft3";
629 PropertyManager->Tie( property_name, &MaxVolume);
630 PropertyManager->GetNode()->SetWritable( property_name, false );
631
632 property_name = base_property_name + "/temp-R";
633 PropertyManager->Tie( property_name, &Temperature);
634
635 property_name = base_property_name + "/pressure-psf";
636 PropertyManager->Tie( property_name, &Pressure);
637
638 property_name = base_property_name + "/volume-ft3";
639 PropertyManager->Tie( property_name, &Volume);
640
641 property_name = base_property_name + "/contents-mol";
642 PropertyManager->Tie( property_name, &Contents);
643
644 property_name = base_property_name + "/valve_open";
645 PropertyManager->Tie( property_name, &ValveOpen);
646
647 Debug(0);
648
649 // Read heat transfer coefficients
650 if (Element* heat = el->FindElement("heat")) {
651 Element* function_element = heat->FindElement("function");
652 while (function_element) {
653 HeatTransferCoeff.push_back(new FGFunction(exec, function_element));
654 function_element = heat->FindNextElement("function");
655 }
656 }
657 // Read blower input function
658 if (Element* blower = el->FindElement("blower_input")) {
659 Element* function_element = blower->FindElement("function");
660 BlowerInput = new FGFunction(exec, function_element);
661 }
662}
double GetTemperature(void) const
Get the current gas temperature inside the gas cell.
Definition FGGasCell.h:220
int GetIndex(void) const
Get the index of this gas cell.
Definition FGGasCell.h:190
double GetPressure(void) const
Get the current gas pressure inside the gas cell.
Definition FGGasCell.h:224

◆ ~FGBallonet()

~FGBallonet ( )

Definition at line 666 of file FGGasCell.cpp.

667{
668 unsigned int i;
669
670 for (i = 0; i < HeatTransferCoeff.size(); i++) delete HeatTransferCoeff[i];
671 HeatTransferCoeff.clear();
672
673 delete BlowerInput;
674 BlowerInput = NULL;
675
676 Debug(1);
677}

Member Function Documentation

◆ Calculate()

void Calculate ( double  dt)

Runs the ballonet model; called by FGGasCell.

Definition at line 681 of file FGGasCell.cpp.

682{
683 const double ParentPressure = Parent->GetPressure(); // [lbs/ft^2]
684 const double AirPressure = in.Pressure; // [lbs/ft^2]
685
686 const double OldTemperature = Temperature;
687 const double OldPressure = Pressure;
688 unsigned int i;
689
690 //-- Gas temperature --
691
692 // The model is based on the ideal gas law.
693 // However, it does look a bit fishy. Please verify.
694 // dT/dt = dU / (Cv n R)
695 dU = 0.0;
696 for (i = 0; i < HeatTransferCoeff.size(); i++) {
697 dU += HeatTransferCoeff[i]->GetValue();
698 }
699 // dt is already accounted for in dVolumeIdeal.
700 if (Contents > 0) {
701 Temperature +=
702 (dU * dt - Pressure * dVolumeIdeal) / (Cv_air * Contents * R);
703 } else {
704 Temperature = Parent->GetTemperature();
705 }
706
707 //-- Pressure --
708 const double IdealPressure = Contents * R * Temperature / MaxVolume;
709 // The pressure is at least that of the parent gas cell.
710 Pressure = max(IdealPressure, ParentPressure);
711
712 //-- Blower input --
713 if (BlowerInput) {
714 const double AddedVolume = BlowerInput->GetValue() * dt;
715 if (AddedVolume > 0.0) {
716 Contents += Pressure * AddedVolume / (R * Temperature);
717 }
718 }
719
720 //-- Pressure relief and manual valving --
721 // FIXME: Presently the effect of valving is computed using
722 // an ad hoc formula which might not be a good representation
723 // of reality.
724 if ((ValveCoefficient > 0.0) &&
725 ((ValveOpen > 0.0) || (Pressure > AirPressure + MaxOverpressure))) {
726 const double DeltaPressure = Pressure - AirPressure;
727 const double VolumeValved =
728 ((Pressure > AirPressure + MaxOverpressure) ? 1.0 : ValveOpen) *
729 ValveCoefficient * DeltaPressure * dt;
730 // FIXME: Too small values of Contents sometimes leads to NaN.
731 // Currently the minimum is restricted to a safe value.
732 Contents =
733 max(1.0, Contents - Pressure * VolumeValved / (R * Temperature));
734 }
735
736 //-- Volume --
737 Volume = Contents * R * Temperature / Pressure;
738 dVolumeIdeal =
739 Contents * R * (Temperature / Pressure - OldTemperature / OldPressure);
740
741 // Compute the inertia of the ballonet.
742 // Consider the ballonet as a shape of uniform density.
743 // FIXME: If the ballonet isn't ellipsoid or cylindrical the inertia will
744 // be wrong.
745 ballonetJ.InitMatrix();
746 const double mass = Contents * M_air;
747 double Ixx, Iyy, Izz;
748 if ((Xradius != 0.0) && (Yradius != 0.0) && (Zradius != 0.0) &&
749 (Xwidth == 0.0) && (Ywidth == 0.0) && (Zwidth == 0.0)) {
750 // Ellipsoid volume.
751 Ixx = (1.0 / 5.0) * mass * (Yradius*Yradius + Zradius*Zradius);
752 Iyy = (1.0 / 5.0) * mass * (Xradius*Xradius + Zradius*Zradius);
753 Izz = (1.0 / 5.0) * mass * (Xradius*Xradius + Yradius*Yradius);
754 } else if ((Xradius == 0.0) && (Yradius != 0.0) && (Zradius != 0.0) &&
755 (Xwidth != 0.0) && (Ywidth == 0.0) && (Zwidth == 0.0)) {
756 // Cylindrical volume (might not be valid with an elliptical cross-section).
757 Ixx = (1.0 / 2.0) * mass * Yradius * Zradius;
758 Iyy =
759 (1.0 / 4.0) * mass * Yradius * Zradius +
760 (1.0 / 12.0) * mass * Xwidth * Xwidth;
761 Izz =
762 (1.0 / 4.0) * mass * Yradius * Zradius +
763 (1.0 / 12.0) * mass * Xwidth * Xwidth;
764 } else {
765 // Not supported. Revert to pointmass model.
766 Ixx = Iyy = Izz = 0.0;
767 }
768 // The volume is symmetric, so Ixy = Ixz = Iyz = 0.
769 ballonetJ(1,1) = Ixx;
770 ballonetJ(2,2) = Iyy;
771 ballonetJ(3,3) = Izz;
772 // Transform the moments of inertia to the body frame.
773 ballonetJ += MassBalance->GetPointmassInertia(GetMass(), GetXYZ());
774}
double GetMass(void) const
Get the current mass of the ballonets.
Definition FGGasCell.h:323
const FGColumnVector3 & GetXYZ(void) const
Get the center of gravity location of the ballonet.
Definition FGGasCell.h:316
double GetValue(void) const override
Retrieves the value of the function object.
void InitMatrix(void)
Initialize the matrix.
+ Here is the call graph for this function:

◆ GetHeatFlow()

double GetHeatFlow ( void  ) const
inline

Get the current heat flow into the ballonet.

Returns
heat flow in lbs ft / sec.

Definition at line 335 of file FGGasCell.h.

335{return dU;} // [lbs ft / sec]

◆ GetInertia()

const FGMatrix33 & GetInertia ( void  ) const
inline

Get the moments of inertia of the ballonet.

Returns
moments of inertia matrix in the body frame in slug ft2.

Definition at line 328 of file FGGasCell.h.

328{return ballonetJ;}

◆ GetMass()

double GetMass ( void  ) const
inline

Get the current mass of the ballonets.

Returns
mass in slug.

Definition at line 323 of file FGGasCell.h.

323{return Contents * M_air;}
+ Here is the caller graph for this function:

◆ GetVolume()

double GetVolume ( void  ) const
inline

Get the current volume of the ballonet.

Returns
volume in ft3.

Definition at line 332 of file FGGasCell.h.

332{return Volume;}

◆ GetXYZ() [1/2]

double GetXYZ ( int  idx) const
inline

Get the center of gravity location of the ballonet.

Returns
CoG location in the structural frame in inches.

Definition at line 319 of file FGGasCell.h.

319{return vXYZ(idx);}

◆ GetXYZ() [2/2]

const FGColumnVector3 & GetXYZ ( void  ) const
inline

Get the center of gravity location of the ballonet.

Returns
CoG location in the structural frame in inches.

Definition at line 316 of file FGGasCell.h.

316{return vXYZ;}
+ Here is the caller graph for this function:

Member Data Documentation

◆ in

const struct FGGasCell::Inputs& in

Definition at line 337 of file FGGasCell.h.


The documentation for this class was generated from the following files: