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
FGFCS.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGFCS.cpp
4 Author: Jon Berndt
5 Date started: 12/12/98
6 Purpose: Model the flight controls
7 Called by: FDMExec
8
9 ------------- Copyright (C) 1999 Jon S. Berndt (jon@jsbsim.org) -------------
10
11 This program is free software; you can redistribute it and/or modify it under
12 the terms of the GNU Lesser General Public License as published by the Free
13 Software Foundation; either version 2 of the License, or (at your option) any
14 later version.
15
16 This program is distributed in the hope that it will be useful, but WITHOUT
17 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
18 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
19 details.
20
21 You should have received a copy of the GNU Lesser General Public License along
22 with this program; if not, write to the Free Software Foundation, Inc., 59
23 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
24
25 Further information about the GNU Lesser General Public License can also be
26 found on the world wide web at http://www.gnu.org.
27
28FUNCTIONAL DESCRIPTION
29--------------------------------------------------------------------------------
30This class models the flight controls for a specific airplane
31
32HISTORY
33--------------------------------------------------------------------------------
3412/12/98 JSB Created
35
36%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
37INCLUDES
38%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
39
40#include <iomanip>
41#include <array>
42
43#include "FGFCS.h"
44#include "input_output/FGModelLoader.h"
45
46#include "models/flight_control/FGFilter.h"
47#include "models/flight_control/FGDeadBand.h"
48#include "models/flight_control/FGGain.h"
49#include "models/flight_control/FGPID.h"
50#include "models/flight_control/FGSwitch.h"
51#include "models/flight_control/FGSummer.h"
52#include "models/flight_control/FGKinemat.h"
53#include "models/flight_control/FGFCSFunction.h"
54#include "models/flight_control/FGActuator.h"
55#include "models/flight_control/FGAccelerometer.h"
56#include "models/flight_control/FGMagnetometer.h"
57#include "models/flight_control/FGGyro.h"
58#include "models/flight_control/FGWaypoint.h"
59#include "models/flight_control/FGAngles.h"
60#include "models/flight_control/FGDistributor.h"
61#include "models/flight_control/FGLinearActuator.h"
62
63#include "FGFCSChannel.h"
64
65using namespace std;
66
67namespace JSBSim {
68
69/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
70CLASS IMPLEMENTATION
71%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
72
73FGFCS::FGFCS(FGFDMExec* fdm) : FGModel(fdm), ChannelRate(1)
74{
75 int i;
76 Name = "FGFCS";
77 systype = stFCS;
78
79 DaCmd = DeCmd = DrCmd = DfCmd = DsbCmd = DspCmd = 0;
80 PTrimCmd = YTrimCmd = RTrimCmd = 0.0;
81 GearCmd = GearPos = 1; // default to gear down
82 BrakePos.resize(FGLGear::bgNumBrakeGroups);
83 TailhookPos = WingFoldPos = 0.0;
84
85 bind();
86 for (i=0;i<NForms;i++) {
87 DePos[i] = DaLPos[i] = DaRPos[i] = DrPos[i] = 0.0;
88 DfPos[i] = DsbPos[i] = DspPos[i] = 0.0;
89 }
90
91 Debug(0);
92}
93
94//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
95
97{
98 ThrottleCmd.clear();
99 ThrottlePos.clear();
100 MixtureCmd.clear();
101 MixturePos.clear();
102 PropAdvanceCmd.clear();
103 PropAdvance.clear();
104 PropFeatherCmd.clear();
105 PropFeather.clear();
106
107 unsigned int i;
108
109 for (i=0;i<SystemChannels.size();i++) delete SystemChannels[i];
110 SystemChannels.clear();
111
112 Debug(1);
113}
114
115//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
116
117bool FGFCS::InitModel(void)
118{
119 if (!FGModel::InitModel()) return false;
120
121 unsigned int i;
122
123 for (i=0; i<ThrottlePos.size(); i++) ThrottlePos[i] = 0.0;
124 for (i=0; i<MixturePos.size(); i++) MixturePos[i] = 0.0;
125 for (i=0; i<ThrottleCmd.size(); i++) ThrottleCmd[i] = 0.0;
126 for (i=0; i<MixtureCmd.size(); i++) MixtureCmd[i] = 0.0;
127 for (i=0; i<PropAdvance.size(); i++) PropAdvance[i] = 0.0;
128 for (i=0; i<PropFeather.size(); i++) PropFeather[i] = 0.0;
129
130 DaCmd = DeCmd = DrCmd = DfCmd = DsbCmd = DspCmd = 0;
131 PTrimCmd = YTrimCmd = RTrimCmd = 0.0;
132 TailhookPos = WingFoldPos = 0.0;
133
134 for (i=0;i<NForms;i++) {
135 DePos[i] = DaLPos[i] = DaRPos[i] = DrPos[i] = 0.0;
136 DfPos[i] = DsbPos[i] = DspPos[i] = 0.0;
137 }
138
139 // Reset the channels components.
140 for (unsigned int i=0; i<SystemChannels.size(); i++) SystemChannels[i]->Reset();
141
142 return true;
143}
144
145//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
146// Notes: In this logic the default engine commands are set. This is simply a
147// sort of safe-mode method in case the user has not defined control laws for
148// throttle, mixture, and prop-advance. The throttle, mixture, and prop advance
149// positions are set equal to the respective commands. Any control logic that is
150// actually present in the flight_control or autopilot section will override
151// these simple assignments.
152
153bool FGFCS::Run(bool Holding)
154{
155 unsigned int i;
156
157 if (FGModel::Run(Holding)) return true; // fast exit if nothing to do
158 if (Holding) return false;
159
160 RunPreFunctions();
161
162 for (i=0; i<ThrottlePos.size(); i++) ThrottlePos[i] = ThrottleCmd[i];
163 for (i=0; i<MixturePos.size(); i++) MixturePos[i] = MixtureCmd[i];
164 for (i=0; i<PropAdvance.size(); i++) PropAdvance[i] = PropAdvanceCmd[i];
165 for (i=0; i<PropFeather.size(); i++) PropFeather[i] = PropFeatherCmd[i];
166
167 // Execute system channels in order
168 for (i=0; i<SystemChannels.size(); i++) {
169 if (debug_lvl & 4) cout << " Executing System Channel: " << SystemChannels[i]->GetName() << endl;
170 ChannelRate = SystemChannels[i]->GetRate();
171 SystemChannels[i]->Execute();
172 }
173 ChannelRate = 1;
174
175 RunPostFunctions();
176
177 return false;
178}
179
180//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
181
182void FGFCS::SetDaLPos( int form , double pos )
183{
184 switch(form) {
185 case ofRad:
186 DaLPos[ofRad] = pos;
187 DaLPos[ofDeg] = pos*radtodeg;
188 break;
189 case ofDeg:
190 DaLPos[ofRad] = pos*degtorad;
191 DaLPos[ofDeg] = pos;
192 break;
193 case ofNorm:
194 DaLPos[ofNorm] = pos;
195 }
196 DaLPos[ofMag] = fabs(DaLPos[ofRad]);
197}
198
199//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
200
201void FGFCS::SetDaRPos( int form , double pos )
202{
203 switch(form) {
204 case ofRad:
205 DaRPos[ofRad] = pos;
206 DaRPos[ofDeg] = pos*radtodeg;
207 break;
208 case ofDeg:
209 DaRPos[ofRad] = pos*degtorad;
210 DaRPos[ofDeg] = pos;
211 break;
212 case ofNorm:
213 DaRPos[ofNorm] = pos;
214 }
215 DaRPos[ofMag] = fabs(DaRPos[ofRad]);
216}
217
218//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
219
220void FGFCS::SetDePos( int form , double pos )
221{
222 switch(form) {
223 case ofRad:
224 DePos[ofRad] = pos;
225 DePos[ofDeg] = pos*radtodeg;
226 break;
227 case ofDeg:
228 DePos[ofRad] = pos*degtorad;
229 DePos[ofDeg] = pos;
230 break;
231 case ofNorm:
232 DePos[ofNorm] = pos;
233 }
234 DePos[ofMag] = fabs(DePos[ofRad]);
235}
236
237//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
238
239void FGFCS::SetDrPos( int form , double pos )
240{
241 switch(form) {
242 case ofRad:
243 DrPos[ofRad] = pos;
244 DrPos[ofDeg] = pos*radtodeg;
245 break;
246 case ofDeg:
247 DrPos[ofRad] = pos*degtorad;
248 DrPos[ofDeg] = pos;
249 break;
250 case ofNorm:
251 DrPos[ofNorm] = pos;
252 }
253 DrPos[ofMag] = fabs(DrPos[ofRad]);
254}
255
256//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
257
258void FGFCS::SetDfPos( int form , double pos )
259{
260 switch(form) {
261 case ofRad:
262 DfPos[ofRad] = pos;
263 DfPos[ofDeg] = pos*radtodeg;
264 break;
265 case ofDeg:
266 DfPos[ofRad] = pos*degtorad;
267 DfPos[ofDeg] = pos;
268 break;
269 case ofNorm:
270 DfPos[ofNorm] = pos;
271 }
272 DfPos[ofMag] = fabs(DfPos[ofRad]);
273}
274
275//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
276
277void FGFCS::SetDsbPos( int form , double pos )
278{
279 switch(form) {
280 case ofRad:
281 DsbPos[ofRad] = pos;
282 DsbPos[ofDeg] = pos*radtodeg;
283 break;
284 case ofDeg:
285 DsbPos[ofRad] = pos*degtorad;
286 DsbPos[ofDeg] = pos;
287 break;
288 case ofNorm:
289 DsbPos[ofNorm] = pos;
290 }
291 DsbPos[ofMag] = fabs(DsbPos[ofRad]);
292}
293
294//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
295
296void FGFCS::SetDspPos( int form , double pos )
297{
298 switch(form) {
299 case ofRad:
300 DspPos[ofRad] = pos;
301 DspPos[ofDeg] = pos*radtodeg;
302 break;
303 case ofDeg:
304 DspPos[ofRad] = pos*degtorad;
305 DspPos[ofDeg] = pos;
306 break;
307 case ofNorm:
308 DspPos[ofNorm] = pos;
309 }
310 DspPos[ofMag] = fabs(DspPos[ofRad]);
311}
312
313//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
314
315void FGFCS::SetThrottleCmd(int engineNum, double setting)
316{
317 if (engineNum < (int)ThrottleCmd.size()) {
318 if (engineNum < 0) {
319 for (unsigned int ctr=0; ctr<ThrottleCmd.size(); ctr++)
320 ThrottleCmd[ctr] = setting;
321 } else {
322 ThrottleCmd[engineNum] = setting;
323 }
324 } else {
325 cerr << "Throttle " << engineNum << " does not exist! " << ThrottleCmd.size()
326 << " engines exist, but attempted throttle command is for engine "
327 << engineNum << endl;
328 }
329}
330
331//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
332
333void FGFCS::SetThrottlePos(int engineNum, double setting)
334{
335 if (engineNum < (int)ThrottlePos.size()) {
336 if (engineNum < 0) {
337 for (unsigned int ctr=0; ctr<ThrottlePos.size(); ctr++)
338 ThrottlePos[ctr] = setting;
339 } else {
340 ThrottlePos[engineNum] = setting;
341 }
342 } else {
343 cerr << "Throttle " << engineNum << " does not exist! " << ThrottlePos.size()
344 << " engines exist, but attempted throttle position setting is for engine "
345 << engineNum << endl;
346 }
347}
348
349//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
350
351double FGFCS::GetThrottleCmd(int engineNum) const
352{
353 if (engineNum < (int)ThrottleCmd.size()) {
354 if (engineNum < 0) {
355 cerr << "Cannot get throttle value for ALL engines" << endl;
356 } else {
357 return ThrottleCmd[engineNum];
358 }
359 } else {
360 cerr << "Throttle " << engineNum << " does not exist! " << ThrottleCmd.size()
361 << " engines exist, but throttle setting for engine " << engineNum
362 << " is selected" << endl;
363 }
364 return 0.0;
365}
366
367//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
368
369double FGFCS::GetThrottlePos(int engineNum) const
370{
371 if (engineNum < (int)ThrottlePos.size()) {
372 if (engineNum < 0) {
373 cerr << "Cannot get throttle value for ALL engines" << endl;
374 } else {
375 return ThrottlePos[engineNum];
376 }
377 } else {
378 cerr << "Throttle " << engineNum << " does not exist! " << ThrottlePos.size()
379 << " engines exist, but attempted throttle position setting is for engine "
380 << engineNum << endl;
381 }
382 return 0.0;
383}
384
385//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
386
387void FGFCS::SetMixtureCmd(int engineNum, double setting)
388{
389 if (engineNum < (int)MixtureCmd.size()) {
390 if (engineNum < 0) {
391 for (unsigned int ctr=0; ctr<MixtureCmd.size(); ctr++)
392 MixtureCmd[ctr] = setting;
393 } else {
394 MixtureCmd[engineNum] = setting;
395 }
396 }
397}
398
399//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
400
401void FGFCS::SetMixturePos(int engineNum, double setting)
402{
403 if (engineNum < (int)MixturePos.size()) {
404 if (engineNum < 0) {
405 for (unsigned int ctr=0; ctr<MixtureCmd.size(); ctr++)
406 MixturePos[ctr] = MixtureCmd[ctr];
407 } else {
408 MixturePos[engineNum] = setting;
409 }
410 }
411}
412
413//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
414
415void FGFCS::SetPropAdvanceCmd(int engineNum, double setting)
416{
417 if (engineNum < (int)PropAdvanceCmd.size()) {
418 if (engineNum < 0) {
419 for (unsigned int ctr=0; ctr<PropAdvanceCmd.size(); ctr++)
420 PropAdvanceCmd[ctr] = setting;
421 } else {
422 PropAdvanceCmd[engineNum] = setting;
423 }
424 }
425}
426
427//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
428
429void FGFCS::SetPropAdvance(int engineNum, double setting)
430{
431 if (engineNum < (int)PropAdvance.size()) {
432 if (engineNum < 0) {
433 for (unsigned int ctr=0; ctr<PropAdvanceCmd.size(); ctr++)
434 PropAdvance[ctr] = PropAdvanceCmd[ctr];
435 } else {
436 PropAdvance[engineNum] = setting;
437 }
438 }
439}
440
441//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
442
443void FGFCS::SetFeatherCmd(int engineNum, bool setting)
444{
445 if (engineNum < (int)PropFeatherCmd.size()) {
446 if (engineNum < 0) {
447 for (unsigned int ctr=0; ctr<PropFeatherCmd.size(); ctr++)
448 PropFeatherCmd[ctr] = setting;
449 } else {
450 PropFeatherCmd[engineNum] = setting;
451 }
452 }
453}
454
455//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
456
457void FGFCS::SetPropFeather(int engineNum, bool setting)
458{
459 if (engineNum < (int)PropFeather.size()) {
460 if (engineNum < 0) {
461 for (unsigned int ctr=0; ctr<PropFeatherCmd.size(); ctr++)
462 PropFeather[ctr] = PropFeatherCmd[ctr];
463 } else {
464 PropFeather[engineNum] = setting;
465 }
466 }
467}
468
469//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
470
471bool FGFCS::Load(Element* document)
472{
473 if (document->GetName() == "autopilot") {
474 Name = "Autopilot: ";
475 systype = stAutoPilot;
476 } else if (document->GetName() == "flight_control") {
477 Name = "FCS: ";
478 systype = stFCS;
479 } else if (document->GetName() == "system") {
480 Name = "System: ";
481 systype = stSystem;
482 }
483
484 // Load interface properties from document
485 if (!FGModel::Upload(document, true))
486 return false;
487
488 Name += document->GetAttributeValue("name");
489
490 Debug(2);
491
492 Element* channel_element = document->FindElement("channel");
493
494 while (channel_element) {
495
496 FGFCSChannel* newChannel = 0;
497
498 string sOnOffProperty = channel_element->GetAttributeValue("execute");
499 string sChannelName = channel_element->GetAttributeValue("name");
500
501 if (!channel_element->GetAttributeValue("execrate").empty())
502 ChannelRate = channel_element->GetAttributeValueAsNumber("execrate");
503 else
504 ChannelRate = 1;
505
506 if (sOnOffProperty.length() > 0) {
507 FGPropertyNode* OnOffPropertyNode = PropertyManager->GetNode(sOnOffProperty);
508 if (OnOffPropertyNode == 0) {
509 cerr << channel_element->ReadFrom() << highint << fgred
510 << "The On/Off property, " << sOnOffProperty << " specified for channel "
511 << channel_element->GetAttributeValue("name") << " is undefined or not "
512 << "understood. The simulation will abort" << reset << endl;
513 throw("Bad system definition");
514 } else
515 newChannel = new FGFCSChannel(this, sChannelName, ChannelRate,
516 OnOffPropertyNode);
517 } else
518 newChannel = new FGFCSChannel(this, sChannelName, ChannelRate);
519
520 SystemChannels.push_back(newChannel);
521
522 if (debug_lvl > 0)
523 cout << endl << highint << fgblue << " Channel "
524 << normint << channel_element->GetAttributeValue("name") << reset << endl;
525
526 Element* component_element = channel_element->GetElement();
527 while (component_element) {
528 try {
529 if ((component_element->GetName() == string("lag_filter")) ||
530 (component_element->GetName() == string("lead_lag_filter")) ||
531 (component_element->GetName() == string("washout_filter")) ||
532 (component_element->GetName() == string("second_order_filter")) )
533 {
534 newChannel->Add(new FGFilter(this, component_element));
535 } else if ((component_element->GetName() == string("pure_gain")) ||
536 (component_element->GetName() == string("scheduled_gain")) ||
537 (component_element->GetName() == string("aerosurface_scale")))
538 {
539 newChannel->Add(new FGGain(this, component_element));
540 } else if (component_element->GetName() == string("summer")) {
541 newChannel->Add(new FGSummer(this, component_element));
542 } else if (component_element->GetName() == string("deadband")) {
543 newChannel->Add(new FGDeadBand(this, component_element));
544 } else if (component_element->GetName() == string("switch")) {
545 newChannel->Add(new FGSwitch(this, component_element));
546 } else if (component_element->GetName() == string("kinematic")) {
547 newChannel->Add(new FGKinemat(this, component_element));
548 } else if (component_element->GetName() == string("fcs_function")) {
549 newChannel->Add(new FGFCSFunction(this, component_element));
550 } else if (component_element->GetName() == string("pid")) {
551 newChannel->Add(new FGPID(this, component_element));
552 } else if (component_element->GetName() == string("integrator")) {
553 // <integrator> is equivalent to <pid type="trap">
554 Element* c1_el = component_element->FindElement("c1");
555 if (!c1_el) {
556 cerr << component_element->ReadFrom();
557 throw("INTEGRATOR component " + component_element->GetAttributeValue("name")
558 + " does not provide the parameter <c1>");
559 }
560 c1_el->ChangeName("ki");
561 if (!c1_el->HasAttribute("type"))
562 c1_el->AddAttribute("type", "trap");
563 newChannel->Add(new FGPID(this, component_element));
564 } else if (component_element->GetName() == string("actuator")) {
565 newChannel->Add(new FGActuator(this, component_element));
566 } else if (component_element->GetName() == string("sensor")) {
567 newChannel->Add(new FGSensor(this, component_element));
568 } else if (component_element->GetName() == string("accelerometer")) {
569 newChannel->Add(new FGAccelerometer(this, component_element));
570 } else if (component_element->GetName() == string("magnetometer")) {
571 newChannel->Add(new FGMagnetometer(this, component_element));
572 } else if (component_element->GetName() == string("gyro")) {
573 newChannel->Add(new FGGyro(this, component_element));
574 } else if ((component_element->GetName() == string("waypoint_heading")) ||
575 (component_element->GetName() == string("waypoint_distance")))
576 {
577 newChannel->Add(new FGWaypoint(this, component_element));
578 } else if (component_element->GetName() == string("angle")) {
579 newChannel->Add(new FGAngles(this, component_element));
580 } else if (component_element->GetName() == string("distributor")) {
581 newChannel->Add(new FGDistributor(this, component_element));
582 } else if (component_element->GetName() == string("linear_actuator")) {
583 newChannel->Add(new FGLinearActuator(this, component_element));
584 } else {
585 cerr << "Unknown FCS component: " << component_element->GetName() << endl;
586 }
587 } catch(string& s) {
588 cerr << highint << fgred << endl << " " << s << endl;
589 cerr << reset << endl;
590 return false;
591 }
592 component_element = channel_element->GetNextElement();
593 }
594 channel_element = document->FindNextElement("channel");
595 }
596
597 PostLoad(document, FDMExec);
598
599 return true;
600}
601
602//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
603
605{
606 return BrakePos[bg];
607}
608
609//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
610
611SGPath FGFCS::FindFullPathName(const SGPath& path) const
612{
613 SGPath name = FGModel::FindFullPathName(path);
614 if (systype != stSystem || !name.isNull()) return name;
615
616#ifdef _WIN32
617 const array<string, 1> dir_names = {"Systems"};
618#else
619 // Check alternative capitalization for case sensitive OSes.
620 const array<string, 2> dir_names = {"Systems", "systems"};
621#endif
622
623 for(const string& dir_name: dir_names) {
624 name = CheckPathName(FDMExec->GetFullAircraftPath()/dir_name, path);
625 if (!name.isNull()) return name;
626 }
627
628 return CheckPathName(FDMExec->GetSystemsPath(), path);
629}
630
631//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
632
633string FGFCS::GetComponentStrings(const string& delimiter) const
634{
635 string CompStrings = "";
636 bool firstime = true;
637 int total_count=0;
638
639 for (unsigned int i=0; i<SystemChannels.size(); i++)
640 {
641 for (unsigned int c=0; c<SystemChannels[i]->GetNumComponents(); c++)
642 {
643 if (firstime) firstime = false;
644 else CompStrings += delimiter;
645
646 CompStrings += SystemChannels[i]->GetComponent(c)->GetName();
647 total_count++;
648 }
649 }
650
651 return CompStrings;
652}
653
654//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
655
656string FGFCS::GetComponentValues(const string& delimiter) const
657{
658 std::ostringstream buf;
659
660 bool firstime = true;
661 int total_count=0;
662
663 for (unsigned int i=0; i<SystemChannels.size(); i++)
664 {
665 for (unsigned int c=0; c<SystemChannels[i]->GetNumComponents(); c++)
666 {
667 if (firstime) firstime = false;
668 else buf << delimiter;
669
670 buf << setprecision(9) << SystemChannels[i]->GetComponent(c)->GetOutput();
671 total_count++;
672 }
673 }
674
675 return buf.str();
676}
677
678//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
679
680void FGFCS::AddThrottle(void)
681{
682 ThrottleCmd.push_back(0.0);
683 ThrottlePos.push_back(0.0);
684 MixtureCmd.push_back(0.0); // assume throttle and mixture are coupled
685 MixturePos.push_back(0.0);
686 PropAdvanceCmd.push_back(0.0); // assume throttle and prop pitch are coupled
687 PropAdvance.push_back(0.0);
688 PropFeatherCmd.push_back(false);
689 PropFeather.push_back(false);
690
691 unsigned int num = (unsigned int)ThrottleCmd.size()-1;
692 bindThrottle(num);
693}
694
695//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
696
697double FGFCS::GetDt(void) const
698{
699 return FDMExec->GetDeltaT()*rate;
700}
701
702//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
703
704void FGFCS::bind(void)
705{
706 PropertyManager->Tie("fcs/aileron-cmd-norm", this, &FGFCS::GetDaCmd, &FGFCS::SetDaCmd);
707 PropertyManager->Tie("fcs/elevator-cmd-norm", this, &FGFCS::GetDeCmd, &FGFCS::SetDeCmd);
708 PropertyManager->Tie("fcs/rudder-cmd-norm", this, &FGFCS::GetDrCmd, &FGFCS::SetDrCmd);
709 PropertyManager->Tie("fcs/flap-cmd-norm", this, &FGFCS::GetDfCmd, &FGFCS::SetDfCmd);
710 PropertyManager->Tie("fcs/speedbrake-cmd-norm", this, &FGFCS::GetDsbCmd, &FGFCS::SetDsbCmd);
711 PropertyManager->Tie("fcs/spoiler-cmd-norm", this, &FGFCS::GetDspCmd, &FGFCS::SetDspCmd);
712 PropertyManager->Tie("fcs/pitch-trim-cmd-norm", this, &FGFCS::GetPitchTrimCmd, &FGFCS::SetPitchTrimCmd);
713 PropertyManager->Tie("fcs/roll-trim-cmd-norm", this, &FGFCS::GetRollTrimCmd, &FGFCS::SetRollTrimCmd);
714 PropertyManager->Tie("fcs/yaw-trim-cmd-norm", this, &FGFCS::GetYawTrimCmd, &FGFCS::SetYawTrimCmd);
715
716 PropertyManager->Tie("fcs/left-aileron-pos-rad", this, ofRad, &FGFCS::GetDaLPos, &FGFCS::SetDaLPos);
717 PropertyManager->Tie("fcs/left-aileron-pos-deg", this, ofDeg, &FGFCS::GetDaLPos, &FGFCS::SetDaLPos);
718 PropertyManager->Tie("fcs/left-aileron-pos-norm", this, ofNorm, &FGFCS::GetDaLPos, &FGFCS::SetDaLPos);
719 PropertyManager->Tie("fcs/mag-left-aileron-pos-rad", this, ofMag, &FGFCS::GetDaLPos);
720
721 PropertyManager->Tie("fcs/right-aileron-pos-rad", this, ofRad, &FGFCS::GetDaRPos, &FGFCS::SetDaRPos);
722 PropertyManager->Tie("fcs/right-aileron-pos-deg", this, ofDeg, &FGFCS::GetDaRPos, &FGFCS::SetDaRPos);
723 PropertyManager->Tie("fcs/right-aileron-pos-norm", this, ofNorm, &FGFCS::GetDaRPos, &FGFCS::SetDaRPos);
724 PropertyManager->Tie("fcs/mag-right-aileron-pos-rad", this, ofMag, &FGFCS::GetDaRPos);
725
726 PropertyManager->Tie("fcs/elevator-pos-rad", this, ofRad, &FGFCS::GetDePos, &FGFCS::SetDePos);
727 PropertyManager->Tie("fcs/elevator-pos-deg", this, ofDeg, &FGFCS::GetDePos, &FGFCS::SetDePos);
728 PropertyManager->Tie("fcs/elevator-pos-norm", this, ofNorm, &FGFCS::GetDePos, &FGFCS::SetDePos);
729 PropertyManager->Tie("fcs/mag-elevator-pos-rad", this, ofMag, &FGFCS::GetDePos);
730
731 PropertyManager->Tie("fcs/rudder-pos-rad", this,ofRad, &FGFCS::GetDrPos, &FGFCS::SetDrPos);
732 PropertyManager->Tie("fcs/rudder-pos-deg", this,ofDeg, &FGFCS::GetDrPos, &FGFCS::SetDrPos);
733 PropertyManager->Tie("fcs/rudder-pos-norm", this,ofNorm, &FGFCS::GetDrPos, &FGFCS::SetDrPos);
734 PropertyManager->Tie("fcs/mag-rudder-pos-rad", this,ofMag, &FGFCS::GetDrPos);
735
736 PropertyManager->Tie("fcs/flap-pos-rad", this,ofRad, &FGFCS::GetDfPos, &FGFCS::SetDfPos);
737 PropertyManager->Tie("fcs/flap-pos-deg", this,ofDeg, &FGFCS::GetDfPos, &FGFCS::SetDfPos);
738 PropertyManager->Tie("fcs/flap-pos-norm", this,ofNorm, &FGFCS::GetDfPos, &FGFCS::SetDfPos);
739
740 PropertyManager->Tie("fcs/speedbrake-pos-rad", this,ofRad, &FGFCS::GetDsbPos, &FGFCS::SetDsbPos);
741 PropertyManager->Tie("fcs/speedbrake-pos-deg", this,ofDeg, &FGFCS::GetDsbPos, &FGFCS::SetDsbPos);
742 PropertyManager->Tie("fcs/speedbrake-pos-norm", this,ofNorm, &FGFCS::GetDsbPos, &FGFCS::SetDsbPos);
743 PropertyManager->Tie("fcs/mag-speedbrake-pos-rad", this,ofMag, &FGFCS::GetDsbPos);
744
745 PropertyManager->Tie("fcs/spoiler-pos-rad", this, ofRad, &FGFCS::GetDspPos, &FGFCS::SetDspPos);
746 PropertyManager->Tie("fcs/spoiler-pos-deg", this, ofDeg, &FGFCS::GetDspPos, &FGFCS::SetDspPos);
747 PropertyManager->Tie("fcs/spoiler-pos-norm", this, ofNorm, &FGFCS::GetDspPos, &FGFCS::SetDspPos);
748 PropertyManager->Tie("fcs/mag-spoiler-pos-rad", this, ofMag, &FGFCS::GetDspPos);
749
750 PropertyManager->Tie("gear/gear-pos-norm", this, &FGFCS::GetGearPos, &FGFCS::SetGearPos);
751 PropertyManager->Tie("gear/gear-cmd-norm", this, &FGFCS::GetGearCmd, &FGFCS::SetGearCmd);
752 PropertyManager->Tie("fcs/left-brake-cmd-norm", this, &FGFCS::GetLBrake, &FGFCS::SetLBrake);
753 PropertyManager->Tie("fcs/right-brake-cmd-norm", this, &FGFCS::GetRBrake, &FGFCS::SetRBrake);
754 PropertyManager->Tie("fcs/center-brake-cmd-norm", this, &FGFCS::GetCBrake, &FGFCS::SetCBrake);
755
756 PropertyManager->Tie("gear/tailhook-pos-norm", this, &FGFCS::GetTailhookPos, &FGFCS::SetTailhookPos);
757 PropertyManager->Tie("fcs/wing-fold-pos-norm", this, &FGFCS::GetWingFoldPos, &FGFCS::SetWingFoldPos);
758 PropertyManager->Tie("simulation/channel-dt", this, &FGFCS::GetChannelDeltaT);
759}
760
761//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
762// Technically, this function should probably bind propulsion type specific controls
763// rather than mixture and prop-advance.
764
765void FGFCS::bindThrottle(unsigned int num)
766{
767 string tmp;
768
769 tmp = CreateIndexedPropertyName("fcs/throttle-cmd-norm", num);
770 PropertyManager->Tie( tmp.c_str(), this, num, &FGFCS::GetThrottleCmd,
772 tmp = CreateIndexedPropertyName("fcs/throttle-pos-norm", num);
773 PropertyManager->Tie( tmp.c_str(), this, num, &FGFCS::GetThrottlePos,
775 tmp = CreateIndexedPropertyName("fcs/mixture-cmd-norm", num);
776 PropertyManager->Tie( tmp.c_str(), this, num, &FGFCS::GetMixtureCmd,
778 tmp = CreateIndexedPropertyName("fcs/mixture-pos-norm", num);
779 PropertyManager->Tie( tmp.c_str(), this, num, &FGFCS::GetMixturePos,
781 tmp = CreateIndexedPropertyName("fcs/advance-cmd-norm", num);
782 PropertyManager->Tie( tmp.c_str(), this, num, &FGFCS::GetPropAdvanceCmd,
784 tmp = CreateIndexedPropertyName("fcs/advance-pos-norm", num);
785 PropertyManager->Tie( tmp.c_str(), this, num, &FGFCS::GetPropAdvance,
787 tmp = CreateIndexedPropertyName("fcs/feather-cmd-norm", num);
788 PropertyManager->Tie( tmp.c_str(), this, num, &FGFCS::GetFeatherCmd,
790 tmp = CreateIndexedPropertyName("fcs/feather-pos-norm", num);
791 PropertyManager->Tie( tmp.c_str(), this, num, &FGFCS::GetPropFeather,
793}
794
795//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
796// The bitmasked value choices are as follows:
797// unset: In this case (the default) JSBSim would only print
798// out the normally expected messages, essentially echoing
799// the config files as they are read. If the environment
800// variable is not set, debug_lvl is set to 1 internally
801// 0: This requests JSBSim not to output any messages
802// whatsoever.
803// 1: This value explicity requests the normal JSBSim
804// startup messages
805// 2: This value asks for a message to be printed out when
806// a class is instantiated
807// 4: When this value is set, a message is displayed when a
808// FGModel object executes its Run() method
809// 8: When this value is set, various runtime state variables
810// are printed out periodically
811// 16: When set various parameters are sanity checked and
812// a message is printed out when they go out of bounds
813
814void FGFCS::Debug(int from)
815{
816 if (debug_lvl <= 0) return;
817
818 if (debug_lvl & 1) { // Standard console startup message output
819 if (from == 2) { // Loader
820 cout << endl << " " << Name << endl;
821 }
822 }
823 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
824 if (from == 0) cout << "Instantiated: FGFCS" << endl;
825 if (from == 1) cout << "Destroyed: FGFCS" << endl;
826 }
827 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
828 }
829 if (debug_lvl & 8 ) { // Runtime state variables
830 }
831 if (debug_lvl & 16) { // Sanity checking
832 }
833 if (debug_lvl & 64) {
834 if (from == 0) { // Constructor
835 }
836 }
837}
838
839}
Element * FindElement(const std::string &el="")
Searches for a specified element.
const std::string & GetName(void) const
Retrieves the element name.
double GetAttributeValueAsNumber(const std::string &key)
Retrieves an attribute value as a double precision real number.
Element * GetElement(unsigned int el=0)
Returns a pointer to the element requested by index.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
void AddAttribute(const std::string &name, const std::string &value)
Stores an attribute belonging to this element.
std::string ReadFrom(void) const
Return a string that contains a description of the location where the current XML element was read fr...
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
bool HasAttribute(const std::string &key)
Determines if an element has the supplied attribute.
Element * GetNextElement(void)
Returns a pointer to the next element in the list.
Encapsulates a Accelerometer component for the flight control system.
Encapsulates an Actuator component for the flight control system.
Definition FGActuator.h:128
Provides a way to determine the smallest included angle.
Definition FGAngles.h:78
Models a deadband object.
Definition FGDeadBand.h:84
Encapsulates a distributor for the flight control system.
void Add(FGFCSComponent *comp)
Adds a component to a channel.
Models a FCSFunction object.
void SetThrottlePos(int engine, double cmd)
Sets the actual throttle setting for the specified engine.
Definition FGFCS.cpp:333
void SetPitchTrimCmd(double cmd)
Sets the pitch trim command.
Definition FGFCS.h:410
void SetDaLPos(int form, double pos)
Sets the left aileron position.
Definition FGFCS.cpp:182
bool Load(Element *el) override
Loads the Flight Control System.
Definition FGFCS.cpp:471
void SetGearPos(double gearpos)
Set the gear extend/retract position, defaults to down.
Definition FGFCS.h:498
double GetCBrake(void) const
Gets the center brake.
Definition FGFCS.h:550
void SetDsbPos(int form, double pos)
Sets the speedbrake position.
Definition FGFCS.cpp:277
double GetLBrake(void) const
Gets the left brake.
Definition FGFCS.h:542
double GetDfPos(int form=ofRad) const
Gets the flaps position.
Definition FGFCS.h:319
void SetDsbCmd(double cmd)
Sets the speedbrake command.
Definition FGFCS.h:402
bool GetFeatherCmd(int engine) const
Gets the prop feather command.
Definition FGFCS.h:265
double GetDfCmd(void) const
Gets the flaps command.
Definition FGFCS.h:229
double GetDeCmd(void) const
Gets the elevator command.
Definition FGFCS.h:217
std::string GetComponentStrings(const std::string &delimiter) const
Retrieves all component names for inclusion in output stream.
Definition FGFCS.cpp:633
void SetMixtureCmd(int engine, double cmd)
Sets the mixture command for the specified engine.
Definition FGFCS.cpp:387
double GetTailhookPos(void) const
Gets the tailhook position (0 up, 1 down)
Definition FGFCS.h:346
void SetPropFeather(int engine, bool cmd)
Sets the actual prop feather setting for the specified engine.
Definition FGFCS.cpp:457
double GetPropAdvanceCmd(int engine) const
Gets the prop pitch command.
Definition FGFCS.h:260
void SetFeatherCmd(int engine, bool cmd)
Sets the propeller feather command for the specified engine.
Definition FGFCS.cpp:443
void SetDeCmd(double cmd)
Sets the elevator command.
Definition FGFCS.h:386
double GetDePos(int form=ofRad) const
Gets the elevator position.
Definition FGFCS.h:299
void SetDrPos(int form, double pos)
Sets the rudder position.
Definition FGFCS.cpp:239
void SetMixturePos(int engine, double cmd)
Sets the actual mixture setting for the specified engine.
Definition FGFCS.cpp:401
void SetDaCmd(double cmd)
Sets the aileron command.
Definition FGFCS.h:382
double GetDsbPos(int form=ofRad) const
Gets the speedbrake position.
Definition FGFCS.h:309
double GetRBrake(void) const
Gets the right brake.
Definition FGFCS.h:546
void SetDaRPos(int form, double pos)
Sets the right aileron position.
Definition FGFCS.cpp:201
void SetDePos(int form, double pos)
Sets the elevator position.
Definition FGFCS.cpp:220
void SetRBrake(double cmd)
Sets the right brake group.
Definition FGFCS.h:527
void SetDrCmd(double cmd)
Sets the rudder command.
Definition FGFCS.h:390
double GetDrCmd(void) const
Gets the rudder command.
Definition FGFCS.h:221
FGFCS(FGFDMExec *)
Constructor.
Definition FGFCS.cpp:73
bool Run(bool Holding) override
Runs the Flight Controls model; called by the Executive Can pass in a value indicating if the executi...
Definition FGFCS.cpp:153
double GetDaCmd(void) const
Gets the aileron command.
Definition FGFCS.h:213
double GetDaLPos(int form=ofRad) const
Gets the left aileron position.
Definition FGFCS.h:289
double GetRollTrimCmd(void) const
Gets the aileron trim command.
Definition FGFCS.h:277
~FGFCS() override
Destructor.
Definition FGFCS.cpp:96
void SetDfPos(int form, double pos)
Sets the flaps position.
Definition FGFCS.cpp:258
double GetDaRPos(int form=ofRad) const
Gets the right aileron position.
Definition FGFCS.h:294
void SetDspPos(int form, double pos)
Sets the spoiler position.
Definition FGFCS.cpp:296
void SetTailhookPos(double hookpos)
Set the tailhook position.
Definition FGFCS.h:502
void SetDfCmd(double cmd)
Sets the flaps command.
Definition FGFCS.h:398
double GetBrake(FGLGear::BrakeGroup bg)
Gets the brake for a specified group.
Definition FGFCS.cpp:604
double GetWingFoldPos(void) const
Gets the wing fold position (0 unfolded, 1 folded)
Definition FGFCS.h:350
double GetDspCmd(void) const
Gets the spoiler command.
Definition FGFCS.h:237
void SetPropAdvance(int engine, double cmd)
Sets the actual prop pitch setting for the specified engine.
Definition FGFCS.cpp:429
std::string GetComponentValues(const std::string &delimiter) const
Retrieves all component outputs for inclusion in output stream.
Definition FGFCS.cpp:656
double GetPitchTrimCmd(void) const
Gets the pitch trim command.
Definition FGFCS.h:269
void SetThrottleCmd(int engine, double cmd)
Sets the throttle command for the specified engine.
Definition FGFCS.cpp:315
double GetGearPos(void) const
Gets the gear position (0 up, 1 down), defaults to down.
Definition FGFCS.h:342
double GetDsbCmd(void) const
Gets the speedbrake command.
Definition FGFCS.h:233
void SetDspCmd(double cmd)
Sets the spoilers command.
Definition FGFCS.h:406
void SetLBrake(double cmd)
Sets the left brake group.
Definition FGFCS.h:523
double GetDspPos(int form=ofRad) const
Gets the spoiler position.
Definition FGFCS.h:314
void SetCBrake(double cmd)
Sets the center brake group.
Definition FGFCS.h:531
double GetYawTrimCmd(void) const
Gets the rudder trim command.
Definition FGFCS.h:273
void SetPropAdvanceCmd(int engine, double cmd)
Sets the propeller pitch command for the specified engine.
Definition FGFCS.cpp:415
void SetRollTrimCmd(double cmd)
Sets the aileron trim command.
Definition FGFCS.h:418
void SetYawTrimCmd(double cmd)
Sets the rudder trim command.
Definition FGFCS.h:414
double GetGearCmd(void) const
Get the gear extend/retract command.
Definition FGFCS.h:282
void SetWingFoldPos(double foldpos)
Set the wing fold position.
Definition FGFCS.h:506
double GetDrPos(int form=ofRad) const
Gets the rudder position.
Definition FGFCS.h:304
void SetGearCmd(double gearcmd)
Set the gear extend/retract command, defaults to down.
Definition FGFCS.h:436
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
const SGPath & GetFullAircraftPath(void)
Retrieves the full aircraft path name.
Definition FGFDMExec.h:401
double GetDeltaT(void) const
Returns the simulation delta T.
Definition FGFDMExec.h:552
const SGPath & GetSystemsPath(void)
Retrieves the systems path.
Definition FGFDMExec.h:399
Encapsulates a filter for the flight control system.
Definition FGFilter.h:216
Encapsulates a gain component for the flight control system.
Definition FGGain.h:218
Encapsulates a Gyro component for the flight control system.
Definition FGGyro.h:108
static char normint[6]
normal intensity text
Definition FGJSBBase.h:154
static char fgred[6]
red text
Definition FGJSBBase.h:166
static char fgblue[6]
blue text
Definition FGJSBBase.h:162
static char reset[5]
resets text properties
Definition FGJSBBase.h:156
static char highint[5]
highlights text
Definition FGJSBBase.h:150
Encapsulates a kinematic (mechanical) component for the flight control system.
Definition FGKinemat.h:113
BrakeGroup
Brake grouping enumerators.
Definition FGLGear.h:216
Models a flight control system summing component.
Encapsulates a magnetometer component for the flight control system.
Base class for all scheduled JSBSim models.
Definition FGModel.h:70
virtual bool Run(bool Holding)
Runs the model; called by the Executive.
Definition FGModel.cpp:89
bool Upload(Element *el, bool preLoad)
Uploads this model in memory.
Definition FGModel.cpp:110
Encapsulates a PID control component for the flight control system.
Definition FGPID.h:126
Class wrapper for property handling.
Encapsulates a Sensor component for the flight control system.
Definition FGSensor.h:128
Models a flight control system summing component.
Definition FGSummer.h:103
Encapsulates a switch for the flight control system.
Definition FGSwitch.h:131
Models a Waypoint object.
Definition FGWaypoint.h:98