41#include "input_output/FGXMLElement.h"
52 : TankNumber(tank_number)
54 string token, strFuelName;
60 InitialTemperature = Temperature = -9999.0;
61 Ixx = Iyy = Izz = 0.0;
63 Radius = Contents = Standpipe = Length = InnerRadius = 0.0;
65 InitialStandpipe = 0.0;
66 Capacity = 0.00001; UnusableVol = 0.0;
67 Priority = InitialPriority = 1;
69 vXYZ_drain.InitMatrix();
70 ixx_unit = iyy_unit = izz_unit = 1.0;
71 grainType = gtUNKNOWN;
74 if (type ==
"FUEL") Type = ttFUEL;
75 else if (type ==
"OXIDIZER") Type = ttOXIDIZER;
76 else Type = ttUNKNOWN;
82 else cerr << el->
ReadFrom() <<
"No location found for this tank."
114 SetPriority( InitialPriority );
118 <<
"Tank capacity must not be zero. Reset to 0.00001 lbs!" << endl;
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");
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");
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;
142 PctFull = 100.0*Contents/Capacity;
150 if (strGType ==
"CYLINDRICAL") grainType = gtCYLINDRICAL;
151 else if (strGType ==
"ENDBURNING") grainType = gtENDBURNING;
152 else if (strGType ==
"FUNCTION") {
153 grainType = gtFUNCTION;
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.");
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.");
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.");
185 cerr << el->
ReadFrom() <<
"Unknown propellant grain type specified"
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;
202 Volume = M_PI * Length * (Radius*Radius - InnerRadius*InnerRadius);
205 Volume = M_PI * Length * Radius * Radius;
212 const string s(
"Unknown grain type found in this rocket engine definition.");
213 cerr << el->
ReadFrom() << endl << s << endl;
217 Density = (Capacity*lbtoslug)/Volume;
222 if (Temperature != -9999.0) InitialTemperature = Temperature =
FahrenheitToCelsius(Temperature);
223 Area = 40.0 * pow(Capacity/1975, 0.666666667);
228 bind(PropertyManager.get());
244 SetTemperature( InitialTemperature );
245 SetStandpipe ( InitialStandpipe );
246 SetContents ( InitialContents );
247 PctFull = 100.0*Contents/Capacity;
248 SetPriority( InitialPriority );
256 return vXYZ_drain + (Contents/Capacity)*(vXYZ - vXYZ_drain);
261double FGTank::GetXYZ(
int idx)
const
263 return vXYZ_drain(idx) + (Contents/Capacity)*(vXYZ(idx)-vXYZ_drain(idx));
270 double remaining = Contents - used;
278 remaining = Contents;
281 PctFull = 100.0*Contents/Capacity;
289double FGTank::Fill(
double amount)
291 double overage = 0.0;
295 if (Contents > Capacity) {
296 overage = Contents - Capacity;
300 PctFull = Contents/Capacity*100.0;
310void FGTank::SetContents(
double amount)
313 if (Contents > Capacity) {
317 PctFull = Contents/Capacity*100.0;
325void FGTank::SetContentsGallons(
double gallons)
327 SetContents(gallons * Density);
335 if(ExternalFlow < 0.)
Drain( -ExternalFlow *dt);
336 else Fill(ExternalFlow * dt);
338 if (Temperature == -9999.0)
return 0.0;
339 double HeatCapacity = 900.0;
340 double TempFlowFactor = 1.115;
341 double Tdiff = TAT_C - Temperature;
343 if (fabs(Tdiff) > 0.1 && Contents > 0.01) {
344 dTemp = (TempFlowFactor * Area * Tdiff * dt) / (Contents * HeatCapacity);
347 return Temperature += (dTemp + dTemp);
367void FGTank::CalculateInertias(
void)
369 double Mass = Contents*lbtoslug;
371 double Rad2 = Radius*Radius;
373 if (grainType != gtUNKNOWN) {
376 Volume = (Contents*lbtoslug)/Density;
377 }
else if (Contents <= 0.0) {
380 const string s(
" Solid propellant grain density is zero!");
381 cerr << endl << s << endl;
382 throw BaseException(s);
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;
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);
400 Ixx = function_ixx->
GetValue()*ixx_unit;
401 Iyy = function_iyy->
GetValue()*iyy_unit;
402 Izz = function_izz->
GetValue()*izz_unit;
406 const string s(
"Unknown grain type found.");
408 throw BaseException(s);
414 if (Radius > 0.0) Ixx = Iyy = Izz = Mass * InertiaFactor * 0.4 * Radius * Radius / 144.0;
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;
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;
449 cerr <<
"Unknown fuel type specified: "<< name << endl;
459 string property_name, base_property_name;
460 base_property_name = CreateIndexedPropertyName(
"propulsion/tank", TankNumber);
461 property_name = base_property_name +
"/contents-lbs";
463 &FGTank::SetContents );
464 property_name = base_property_name +
"/unusable-volume-gal";
467 property_name = base_property_name +
"/pct-full";
469 property_name = base_property_name +
"/density-lbs_per_gal";
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);
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);
513void FGTank::Debug(
int from)
515 if (debug_lvl <= 0)
return;
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;
540 if (debug_lvl & 2 ) {
541 if (from == 0) cout <<
"Instantiated: FGTank" << endl;
542 if (from == 1) cout <<
"Destroyed: FGTank" << endl;
544 if (debug_lvl & 4 ) {
546 if (debug_lvl & 8 ) {
548 if (debug_lvl & 16) {
550 if (debug_lvl & 64) {
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.
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Represents a mathematical function.
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.
void Tie(const std::string &name, T *pointer)
Tie a property to an external variable.
double GetDensity(void) const
Returns the fuel density.
double ProcessFuelName(const std::string &name)
Returns the density of a named fuel type.
double GetContents(void) const
Gets the contents of the tank.
FGTank(FGFDMExec *exec, Element *el, int tank_number)
Constructor.
void ResetToIC(void)
Resets the tank parameters to the initial conditions.
double Drain(double used)
Removes fuel from the tank.
double GetUnusable(void) const
Returns the amount of unusable fuel in the tank.
double Calculate(double dt, double TempC)
Performs local, tanks-specific calculations, such as fuel temperature.
void SetUnusableVolume(double volume)
Sets the volume of unusable fuel in the tank.
double GetPctFull(void) const
Gets the tank fill level.
double GetUnusableVolume(void) const
Returns the unusable volume of fuel in the tank.