Avionics
Dropship Simulator
FADEC.cpp
Go to the documentation of this file.
1 #include "Module.h"
2 
4 
5 void Fadec::FrameMove(float fElapsed)
6 {
8  if (fElapsed > 1.0f || !bass)
9  {
10  if (flightControls.size() == 1 && flightControls.at(0)->IsActive())
11  {
12  *fadecFaultPtr |= Systems::Fault::NoSignal; // ENG 1 TLA FAIL
14  }
15  return;
16  }
17 
18 #pragma region Command Handler
19  for (UINT i = 0; i < bus->commandStream.size(); i++)
20  {
21  Command command = bus->commandStream.at(i);
22  if (command.delay != 0.0f) continue;
23 
24  if (command.name == "EngineStart")
25  {
26  if (command.ivalue == engineOrdinal)
27  {
29  if (engineStart || engineRun)
30  {
31  *N1SetFloatPtr *= 0.9f;
32  // bus->AirDensity *= 2.0f;
33  // bus->AirDensity *= 0.5f;
34  }
35  else
36  {
37  engineStart = true;
38  }
39  bus->commandStream.erase(bus->commandStream.begin() + i);
40  }
41  }
42  else if (command.name == "EngineStop")
43  {
44  if (command.ivalue == engineOrdinal)
45  {
47  {
48  engineStart = engineRun = false;
49  }
50  else
51  {
52  engineStopError = true;
53  engineStopTimer = 0.0f;
54  }
55  bus->commandStream.erase(bus->commandStream.begin() + i);
56  }
57  }
58  }
59 #pragma endregion
60 
61 
62 #pragma region Game Controller Processing
63  if (flightControls.size() == 1 && flightControls.at(0)->IsActive())
66  {
68  {
70 
72  static bool moved = false;
73  if (moved || (bus->EngineThrustCommand[engineOrdinal] != 0.5f && bus->EngineThrustCommand[engineOrdinal] != 0.0f))
74  moved = true;
75  else
77  }
78  else
79  {
81  }
82 
83  *fadecFaultPtr &= ~Systems::Fault::NoSignal; // ENG 1 TLA FAIL
84  }
85  else
86  {
87  *fadecFaultPtr |= Systems::Fault::NoSignal; // ENG 1 TLA FAIL
89  }
90 #pragma endregion
91 
92 
93  // ENG 1 TLA NOT IDLE
94  // set fault on FADEC component?
96  {
97  *fadecFaultPtr |= Systems::Fault::Status1; // ENG_1_TLA_NOT_IDLE
98  }
99  else
100  {
102  *fadecFaultPtr &= ~Systems::Fault::Status1; // ENG_1_TLA_NOT_IDLE
103  }
104 
105  if (!engineStart && engineRun && !engineLit) // shit is this a FADEC fault or an N1 fault? same deal with FIRE is that an N1 fault or a fire detection system fault?!
106  {
108  }
109  else
110  {
112  }
113 
114  if (!engineStart && !engineRun)
115  {
117  }
118  else
119  {
121  }
122 
123 
124  if (engineStopError)
125  {
126  engineStopTimer += fElapsed;
127  if (engineStopTimer >= 3.0f)
128  engineStopError = false;
129  }
130 
131 
133  if (engineStart)
134  {
135  float maxEffectOfStarter = 0.5f; // 50% N2
137  // .5-0 = .5, .5-.5 = 0
138  *starterSetFloatPtr = (maxEffectOfStarter - min(maxEffectOfStarter, *N2SetFloatPtr)) / maxEffectOfStarter;
139 
140  /* *** here is the starter effectivity *** */
141  *N2SetFloatPtr += *starterGetFloatPtr * fElapsed * 0.042f;
142 
143  if (*N2GetFloatPtr >= 0.07f) // reading
144  {
146  *ignitionSetFloatPtr = 1.0f; // loop A vs. B? probably need 2 components ... per engine... fadec cycles through them
147  }
148  if (*N2GetFloatPtr > 0.5f) // reading
149  {
150  engineStart = false;
151  engineRun = true;
153  *ignitionSetFloatPtr = 0.0f; // loop A vs. B? probably need 2 components ... per engine... fadec cycles through them
154  }
155  }
156  else
157  {
158  *starterSetFloatPtr = 0.0f;
160  *ignitionSetFloatPtr = 0.0f; // loop A vs. B? probably need 2 components ... per engine... fadec cycles through them
161  }
162  bass->Play(prefix + "EngineStarter", -1.0f, *starterSetFloatPtr * 0.75f);
163 
164 
165  bass->Play(prefix + "EngineN2", *N2SetFloatPtr * 2.0f, min(*N2SetFloatPtr, 1.0f));
166 
167 
168  float airFrictionFactor = 7.0f * bus->AirDensity;
169 
172  float compressionRatio = (*N1SetFloatPtr + *N2SetFloatPtr) * 19.5f + 1.0f; // (1+1)*19.5 = 39X + 1X = 40X
173 
174  float tempN1 = powf(*N2SetFloatPtr, 2.7f); // find N1 speed based on N2 (the sound file info we have is based on N1 speed)
176  // should be based on air density, no transfer if there is no density
177  /* *** here is the N2 to N1 and vice-versa effectivity *** */
178  float tempdiff = tempN1 - *N1SetFloatPtr;
179  float N1toN1diff = tempdiff * fElapsed * compressionRatio * airFrictionFactor;
180  *N1SetFloatPtr += N1toN1diff;
181  *N2SetFloatPtr -= min(N1toN1diff, *N2SetFloatPtr);
182 
183 
184  bass->Play(prefix + "EngineN1", *N1SetFloatPtr * 2.0f, min(*N1SetFloatPtr, 1.0f));
185 
186 
188  *N1SetFloatPtr += bus->IndicatedAirspeed * fElapsed; // 3 or 4% in a strong wind, airspeed is in m/s strong wind is 0.013
189 
191  *N2SetFloatPtr -= min(0.002f * fElapsed, fabsf(*N2SetFloatPtr)) * copysignf(1, *N2SetFloatPtr);
192  *N1SetFloatPtr -= min(0.002f * fElapsed, fabsf(*N1SetFloatPtr)) * copysignf(1, *N1SetFloatPtr); // bigger shaft, longer shaft
193 
195  /* *** here is the air friction effectivity *** */
196  *N2SetFloatPtr -= min(powf(*N2SetFloatPtr, 2.0f) * fElapsed * airFrictionFactor *compressionRatio, fabsf(*N2SetFloatPtr)) * copysignf(1, *N2SetFloatPtr);
197  *N1SetFloatPtr -= min(powf(*N1SetFloatPtr, 2.0f) * fElapsed * airFrictionFactor *compressionRatio, fabsf(*N1SetFloatPtr)) * copysignf(1, *N1SetFloatPtr);
198 
199 
201  if (*N2GetFloatPtr >= 0.2f && (engineStart || engineRun)) // fuel starts at 20% reading
202  {
203  // meters idle flow based on N2 reading
204  if (*N2GetFloatPtr < 0.5f)
205  {
207  //if (!bus->OnGround || *ITTSetFloatPtr <= 120.0f)
208  {
209  // 70 at 20% to 225 at 60% and 186.25 at 50%
210  // 0.03875 per percent above 20
211  *combustionSetFloatPtr = (*N2GetFloatPtr - 0.2f) * 600.0f + 70.0f; // get it up there a little faster hopefully with gusto
212 
213  lightOffTimer += fElapsed;
214  if (lightOffTimer > 5.0f && !engineLit) // abort start
215  {
216  engineStart = false;
217  engineRun = false;
218  lightOffTimer = 0.0f;
219  }
220  }
221  }
222  else
223  {
224  lightOffTimer = 0.0f;
225  float desired = max(0, bus->EngineThrustCommand[engineOrdinal]) * 1138.0f + 225.0f; // adjust for altitude?
226  float diff = desired - *combustionSetFloatPtr;
227  if (diff > 190) diff = 190;
228  if (diff < -190) diff = -190;
229  *combustionSetFloatPtr += diff * fElapsed;
230  }
231  }
232  else
233  {
234  *combustionSetFloatPtr = 0.0f;
235  if (*ITTSetFloatPtr < 245.0f) // autoignition temp
236  engineLit = false;
237  lightOffTimer = 0.0f;
238  }
239 
240 
242  if (*ignitionGetFloatPtr == 1.0f)
243  {
244  ignitionTimer += fElapsed;
245  if (ignitionTimer > 0.25f)
246  {
247  ignitionTimer = 0.0f;
248  bass->Play(prefix + "EngineIgniter", -1.0f, 0.1f); // pretty quiet
249  if (*combustionSetFloatPtr > 0.0f)
250  {
252  if (rand() > 16384) // 50% chance, should be dictated by damage, service life
253  {
254  engineLit = true;
255  }
256  }
257  }
258  }
259 
260 
262  if (engineLit)
263  {
264  /* *** here is the combustion effectivity *** */
265  float thrustEffectivity = 0.000475f;
266 
267  float n2increaser = *combustionSetFloatPtr * fElapsed * thrustEffectivity;
268  // not sure why but we need to ramp this N2 to N1 nonsense
269 
270  float n1rampNonsense = *combustionSetFloatPtr / 1363.0f * 0.975f + 0.025f;
271  float n1increaser = n2increaser * n1rampNonsense;
272 
273 
274  float ratio = *combustionSetFloatPtr / compressionRatio;
275  if (ratio > 35.0f)
276  {
277  // complete power loss at 40? modulate=1 at 35, 0 at 40
278  float modulate = max(40.0f - ratio, 0.0f) / 5.0f;
279  n2increaser *= modulate;
280  n1increaser *= modulate;
281  bass->Play(prefix + "EngineStall");
282  }
283 
284  *N2SetFloatPtr += n2increaser; // that fuel flow should give us 60% idle N2
285  *N1SetFloatPtr += n1increaser; // that fuel flow should give us 60% idle N2
286 
287  if (*N2GetFloatPtr > 1.01f)
288  engineRun = false;
289 
290  // so if the oncoming pressure is too low for the combustion chamber pressure
291 
292  // so there is a maximum pressure you can create in the combustion chamber before you overcome the pressure coming from the n1/n2
293  // if you exceed that then you get a compressor stall/surge and lose power and damage n1 and n2
294  // surge?
295  // i am thinking surge comes out of high pressure turbine ... the combustion flow goes forward instead of back
296 
297 
298  // loudness based on N1?
299  //*N1SetFloatPtr *= 0.875f;
300  //*N2SetFloatPtr *= 0.875f;
301  }
302  bass->Play(prefix + "EngineJet", -1.0f, *combustionSetFloatPtr / 1363.0f);
303 
304 
305  // airspeed to combustion is temp ramp
306  // compression will raise the temperature
307  float desiredTemp = bus->OATorTAT * compressionRatio;
308  // combustion will raise the temperature
309  // for some reason I am 90 degrees off at 25% and perfect on the high end
310  float tempeff;
311  if (*N1SetFloatPtr < 0.25f)
312  tempeff = powf(*N1SetFloatPtr / 0.25f, 0.0001f) * 95.0f;
313  else
314  tempeff = (1.0f - *N1SetFloatPtr) / 0.75f * 95.0f;
315  float combustTemp = desiredTemp + *combustionSetFloatPtr * 1.45f + tempeff; // max adiabatic temp 2,230 1.636
316  if (combustTemp > *ITTSetFloatPtr)
317  *ITTSetFloatPtr += (combustTemp - *ITTSetFloatPtr) * fElapsed; // not dependant on airdensity because we are filling the chamber with fuel/oxidizer regardless?
318 
319  // we are actually injecting ambient air into the combustion chamber at a rate of N1
320  *ITTSetFloatPtr += (desiredTemp - *ITTSetFloatPtr) * fElapsed * airFrictionFactor * powf(*N1SetFloatPtr, 2.0f) * 80.0f;
321 
322 
323  float radiant = 0.1f; // 1°C per second
324  *ITTSetFloatPtr += (desiredTemp - *ITTSetFloatPtr) * fElapsed * airFrictionFactor * radiant;
325 
326 
328 
329 
343 
344 
347 
354 
359 
361 }
362 
363 Fadec::Fadec(Bus* prmBus, int prmSoundDevice, int prmAxis, int prmOrdinal, std::string prmPrefix) : Module(prmBus)
364 {
365  soundDevice = prmSoundDevice;
366  joystickAxis = prmAxis;
367  engineOrdinal = prmOrdinal;
368  prefix = prmPrefix;
369 
371 }
372 
373 void Fadec::Initialize(Devices::Joystick* prmFlightControl, Devices::Bass* prmBass)
374 {
375  if (prmFlightControl)
376  flightControls.push_back(prmFlightControl);
377  if (prmBass)
378  bass = prmBass;
379 }
380 
381 void Fadec::ConnectComponent(std::string type, std::string guid)
382 {
383  if (type == "FADEC")
384  {
386  fadecGetFloatPtr = bus->GetComponentCurrentStatePtr(guid);
388  }
389  if (type == "Starter")
390  {
392  // we set based on differential ... but then actual work accomplished for the noise level is based on damage...?
393  starterGetFloatPtr = bus->GetComponentCurrentStatePtr(guid);
394  }
395  if (type == "N2")
396  {
398  // not sure how to reconcile differcen between N2 set and get ... there are more complicated things going on than just the speed (core lock fault?)
399  N2GetFloatPtr = bus->GetComponentCurrentStatePtr(guid);
400  }
401  if (type == "N1")
402  {
404  // So we drive the N1 directly ... but it can still run faults which will jack with the indication... and anything that INDICATION is used for in here must use a GET pointer too
405  N1FaultPtr = (DWORD*)bus->GetComponentFaultStatePtr(guid);
406  }
407  if (type == "IgniterA")
408  {
410  ignitionGetFloatPtr = bus->GetComponentCurrentStatePtr(guid);
411  }
412  if (type == "Combustion")
413  {
415  // starter to N2 ramp - the further apart they are the more noise the starter makes, more load, more heat
416  // N2 is in percentage of max, N2 doesn't affect starter if N2 is faster than 50%
417  }
418  if (type == "ITT")
419  {
422  }
423 }
424 
425 /*
426 CAS messages
427 http://screencast.com/t/HTZAvzFF
428 http://screencast.com/t/74G6ns10VFtA
429 */
Devices::Bass * bass
Definition: Module.h:111
float EngineThrustCommand[enginesC]
(57) Thrust command (when an information source is installed);
Definition: Bus.h:231
float EngineThrustOutput[enginesC]
(9) Thrust/power of each engine—primary flight crew reference;
Definition: Bus.h:55
void FrameMove(float fElapsedTime) override
http://aviation.stackexchange.com/questions/1959/how-are-turbine-engines-started
Definition: FADEC.cpp:5
std::vector< Command > commandStream
Definition: Bus.h:20
bool engineRun
Definition: Module.h:114
Systems::Fault * GetComponentFaultStatePtr(std::string guidStr)
Definition: Bus.cpp:141
std::string prefix
Definition: Module.h:138
float * ignitionSetFloatPtr
Definition: Module.h:129
float * ITTSetFloatPtr
Definition: Module.h:132
float DesiredThrust
Definition: Bus.h:125
These have to be in this order.
Definition: BaseComponent.h:12
DWORD * N1FaultPtr
Definition: Module.h:127
float IndicatedAirspeed
(3) Indicated Airspeed in km per second?
Definition: Bus.h:42
bool engineStopError
Definition: Module.h:118
int joystickAxis
Definition: Module.h:136
Fadec(Bus *prmBus, int prmSoundDevice, int prmAxis, int prmOrdinal, std::string prmPrefix)
Definition: FADEC.cpp:363
float ignitionTimer
Definition: Module.h:117
bool engineLit
Definition: Module.h:115
float AirDensity
Definition: Bus.h:276
bool AutothrottleEngaged
Definition: Bus.h:81
Definition: Bus.h:12
std::string name
command name
Definition: Command.h:11
float * N1SetFloatPtr
Definition: Module.h:128
float OATorTAT
(24) Outside or total air temperature;
Definition: Bus.h:75
float * starterSetFloatPtr
Definition: Module.h:123
void Initialize(Devices::Joystick *prmFlightControl, Devices::Bass *prmBass)
Definition: FADEC.cpp:373
void ConnectComponent(std::string name, std::string guid)
Definition: FADEC.cpp:381
DWORD * fadecFaultPtr
Definition: Module.h:121
Abstract base class for modules By definition, instruments don&#39;t do any of the work (they don&#39;t modif...
Definition: Module.h:11
bool engineStart
Definition: Module.h:113
float * starterGetFloatPtr
Definition: Module.h:124
float engineStopTimer
Definition: Module.h:119
Definition: Command.h:5
float * fadecGetFloatPtr
Definition: Module.h:122
struct Bus::Afcs AFCS
Bus * bus
Definition: Module.h:17
float * N2GetFloatPtr
Definition: Module.h:126
float * N2SetFloatPtr
Definition: Module.h:125
void Play(std::string trigger, float freq=-1.0f, float vol=1.0f)
Definition: BASS.cpp:126
int ivalue
Definition: Command.h:21
float * GetComponentSetStatePtr(std::string guidStr)
Definition: Bus.cpp:109
float * combustionSetFloatPtr
Definition: Module.h:131
int soundDevice
Definition: Module.h:135
float delay
wait number of seconds before executing command
Definition: Command.h:8
std::vector< Devices::Joystick * > flightControls
Definition: Module.h:110
float * ignitionGetFloatPtr
Definition: Module.h:130
int engineOrdinal
Definition: Module.h:137
float lightOffTimer
Definition: Module.h:116