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
FGTank.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGTank.cpp
4 Author: Jon Berndt
5 Date started: 01/21/99
6 Called by: FGAircraft
7
8 ------------- Copyright (C) 1999 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--------------------------------------------------------------------------------
29See header file.
30
31HISTORY
32--------------------------------------------------------------------------------
3301/21/99 JSB Created
34
35%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
36INCLUDES
37%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
38
39#include "FGTank.h"
40#include "FGFDMExec.h"
41#include "input_output/FGXMLElement.h"
42
43using namespace std;
44
45namespace JSBSim {
46
47/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
48CLASS IMPLEMENTATION
49%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
50
51FGTank::FGTank(FGFDMExec* exec, Element* el, int tank_number)
52 : TankNumber(tank_number)
53{
54 string token, strFuelName;
55 Element* element;
56 Element* element_Grain;
57 auto PropertyManager = exec->GetPropertyManager();
58 Area = 1.0;
59 Density = 6.6;
60 InitialTemperature = Temperature = -9999.0;
61 Ixx = Iyy = Izz = 0.0;
62 InertiaFactor = 1.0;
63 Radius = Contents = Standpipe = Length = InnerRadius = 0.0;
64 ExternalFlow = 0.0;
65 InitialStandpipe = 0.0;
66 Capacity = 0.00001; UnusableVol = 0.0;
67 Priority = InitialPriority = 1;
68 vXYZ.InitMatrix();
69 vXYZ_drain.InitMatrix();
70 ixx_unit = iyy_unit = izz_unit = 1.0;
71 grainType = gtUNKNOWN; // This is the default
72
73 string type = el->GetAttributeValue("type");
74 if (type == "FUEL") Type = ttFUEL;
75 else if (type == "OXIDIZER") Type = ttOXIDIZER;
76 else Type = ttUNKNOWN;
77
78 Name = el->GetAttributeValue("name");
79
80 element = el->FindElement("location");
81 if (element) vXYZ = element->FindElementTripletConvertTo("IN");
82 else cerr << el->ReadFrom() << "No location found for this tank."
83 << endl;
84
85 vXYZ_drain = vXYZ; // Set initial drain location to initial tank CG
86
87 element = el->FindElement("drain_location");
88 if (element) {
89 vXYZ_drain = element->FindElementTripletConvertTo("IN");
90 }
91
92 if (el->FindElement("radius"))
93 Radius = el->FindElementValueAsNumberConvertTo("radius", "IN");
94 if (el->FindElement("inertia_factor"))
95 InertiaFactor = el->FindElementValueAsNumber("inertia_factor");
96 if (el->FindElement("capacity"))
97 Capacity = el->FindElementValueAsNumberConvertTo("capacity", "LBS");
98 if (el->FindElement("contents"))
99 InitialContents = Contents = el->FindElementValueAsNumberConvertTo("contents", "LBS");
100 if (el->FindElement("unusable-volume"))
101 UnusableVol = el->FindElementValueAsNumberConvertTo("unusable-volume", "GAL");
102 if (el->FindElement("temperature"))
103 InitialTemperature = Temperature = el->FindElementValueAsNumber("temperature");
104 if (el->FindElement("standpipe"))
105 InitialStandpipe = Standpipe = el->FindElementValueAsNumberConvertTo("standpipe", "LBS");
106 if (el->FindElement("priority"))
107 InitialPriority = Priority = (int)el->FindElementValueAsNumber("priority");
108 if (el->FindElement("density"))
109 Density = el->FindElementValueAsNumberConvertTo("density", "LBS/GAL");
110 if (el->FindElement("type"))
111 strFuelName = el->FindElementValue("type");
112
113
114 SetPriority( InitialPriority ); // this will also set the Selected flag
115
116 if (Capacity == 0) {
117 cerr << el->ReadFrom()
118 << "Tank capacity must not be zero. Reset to 0.00001 lbs!" << endl;
119 Capacity = 0.00001;
120 Contents = 0.0;
121 }
122 if (Capacity <= GetUnusable()) {
123 cerr << el->ReadFrom() << "Tank capacity (" << Capacity
124 << " lbs) is lower than the amount of unusable fuel (" << GetUnusable()
125 << " lbs) for tank " << tank_number
126 << "! Did you accidentally swap unusable and capacity?" << endl;
127 throw("tank definition error");
128 }
129 if (Contents > Capacity) {
130 cerr << el->ReadFrom() << "Tank content (" << Contents
131 << " lbs) is greater than tank capacity (" << Capacity
132 << " lbs) for tank " << tank_number
133 << "! Did you accidentally swap contents and capacity?" << endl;
134 throw("tank definition error");
135 }
136 if (Contents < GetUnusable()) {
137 cerr << el->ReadFrom() << "Tank content (" << Contents
138 << " lbs) is lower than the amount of unusable fuel (" << GetUnusable()
139 << " lbs) for tank " << tank_number << endl;
140 }
141
142 PctFull = 100.0*Contents/Capacity; // percent full; 0 to 100.0
143
144 // Check whether this is a solid propellant "tank". Initialize it if true.
145
146 element_Grain = el->FindElement("grain_config");
147 if (element_Grain) {
148
149 string strGType = element_Grain->GetAttributeValue("type");
150 if (strGType == "CYLINDRICAL") grainType = gtCYLINDRICAL;
151 else if (strGType == "ENDBURNING") grainType = gtENDBURNING;
152 else if (strGType == "FUNCTION") {
153 grainType = gtFUNCTION;
154 if (element_Grain->FindElement("ixx") != 0) {
155 Element* element_ixx = element_Grain->FindElement("ixx");
156 if (element_ixx->GetAttributeValue("unit") == "KG*M2") ixx_unit = 1.0/1.35594;
157 if (element_ixx->FindElement("function") != 0) {
158 function_ixx = new FGFunction(exec, element_ixx->FindElement("function"));
159 }
160 } else {
161 throw("For tank "+to_string(TankNumber)+" and when grain_config is specified an ixx must be specified when the FUNCTION grain type is specified.");
162 }
163
164 if (element_Grain->FindElement("iyy")) {
165 Element* element_iyy = element_Grain->FindElement("iyy");
166 if (element_iyy->GetAttributeValue("unit") == "KG*M2") iyy_unit = 1.0/1.35594;
167 if (element_iyy->FindElement("function") != 0) {
168 function_iyy = new FGFunction(exec, element_iyy->FindElement("function"));
169 }
170 } else {
171 throw("For tank "+to_string(TankNumber)+" and when grain_config is specified an iyy must be specified when the FUNCTION grain type is specified.");
172 }
173
174 if (element_Grain->FindElement("izz")) {
175 Element* element_izz = element_Grain->FindElement("izz");
176 if (element_izz->GetAttributeValue("unit") == "KG*M2") izz_unit = 1.0/1.35594;
177 if (element_izz->FindElement("function") != 0) {
178 function_izz = new FGFunction(exec, element_izz->FindElement("function"));
179 }
180 } else {
181 throw("For tank "+to_string(TankNumber)+" and when grain_config is specified an izz must be specified when the FUNCTION grain type is specified.");
182 }
183 }
184 else
185 cerr << el->ReadFrom() << "Unknown propellant grain type specified"
186 << endl;
187
188 if (element_Grain->FindElement("length"))
189 Length = element_Grain->FindElementValueAsNumberConvertTo("length", "IN");
190 if (element_Grain->FindElement("bore_diameter"))
191 InnerRadius = element_Grain->FindElementValueAsNumberConvertTo("bore_diameter", "IN")/2.0;
192
193 // Initialize solid propellant values for debug and runtime use.
194
195 switch (grainType) {
196 case gtCYLINDRICAL:
197 if (Radius <= InnerRadius) {
198 const string s("The bore diameter should be smaller than the total grain diameter!");
199 cerr << element_Grain->ReadFrom() << endl << s << endl;
200 throw BaseException(s);
201 }
202 Volume = M_PI * Length * (Radius*Radius - InnerRadius*InnerRadius); // cubic inches
203 break;
204 case gtENDBURNING:
205 Volume = M_PI * Length * Radius * Radius; // cubic inches
206 break;
207 case gtFUNCTION:
208 Volume = 1; // Volume is irrelevant for the FUNCTION type, but it can't be zero!
209 break;
210 case gtUNKNOWN:
211 {
212 const string s("Unknown grain type found in this rocket engine definition.");
213 cerr << el->ReadFrom() << endl << s << endl;
214 throw BaseException(s);
215 }
216 }
217 Density = (Capacity*lbtoslug)/Volume; // slugs/in^3
218 }
219
220 CalculateInertias();
221
222 if (Temperature != -9999.0) InitialTemperature = Temperature = FahrenheitToCelsius(Temperature);
223 Area = 40.0 * pow(Capacity/1975, 0.666666667);
224
225 // A named fuel type will override a previous density value
226 if (!strFuelName.empty()) Density = ProcessFuelName(strFuelName);
227
228 bind(PropertyManager.get());
229
230 Debug(0);
231}
232
233//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
234
236{
237 Debug(1);
238}
239
240//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
241
243{
244 SetTemperature( InitialTemperature );
245 SetStandpipe ( InitialStandpipe );
246 SetContents ( InitialContents );
247 PctFull = 100.0*Contents/Capacity;
248 SetPriority( InitialPriority );
249 CalculateInertias();
250}
251
252//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
253
254FGColumnVector3 FGTank::GetXYZ(void) const
255{
256 return vXYZ_drain + (Contents/Capacity)*(vXYZ - vXYZ_drain);
257}
258
259//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
260
261double FGTank::GetXYZ(int idx) const
262{
263 return vXYZ_drain(idx) + (Contents/Capacity)*(vXYZ(idx)-vXYZ_drain(idx));
264}
265
266//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
267
268double FGTank::Drain(double used)
269{
270 double remaining = Contents - used;
271
272 if (remaining >= GetUnusable()) { // Reduce contents by amount used.
273 Contents -= used;
274 } else { // This tank must be empty.
275 if (Contents > GetUnusable())
276 Contents = GetUnusable();
277
278 remaining = Contents;
279 }
280
281 PctFull = 100.0*Contents/Capacity;
282 CalculateInertias();
283
284 return remaining;
285}
286
287//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
288
289double FGTank::Fill(double amount)
290{
291 double overage = 0.0;
292
293 Contents += amount;
294
295 if (Contents > Capacity) {
296 overage = Contents - Capacity;
297 Contents = Capacity;
298 PctFull = 100.0;
299 } else {
300 PctFull = Contents/Capacity*100.0;
301 }
302
303 CalculateInertias();
304
305 return overage;
306}
307
308//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
309
310void FGTank::SetContents(double amount)
311{
312 Contents = amount;
313 if (Contents > Capacity) {
314 Contents = Capacity;
315 PctFull = 100.0;
316 } else {
317 PctFull = Contents/Capacity*100.0;
318 }
319
320 CalculateInertias();
321}
322
323//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
324
325void FGTank::SetContentsGallons(double gallons)
326{
327 SetContents(gallons * Density);
328}
329
330
331//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
332
333double FGTank::Calculate(double dt, double TAT_C)
334{
335 if(ExternalFlow < 0.) Drain( -ExternalFlow *dt);
336 else Fill(ExternalFlow * dt);
337
338 if (Temperature == -9999.0) return 0.0;
339 double HeatCapacity = 900.0; // Joules/lbm/C
340 double TempFlowFactor = 1.115; // Watts/sqft/C
341 double Tdiff = TAT_C - Temperature;
342 double dTemp = 0.0; // Temp change due to one surface
343 if (fabs(Tdiff) > 0.1 && Contents > 0.01) {
344 dTemp = (TempFlowFactor * Area * Tdiff * dt) / (Contents * HeatCapacity);
345 }
346
347 return Temperature += (dTemp + dTemp); // For now, assume upper/lower the same
348}
349
350//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
351// This function calculates the moments of inertia for a solid propellant
352// grain - either an end burning cylindrical grain or a bored cylindrical
353// grain, as well as liquid propellants IF a tank radius and inertia factor
354// are given.
355//
356// From NASA CR-383, the MoI of a tank with liquid propellant is specified
357// for baffled and non-baffled tanks as a ratio compared to that in which the
358// propellant is solid. The more baffles, the more "rigid" the propellant and
359// the higher the ratio (up to 1.0). For a cube tank with five baffles, the
360// ratio ranges from 0.5 to 0.7. For a cube tank with no baffles, the ratio is
361// roughly 0.18. One might estimate that for a spherical tank with no baffles
362// the ratio might be somewhere around 0.10 to 0.15. Cylindrical tanks with or
363// without baffles might have biased moment of inertia effects based on the
364// baffle layout and tank geometry. A vector inertia_factor may be supported
365// at some point.
366
367void FGTank::CalculateInertias(void)
368{
369 double Mass = Contents*lbtoslug;
370 double RadSumSqr;
371 double Rad2 = Radius*Radius;
372
373 if (grainType != gtUNKNOWN) { // assume solid propellant
374
375 if (Density > 0.0) {
376 Volume = (Contents*lbtoslug)/Density; // in^3
377 } else if (Contents <= 0.0) {
378 Volume = 0;
379 } else {
380 const string s(" Solid propellant grain density is zero!");
381 cerr << endl << s << endl;
382 throw BaseException(s);
383 }
384
385 switch (grainType) {
386 case gtCYLINDRICAL:
387 InnerRadius = sqrt(Rad2 - Volume/(M_PI * Length));
388 RadSumSqr = (Rad2 + InnerRadius*InnerRadius)/144.0;
389 Ixx = 0.5*Mass*RadSumSqr;
390 Iyy = Mass*(3.0*RadSumSqr + Length*Length/144.0)/12.0;
391 Izz = Iyy;
392 break;
393 case gtENDBURNING:
394 Length = Volume/(M_PI*Rad2);
395 Ixx = 0.5*Mass*Rad2/144.0;
396 Iyy = Mass*(3.0*Rad2 + Length*Length)/(144.0*12.0);
397 Izz = Iyy;
398 break;
399 case gtFUNCTION:
400 Ixx = function_ixx->GetValue()*ixx_unit;
401 Iyy = function_iyy->GetValue()*iyy_unit;
402 Izz = function_izz->GetValue()*izz_unit;
403 break;
404 default:
405 {
406 const string s("Unknown grain type found.");
407 cerr << s << endl;
408 throw BaseException(s);
409 }
410 }
411
412 } else { // assume liquid propellant: shrinking snowball
413
414 if (Radius > 0.0) Ixx = Iyy = Izz = Mass * InertiaFactor * 0.4 * Radius * Radius / 144.0;
415
416 }
417}
418
419//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
420
421double FGTank::ProcessFuelName(const std::string& name)
422{
423 if (name == "AVGAS") return 6.02;
424 else if (name == "JET-A") return 6.74;
425 else if (name == "JET-A1") return 6.74;
426 else if (name == "JET-B") return 6.48;
427 else if (name == "JP-1") return 6.76;
428 else if (name == "JP-2") return 6.38;
429 else if (name == "JP-3") return 6.34;
430 else if (name == "JP-4") return 6.48;
431 else if (name == "JP-5") return 6.81;
432 else if (name == "JP-6") return 6.55;
433 else if (name == "JP-7") return 6.61;
434 else if (name == "JP-8") return 6.66;
435 else if (name == "JP-8+100") return 6.66;
436 //else if (name == "JP-9") return 6.74;
437 //else if (name == "JPTS") return 6.74;
438 else if (name == "RP-1") return 6.73;
439 else if (name == "T-1") return 6.88;
440 else if (name == "ETHANOL") return 6.58;
441 else if (name == "HYDRAZINE")return 8.61;
442 else if (name == "F-34") return 6.66;
443 else if (name == "F-35") return 6.74;
444 else if (name == "F-40") return 6.48;
445 else if (name == "F-44") return 6.81;
446 else if (name == "AVTAG") return 6.48;
447 else if (name == "AVCAT") return 6.81;
448 else {
449 cerr << "Unknown fuel type specified: "<< name << endl;
450 }
451
452 return 6.6;
453}
454
455//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
456
457void FGTank::bind(FGPropertyManager* PropertyManager)
458{
459 string property_name, base_property_name;
460 base_property_name = CreateIndexedPropertyName("propulsion/tank", TankNumber);
461 property_name = base_property_name + "/contents-lbs";
462 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetContents,
463 &FGTank::SetContents );
464 property_name = base_property_name + "/unusable-volume-gal";
465 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetUnusableVolume,
467 property_name = base_property_name + "/pct-full";
468 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetPctFull);
469 property_name = base_property_name + "/density-lbs_per_gal";
470 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetDensity);
471
472 property_name = base_property_name + "/priority";
473 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetPriority,
474 &FGTank::SetPriority );
475 property_name = base_property_name + "/external-flow-rate-pps";
476 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetExternalFlow,
477 &FGTank::SetExternalFlow );
478 property_name = base_property_name + "/local-ixx-slug_ft2";
479 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetIxx);
480 property_name = base_property_name + "/local-iyy-slug_ft2";
481 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetIyy);
482 property_name = base_property_name + "/local-izz-slug_ft2";
483 PropertyManager->Tie( property_name.c_str(), (FGTank*)this, &FGTank::GetIzz);
484
485 property_name = base_property_name + "/x-position";
486 PropertyManager->Tie(property_name.c_str(), (FGTank*)this, &FGTank::GetLocationX, &FGTank::SetLocationX);
487 property_name = base_property_name + "/y-position";
488 PropertyManager->Tie(property_name.c_str(), (FGTank*)this, &FGTank::GetLocationY, &FGTank::SetLocationY);
489 property_name = base_property_name + "/z-position";
490 PropertyManager->Tie(property_name.c_str(), (FGTank*)this, &FGTank::GetLocationZ, &FGTank::SetLocationZ);
491
492}
493
494//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
495// The bitmasked value choices are as follows:
496// unset: In this case (the default) JSBSim would only print
497// out the normally expected messages, essentially echoing
498// the config files as they are read. If the environment
499// variable is not set, debug_lvl is set to 1 internally
500// 0: This requests JSBSim not to output any messages
501// whatsoever.
502// 1: This value explicity requests the normal JSBSim
503// startup messages
504// 2: This value asks for a message to be printed out when
505// a class is instantiated
506// 4: When this value is set, a message is displayed when a
507// FGModel object executes its Run() method
508// 8: When this value is set, various runtime state variables
509// are printed out periodically
510// 16: When set various parameters are sanity checked and
511// a message is printed out when they go out of bounds
512
513void FGTank::Debug(int from)
514{
515 if (debug_lvl <= 0) return;
516
517 if (debug_lvl & 1) { // Standard console startup message output
518 if (from == 0) { // Constructor
519 string type;
520 switch (Type) {
521 case ttFUEL:
522 type = "FUEL";
523 break;
524 case ttOXIDIZER:
525 type = "OXIDIZER";
526 break;
527 default:
528 type = "UNKNOWN";
529 break;
530 }
531
532 cout << " " << Name << " (" << type << ") tank holds " << Capacity << " lbs. " << type << endl;
533 cout << " currently at " << PctFull << "% of maximum capacity" << endl;
534 cout << " Tank location (X, Y, Z): " << vXYZ(eX) << ", " << vXYZ(eY) << ", " << vXYZ(eZ) << endl;
535 cout << " Effective radius: " << Radius << " inches" << endl;
536 cout << " Initial temperature: " << Temperature << " Fahrenheit" << endl;
537 cout << " Priority: " << Priority << endl;
538 }
539 }
540 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
541 if (from == 0) cout << "Instantiated: FGTank" << endl;
542 if (from == 1) cout << "Destroyed: FGTank" << endl;
543 }
544 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
545 }
546 if (debug_lvl & 8 ) { // Runtime state variables
547 }
548 if (debug_lvl & 16) { // Sanity checking
549 }
550 if (debug_lvl & 64) {
551 if (from == 0) { // Constructor
552 }
553 }
554}
555}
Element * FindElement(const std::string &el="")
Searches for a specified element.
FGColumnVector3 FindElementTripletConvertTo(const std::string &target_units)
Composes a 3-element column vector for the supplied location or orientation.
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...
std::string FindElementValue(const std::string &el="")
Searches for the named element and returns the string data belonging to it.
double FindElementValueAsNumberConvertTo(const std::string &el, const std::string &target_units)
Searches for the named element and converts and returns the data belonging to it.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
This class implements a 3 element column vector.
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Definition FGFDMExec.h:421
Represents a mathematical function.
Definition FGFunction.h:765
double GetValue(void) const override
Retrieves the value of the function object.
static constexpr double FahrenheitToCelsius(double fahrenheit)
Converts from degrees Fahrenheit to degrees Celsius.
Definition FGJSBBase.h:220
void Tie(const std::string &name, T *pointer)
Tie a property to an external variable.
Models a fuel tank.
Definition FGTank.h:203
double GetDensity(void) const
Returns the fuel density.
Definition FGTank.h:320
double ProcessFuelName(const std::string &name)
Returns the density of a named fuel type.
Definition FGTank.cpp:421
double GetContents(void) const
Gets the contents of the tank.
Definition FGTank.h:266
FGTank(FGFDMExec *exec, Element *el, int tank_number)
Constructor.
Definition FGTank.cpp:51
void ResetToIC(void)
Resets the tank parameters to the initial conditions.
Definition FGTank.cpp:242
double Drain(double used)
Removes fuel from the tank.
Definition FGTank.cpp:268
~FGTank()
Destructor.
Definition FGTank.cpp:235
double GetUnusable(void) const
Returns the amount of unusable fuel in the tank.
Definition FGTank.h:292
double Calculate(double dt, double TempC)
Performs local, tanks-specific calculations, such as fuel temperature.
Definition FGTank.cpp:333
void SetUnusableVolume(double volume)
Sets the volume of unusable fuel in the tank.
Definition FGTank.h:300
double GetPctFull(void) const
Gets the tank fill level.
Definition FGTank.h:254
double GetUnusableVolume(void) const
Returns the unusable volume of fuel in the tank.
Definition FGTank.h:296