42#include "FGPropeller.h"
43#include "input_output/FGXMLElement.h"
56 Element *table_element, *local_element;
60 MaxPitch = MinPitch = P_Factor = Pitch = Advance = MinRPM = MaxRPM = 0.0;
67 CtFactor = CpFactor = 1.0;
69 cThrust = cPower = CtMach = CpMach = 0;
75 Sense_multiplier = 1.0;
78 Sense_multiplier = -1.0;
103 if (name ==
"C_THRUST") {
104 cThrust =
new FGTable(PropertyManager, table_element);
105 }
else if (name ==
"C_POWER") {
106 cPower =
new FGTable(PropertyManager, table_element);
107 }
else if (name ==
"CT_MACH") {
108 CtMach =
new FGTable(PropertyManager, table_element);
109 }
else if (name ==
"CP_MACH") {
110 CpMach =
new FGTable(PropertyManager, table_element);
112 cerr <<
"Unknown table type: " << name <<
" in propeller definition." << endl;
114 }
catch (std::string& str) {
115 throw(
"Error loading propeller table:" + name +
". " + str);
118 if( (cPower == 0) || (cThrust == 0)){
119 cerr <<
"Propeller configuration must contain C_THRUST and C_POWER tables!" << endl;
125 SetSense(Sense >= 0.0 ? 1.0 : -1.0);
132 cerr <<
"P-Factor value in propeller configuration file must be greater than zero" << endl;
141 vTorque.InitMatrix();
142 D4 = Diameter*Diameter*Diameter*Diameter;
146 string property_name, base_property_name;
147 base_property_name = CreateIndexedPropertyName(
"propulsion/engine", EngineNum);
148 property_name = base_property_name +
"/engine-rpm";
150 property_name = base_property_name +
"/advance-ratio";
151 PropertyManager->Tie( property_name.c_str(), &J );
152 property_name = base_property_name +
"/blade-angle";
153 PropertyManager->Tie( property_name.c_str(), &Pitch );
154 property_name = base_property_name +
"/thrust-coefficient";
156 property_name = base_property_name +
"/propeller-rpm";
158 property_name = base_property_name +
"/helical-tip-Mach";
160 property_name = base_property_name +
"/constant-speed-mode";
163 property_name = base_property_name +
"/prop-induced-velocity_fps";
166 property_name = base_property_name +
"/propeller-power-ftlbps";
167 PropertyManager->Tie( property_name.c_str(), &PowerRequired );
168 property_name = base_property_name +
"/propeller-torque-ftlb";
170 property_name = base_property_name +
"/propeller-sense";
171 PropertyManager->Tie( property_name.c_str(), &Sense );
192 FGThruster::ResetToIC();
216 double omega, PowerAvailable;
218 double Vel = localAeroVel(eU);
219 double rho = in.Density;
220 double RPS = RPM/60.0;
223 double Area = 0.25*Diameter*Diameter*M_PI;
224 double Vtip = RPS * Diameter * M_PI;
225 HelicalTipMach = sqrt(Vtip*Vtip + Vel*Vel) / in.Soundspeed;
227 if (RPS > 0.01) J = Vel / (Diameter * RPS);
228 else J = Vel / Diameter;
232 if (MaxPitch == MinPitch) {
235 ThrustCoeff = cThrust->
GetValue(J, Pitch);
239 ThrustCoeff *= CtFactor;
242 if (CtMach) ThrustCoeff *= CtMach->
GetValue(HelicalTipMach);
244 Thrust = ThrustCoeff*RPS*RPS*D4*rho;
251 double Vel2sum = Vel*abs(Vel) + 2.0*Thrust/(rho*Area);
254 Vinduced = 0.5 * (-Vel + sqrt(Vel2sum));
256 Vinduced = 0.5 * (-Vel - sqrt(-Vel2sum));
261 if (P_Factor > 0.0001) {
262 double tangentialVel = localAeroVel.
Magnitude(eV, eW);
264 if (tangentialVel > 0.0001) {
268 double angle = atan2(tangentialVel, Vel+Vinduced);
269 double factor = Sense * P_Factor * angle / tangentialVel;
270 SetActingLocationY( GetLocationY() + factor * localAeroVel(eW));
271 SetActingLocationZ( GetLocationZ() + factor * localAeroVel(eV));
275 omega = RPS*2.0*M_PI;
278 vTorque(eX) = -Sense*EnginePower / max(0.01, omega);
286 if (omega > 0.01) ExcessTorque = PowerAvailable / omega;
287 else ExcessTorque = PowerAvailable / 1.0;
289 RPM = (RPS + ((ExcessTorque / Ixx) / (2.0 * M_PI)) * in.TotalDeltaT) * 60.0;
291 if (RPM < 0.0) RPM = 0.0;
295 vMn = in.PQRi*(mT*vH) + mT*vTorque;
306 if (MaxPitch == MinPitch) {
311 if (ConstantSpeed != 0) {
320 double rpmReq = MinRPM + (MaxRPM - MinRPM) * Advance;
321 double dRPM = rpmReq - RPM;
324 if (RPM > 200) Pitch -= dRPM * in.TotalDeltaT;
325 if (Pitch < MinPitch) Pitch = MinPitch;
326 else if (Pitch > MaxPitch) Pitch = MaxPitch;
332 double PitchReq = MinPitch - ( MinPitch - ReversePitch ) * Reverse_coef;
335 if (RPM > 200) Pitch += (PitchReq - Pitch) / 200;
337 Pitch += (MaxRPM - RPM) / 50;
338 if (Pitch < ReversePitch) Pitch = ReversePitch;
339 else if (Pitch > MaxPitch) Pitch = MaxPitch;
345 Pitch += (MaxPitch - Pitch) / 300;
359 if (CpMach) cPReq *= CpMach->
GetValue(HelicalTipMach);
361 double RPS = RPM / 60.0;
362 double local_RPS = RPS < 0.01 ? 0.01 : RPS;
364 PowerRequired = cPReq*local_RPS*local_RPS*local_RPS*D5*in.Density;
366 return PowerRequired;
375 double p_pitch = Thrust * Sense * (GetActingLocationZ() - GetLocationZ()) / 12.0;
377 double p_yaw = Thrust * Sense * (GetActingLocationY() - GetLocationY()) / 12.0;
386 std::ostringstream buf;
388 buf << Name <<
" Torque (engine " <<
id <<
")" << delimeter
389 << Name <<
" PFactor Pitch (engine " <<
id <<
")" << delimeter
390 << Name <<
" PFactor Yaw (engine " <<
id <<
")" << delimeter
391 << Name <<
" Thrust (engine " <<
id <<
" in lbs)" << delimeter;
393 buf << Name <<
" Pitch (engine " <<
id <<
")" << delimeter;
394 buf << Name <<
" RPM (engine " <<
id <<
")";
403 std::ostringstream buf;
406 buf << vTorque(eX) << delimeter
407 << vPFactor(ePitch) << delimeter
408 << vPFactor(eYaw) << delimeter
409 << Thrust << delimeter;
411 buf << Pitch << delimeter;
436void FGPropeller::Debug(
int from)
438 if (debug_lvl <= 0)
return;
442 cout <<
"\n Propeller Name: " << Name << endl;
443 cout <<
" IXX = " << Ixx << endl;
444 cout <<
" Diameter = " << Diameter <<
" ft." << endl;
445 cout <<
" Number of Blades = " << numBlades << endl;
446 cout <<
" Gear Ratio = " << GearRatio << endl;
447 cout <<
" Minimum Pitch = " << MinPitch << endl;
448 cout <<
" Maximum Pitch = " << MaxPitch << endl;
449 cout <<
" Minimum RPM = " << MinRPM << endl;
450 cout <<
" Maximum RPM = " << MaxRPM << endl;
472 if (debug_lvl & 2 ) {
473 if (from == 0) cout <<
"Instantiated: FGPropeller" << endl;
474 if (from == 1) cout <<
"Destroyed: FGPropeller" << endl;
476 if (debug_lvl & 4 ) {
478 if (debug_lvl & 8 ) {
480 if (debug_lvl & 16) {
482 if (debug_lvl & 64) {
Element * FindElement(const std::string &el="")
Searches for a specified element.
double GetAttributeValueAsNumber(const std::string &key)
Retrieves an attribute value as a double precision real number.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
Element * GetParent(void)
Returns a pointer to the parent of an element.
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.
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
double GetDataAsNumber(void)
Converts the element data to a number.
bool HasAttribute(const std::string &key)
Determines if an element has the supplied attribute.
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.
double Magnitude(void) const
Length of the vector.
Encapsulates the JSBSim simulation executive.
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Handles matrix math operations.
FGMatrix33 Transposed(void) const
Transposed matrix.
void SetInducedVelocity(double Vi)
Set the propeller induced velocity.
~FGPropeller()
Destructor for FGPropeller - deletes the FGTable objects.
double Calculate(double EnginePower)
Calculates and returns the thrust produced by this propeller.
double GetPowerRequired(void)
Retrieves the power required (or "absorbed") by the propeller - i.e.
int GetConstantSpeed(void) const
Returns a non-zero value if the propeller is constant speed.
double GetRPM(void) const
Retrieves the RPMs of the propeller.
std::string GetThrusterValues(int id, const std::string &delimeter)
Generate the values for the thruster standard CSV output.
void ResetToIC(void)
Reset the initial conditions.
void SetCtFactor(double ctf)
Sets coefficient of thrust multiplier.
void SetConstantSpeed(int mode)
Sets propeller into constant speed mode, or manual pitch mode.
FGColumnVector3 GetPFactor(void) const
Retrieves the P-Factor constant.
void SetCpFactor(double cpf)
Sets coefficient of power multiplier.
double GetInducedVelocity(void) const
Get the propeller induced velocity.
void SetSense(double s)
Sets the rotation sense of the propeller.
double GetHelicalTipMach(void) const
Retrieves the Mach number at the propeller tips.
std::string GetThrusterLabels(int id, const std::string &delimeter)
Generate the labels for the thruster standard CSV output.
double GetTorque(void) const
Retrieves the Torque in foot-pounds (Don't you love the English system?)
double GetEngineRPM(void) const
Calculates the RPMs of the engine based on gear ratio.
FGPropeller(FGFDMExec *exec, Element *el, int num=0)
Constructor for FGPropeller.
bool IsVPitch(void) const
Returns true of this propeller is variable pitch.
double GetThrustCoefficient(void) const
Retrieves the thrust coefficient.
double GetValue(void) const
Get the current table value.
Base class for specific thrusting devices such as propellers, nozzles, etc.