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
FGStateSpace.h
1/*
2 * FGStateSpace.h
3 * Copyright (C) James Goppert 2010 <james.goppert@gmail.com>
4 *
5 * FGStateSpace.h is free software: you can redistribute it and/or modify it
6 * under the terms of the GNU Lesser General Public License as published by the
7 * Free Software Foundation, either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * FGStateSpace.h is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU Lesser General Public License for more details.
14 *
15 * You should have received a copy of the GNU Lesser General Public License along
16 * with this program. If not, see <http://www.gnu.org/licenses/>.
17 */
18
19#ifndef JSBSim_FGStateSpace_H
20#define JSBSim_FGStateSpace_H
21
22#include "FGFDMExec.h"
23#include "models/FGPropulsion.h"
24#include "models/FGAccelerations.h"
25#include "models/propulsion/FGEngine.h"
26#include "models/propulsion/FGThruster.h"
27#include "models/propulsion/FGTurbine.h"
28#include "models/propulsion/FGTurboProp.h"
29#include "models/FGAuxiliary.h"
30#include "models/FGFCS.h"
31#include <fstream>
32#include <iostream>
33#include <limits>
34
35namespace JSBSim
36{
37
39{
40public:
41
42 // component class
44 {
45 protected:
46 FGStateSpace * m_stateSpace;
47 FGFDMExec * m_fdm;
48 std::string m_name, m_unit;
49 public:
50 Component(const std::string & name, const std::string & unit) :
51 m_stateSpace(), m_fdm(), m_name(name), m_unit(unit) {};
52 virtual ~Component() {};
53 virtual double get() const = 0;
54 virtual void set(double val) = 0;
55 virtual double getDeriv() const
56 {
57 // by default should calculate using finite difference approx
58 std::vector<double> x0 = m_stateSpace->x.get();
59 double f0 = get();
60 double dt0 = m_fdm->GetDeltaT();
61 double time0 = m_fdm->GetSimTime();
62 m_fdm->Setdt(1./120.);
63 m_fdm->DisableOutput();
64 m_fdm->Run();
65 double f1 = get();
66 m_stateSpace->x.set(x0);
67 if (m_fdm->GetDebugLevel() > 1)
68 {
69 std::cout << std::scientific
70 << "name: " << m_name
71 << "\nf1: " << f0
72 << "\nf2: " << f1
73 << "\ndt: " << m_fdm->GetDeltaT()
74 << "\tdf/dt: " << (f1-f0)/m_fdm->GetDeltaT()
75 << std::fixed << std::endl;
76 }
77 double deriv = (f1-f0)/m_fdm->GetDeltaT();
78 m_fdm->Setdt(dt0); // restore original value
79 m_fdm->Setsim_time(time0);
80 m_fdm->EnableOutput();
81 return deriv;
82 }
83 void setStateSpace(FGStateSpace * stateSpace)
84 {
85 m_stateSpace = stateSpace;
86 }
87 void setFdm(FGFDMExec * fdm)
88 {
89 m_fdm = fdm;
90 }
91 const std::string & getName() const
92 {
93 return m_name;
94 }
95 const std::string & getUnit() const
96 {
97 return m_unit;
98 }
99 };
100
101 // component vector class
103 {
104 public:
105 ComponentVector(FGFDMExec * fdm, FGStateSpace * stateSpace) :
106 m_stateSpace(stateSpace), m_fdm(fdm), m_components() {}
107 ComponentVector & operator=(ComponentVector & componentVector)
108 {
109 m_stateSpace = componentVector.m_stateSpace;
110 m_fdm = componentVector.m_fdm;
111 m_components = componentVector.m_components;
112 return *this;
113 }
114 ComponentVector(const ComponentVector & componentVector) :
115 m_stateSpace(componentVector.m_stateSpace),
116 m_fdm(componentVector.m_fdm),
117 m_components(componentVector.m_components)
118 {
119 }
120 void add(Component * comp)
121 {
122 comp->setStateSpace(m_stateSpace);
123 comp->setFdm(m_fdm);
124 m_components.push_back(comp);
125 }
126 size_t getSize() const
127 {
128 return m_components.size();
129 }
130 Component * getComp(int i) const
131 {
132 return m_components[i];
133 };
134 Component * getComp(int i)
135 {
136 return m_components[i];
137 };
138 double get(int i) const
139 {
140 return m_components[i]->get();
141 };
142 void set(int i, double val)
143 {
144 m_components[i]->set(val);
145 m_stateSpace->run();
146 };
147 double get(int i)
148 {
149 return m_components[i]->get();
150 };
151 std::vector<double> get() const
152 {
153 std::vector<double> val;
154 for (unsigned int i=0;i<getSize();i++) val.push_back(m_components[i]->get());
155 return val;
156 }
157 void get(double * array) const
158 {
159 for (unsigned int i=0;i<getSize();i++) array[i] = m_components[i]->get();
160 }
161 double getDeriv(int i)
162 {
163 return m_components[i]->getDeriv();
164 };
165 std::vector<double> getDeriv() const
166 {
167 std::vector<double> val;
168 for (unsigned int i=0;i<getSize();i++) val.push_back(m_components[i]->getDeriv());
169 return val;
170 }
171 void getDeriv(double * array) const
172 {
173 for (unsigned int i=0;i<getSize();i++) array[i] = m_components[i]->getDeriv();
174 }
175 void set(std::vector<double> vals)
176 {
177 for (unsigned int i=0;i<getSize();i++) m_components[i]->set(vals[i]);
178 m_stateSpace->run();
179 }
180 void set(double * array)
181 {
182 for (unsigned int i=0;i<getSize();i++) m_components[i]->set(array[i]);
183 m_stateSpace->run();
184 }
185 std::string getName(int i) const
186 {
187 return m_components[i]->getName();
188 };
189 std::vector<std::string> getName() const
190 {
191 std::vector<std::string> name;
192 for (unsigned int i=0;i<getSize();i++) name.push_back(m_components[i]->getName());
193 return name;
194 }
195 std::string getUnit(int i) const
196 {
197 return m_components[i]->getUnit();
198 };
199 std::vector<std::string> getUnit() const
200 {
201 std::vector<std::string> unit;
202 for (unsigned int i=0;i<getSize();i++) unit.push_back(m_components[i]->getUnit());
203 return unit;
204 }
205 void clear() {
206 m_components.clear();
207 }
208 private:
209 FGStateSpace * m_stateSpace;
210 FGFDMExec * m_fdm;
211 std::vector<Component *> m_components;
212 };
213
214 // component vectors
215 ComponentVector x, u, y;
216
217 // constructor
218 FGStateSpace(FGFDMExec * fdm) : x(fdm,this), u(fdm,this), y(fdm,this), m_fdm(fdm) {};
219
220 void setFdm(FGFDMExec * fdm) { m_fdm = fdm; }
221
222 void run() {
223 // initialize
224 m_fdm->Initialize(m_fdm->GetIC().get());
225
226 for (unsigned int i=0; i<m_fdm->GetPropulsion()->GetNumEngines(); i++) {
227 m_fdm->GetPropulsion()->GetEngine(i)->InitRunning();
228 }
229
230 // wait for stable state
231 double cost = stateSum();
232 for(int i=0;i<1000;i++) {
233 m_fdm->GetPropulsion()->GetSteadyState();
234 m_fdm->SetTrimStatus(true);
235 m_fdm->DisableOutput();
236 m_fdm->SuspendIntegration();
237 m_fdm->Run();
238 m_fdm->SetTrimStatus(false);
239 m_fdm->EnableOutput();
240 m_fdm->ResumeIntegration();
241
242 double costNew = stateSum();
243 double dcost = fabs(costNew - cost);
244 if (dcost < std::numeric_limits<double>::epsilon()) {
245 if(m_fdm->GetDebugLevel() > 1) {
246 std::cout << "cost convergd, i: " << i << std::endl;
247 }
248 break;
249 }
250 if (i > 1000) {
251 if(m_fdm->GetDebugLevel() > 1) {
252 std::cout << "cost failed to converge, dcost: "
253 << std::scientific
254 << dcost << std::endl;
255 }
256 break;
257 }
258 cost = costNew;
259 }
260 }
261
262 double stateSum() {
263 double sum = 0;
264 for (unsigned int i=0;i<x.getSize();i++) sum += x.get(i);
265 return sum;
266 }
267
268 void clear() {
269 x.clear();
270 u.clear();
271 y.clear();
272 }
273
274 // deconstructor
275 virtual ~FGStateSpace() {};
276
277 // linearization function
278 void linearize(std::vector<double> x0, std::vector<double> u0, std::vector<double> y0,
279 std::vector< std::vector<double> > & A,
280 std::vector< std::vector<double> > & B,
281 std::vector< std::vector<double> > & C,
282 std::vector< std::vector<double> > & D);
283
284
285private:
286
287 // compute numerical jacobian of a matrix
288 void numericalJacobian(std::vector< std::vector<double> > & J, ComponentVector & y,
289 ComponentVector & x, const std::vector<double> & y0,
290 const std::vector<double> & x0, double h=1e-5, bool computeYDerivative = false);
291
292 // flight dynamcis model
293 FGFDMExec * m_fdm;
294
295public:
296
297 // components
298
299 class Vt : public Component
300 {
301 public:
302 Vt() : Component("Vt","ft/s") {};
303 double get() const
304 {
305 return m_fdm->GetAuxiliary()->GetVt();
306 }
307 void set(double val)
308 {
309 m_fdm->GetIC()->SetVtrueFpsIC(val);
310 }
311 double getDeriv() const
312 {
313
314 return (m_fdm->GetPropagate()->GetUVW(1)*m_fdm->GetAccelerations()->GetUVWdot(1) +
315 m_fdm->GetPropagate()->GetUVW(2)*m_fdm->GetAccelerations()->GetUVWdot(2) +
316 m_fdm->GetPropagate()->GetUVW(3)*m_fdm->GetAccelerations()->GetUVWdot(3))/
317 m_fdm->GetAuxiliary()->GetVt(); // from lewis, vtrue dot
318 }
319
320 };
321
322 class VGround : public Component
323 {
324 public:
325 VGround() : Component("VGround","ft/s") {};
326 double get() const
327 {
328 return m_fdm->GetAuxiliary()->GetVground();
329 }
330 void set(double val)
331 {
332 m_fdm->GetIC()->SetVgroundFpsIC(val);
333 }
334 };
335
336 class AccelX : public Component
337 {
338 public:
339 AccelX() : Component("AccelX","ft/s^2") {};
340 double get() const
341 {
342 return m_fdm->GetAuxiliary()->GetPilotAccel(1);
343 }
344 void set(double val)
345 {
346 // XXX: not possible to implement currently
347 }
348 };
349
350 class AccelY : public Component
351 {
352 public:
353 AccelY() : Component("AccelY","ft/s^2") {};
354 double get() const
355 {
356 return m_fdm->GetAuxiliary()->GetPilotAccel(2);
357 }
358 void set(double val)
359 {
360 // XXX: not possible to implement currently
361 }
362 };
363
364 class AccelZ : public Component
365 {
366 public:
367 AccelZ() : Component("AccelZ","ft/s^2") {};
368 double get() const
369 {
370 return m_fdm->GetAuxiliary()->GetPilotAccel(3);
371 }
372 void set(double val)
373 {
374 // XXX: not possible to implement currently
375 }
376 };
377
378 class Alpha : public Component
379 {
380 public:
381 Alpha() : Component("Alpha","rad") {};
382 double get() const
383 {
384 return m_fdm->GetAuxiliary()->Getalpha();
385 }
386 void set(double val)
387 {
388 double beta = m_fdm->GetIC()->GetBetaDegIC();
389 double psi = m_fdm->GetIC()->GetPsiRadIC();
390 double theta = m_fdm->GetIC()->GetThetaRadIC();
391 m_fdm->GetIC()->SetAlphaRadIC(val);
392 m_fdm->GetIC()->SetBetaRadIC(beta);
393 m_fdm->GetIC()->SetPsiRadIC(psi);
394 m_fdm->GetIC()->SetThetaRadIC(theta);
395 }
396 double getDeriv() const
397 {
398 return m_fdm->GetAuxiliary()->Getadot();
399 }
400 };
401
402 class Theta : public Component
403 {
404 public:
405 Theta() : Component("Theta","rad") {};
406 double get() const
407 {
408 return m_fdm->GetPropagate()->GetEuler(2);
409 }
410 void set(double val)
411 {
412 m_fdm->GetIC()->SetFlightPathAngleRadIC(val-m_fdm->GetIC()->GetAlphaRadIC());
413 //m_fdm->GetIC()->SetThetaRadIC(val);
414 }
415 double getDeriv() const
416 {
417 return m_fdm->GetAuxiliary()->GetEulerRates(2);
418 }
419 };
420
421 class Q : public Component
422 {
423 public:
424 Q() : Component("Q","rad/s") {};
425 double get() const
426 {
427 return m_fdm->GetPropagate()->GetPQR(2);
428 }
429 void set(double val)
430 {
431 m_fdm->GetIC()->SetQRadpsIC(val);
432 }
433 double getDeriv() const
434 {
435 return m_fdm->GetAccelerations()->GetPQRdot(2);
436 }
437 };
438
439 class Alt : public Component
440 {
441 public:
442 Alt() : Component("Alt","ft") {};
443 double get() const
444 {
445 return m_fdm->GetPropagate()->GetAltitudeASL();
446 }
447 void set(double val)
448 {
449 m_fdm->GetIC()->SetAltitudeASLFtIC(val);
450 }
451 double getDeriv() const
452 {
453 return m_fdm->GetPropagate()->Gethdot();
454 }
455 };
456
457 class Beta : public Component
458 {
459 public:
460 Beta() : Component("Beta","rad") {};
461 double get() const
462 {
463 return m_fdm->GetAuxiliary()->Getbeta();
464 }
465 void set(double val)
466 {
467 double psi = m_fdm->GetIC()->GetPsiRadIC();
468 m_fdm->GetIC()->SetBetaRadIC(val);
469 m_fdm->GetIC()->SetPsiRadIC(psi);
470 }
471 double getDeriv() const
472 {
473 return m_fdm->GetAuxiliary()->Getbdot();
474 }
475 };
476
477 class Phi : public Component
478 {
479 public:
480 Phi() : Component("Phi","rad") {};
481 double get() const
482 {
483 return m_fdm->GetPropagate()->GetEuler(1);
484 }
485 void set(double val)
486 {
487 m_fdm->GetIC()->SetPhiRadIC(val);
488 }
489 double getDeriv() const
490 {
491 return m_fdm->GetAuxiliary()->GetEulerRates(1);
492 }
493 };
494
495 class P : public Component
496 {
497 public:
498 P() : Component("P","rad/s") {};
499 double get() const
500 {
501 return m_fdm->GetPropagate()->GetPQR(1);
502 }
503 void set(double val)
504 {
505 m_fdm->GetIC()->SetPRadpsIC(val);
506 }
507 double getDeriv() const
508 {
509 return m_fdm->GetAccelerations()->GetPQRdot(1);
510 }
511 };
512
513 class R : public Component
514 {
515 public:
516 R() : Component("R","rad/s") {};
517 double get() const
518 {
519 return m_fdm->GetPropagate()->GetPQR(3);
520 }
521 void set(double val)
522 {
523 m_fdm->GetIC()->SetRRadpsIC(val);
524 }
525 double getDeriv() const
526 {
527 return m_fdm->GetAccelerations()->GetPQRdot(3);
528 }
529 };
530
531 class Psi : public Component
532 {
533 public:
534 Psi() : Component("Psi","rad") {};
535 double get() const
536 {
537 return m_fdm->GetPropagate()->GetEuler(3);
538 }
539 void set(double val)
540 {
541 m_fdm->GetIC()->SetPsiRadIC(val);
542 }
543 double getDeriv() const
544 {
545 return m_fdm->GetAuxiliary()->GetEulerRates(3);
546 }
547 };
548
549 class ThrottleCmd : public Component
550 {
551 public:
552 ThrottleCmd() : Component("ThtlCmd","norm") {};
553 double get() const
554 {
555 return m_fdm->GetFCS()->GetThrottleCmd(0);
556 }
557 void set(double val)
558 {
559 for (unsigned int i=0;i<m_fdm->GetPropulsion()->GetNumEngines();i++)
560 m_fdm->GetFCS()->SetThrottleCmd(i,val);
561 m_fdm->GetFCS()->Run(true);
562 }
563 };
564
565 class ThrottlePos : public Component
566 {
567 public:
568 ThrottlePos() : Component("ThtlPos","norm") {};
569 double get() const
570 {
571 return m_fdm->GetFCS()->GetThrottlePos(0);
572 }
573 void set(double val)
574 {
575 for (unsigned int i=0;i<m_fdm->GetPropulsion()->GetNumEngines();i++)
576 m_fdm->GetFCS()->SetThrottlePos(i,val);
577 }
578 };
579
580 class DaCmd : public Component
581 {
582 public:
583 DaCmd() : Component("DaCmd","norm") {};
584 double get() const
585 {
586 return m_fdm->GetFCS()->GetDaCmd();
587 }
588 void set(double val)
589 {
590 m_fdm->GetFCS()->SetDaCmd(val);
591 m_fdm->GetFCS()->Run(true);
592 }
593 };
594
595 class DaPos : public Component
596 {
597 public:
598 DaPos() : Component("DaPos","norm") {};
599 double get() const
600 {
601 return m_fdm->GetFCS()->GetDaLPos();
602 }
603 void set(double val)
604 {
605 m_fdm->GetFCS()->SetDaLPos(ofRad,val);
606 m_fdm->GetFCS()->SetDaRPos(ofRad,val); // TODO: check if this is neg.
607 }
608 };
609
610 class DeCmd : public Component
611 {
612 public:
613 DeCmd() : Component("DeCmd","norm") {};
614 double get() const
615 {
616 return m_fdm->GetFCS()->GetDeCmd();
617 }
618 void set(double val)
619 {
620 m_fdm->GetFCS()->SetDeCmd(val);
621 m_fdm->GetFCS()->Run(true);
622 }
623 };
624
625 class DePos : public Component
626 {
627 public:
628 DePos() : Component("DePos","norm") {};
629 double get() const
630 {
631 return m_fdm->GetFCS()->GetDePos();
632 }
633 void set(double val)
634 {
635 m_fdm->GetFCS()->SetDePos(ofRad,val);
636 }
637 };
638
639 class DrCmd : public Component
640 {
641 public:
642 DrCmd() : Component("DrCmd","norm") {};
643 double get() const
644 {
645 return m_fdm->GetFCS()->GetDrCmd();
646 }
647 void set(double val)
648 {
649 m_fdm->GetFCS()->SetDrCmd(val);
650 m_fdm->GetFCS()->Run(true);
651 }
652 };
653
654 class DrPos : public Component
655 {
656 public:
657 DrPos() : Component("DrPos","norm") {};
658 double get() const
659 {
660 return m_fdm->GetFCS()->GetDrPos();
661 }
662 void set(double val)
663 {
664 m_fdm->GetFCS()->SetDrPos(ofRad,val);
665 }
666 };
667
668 class Rpm0 : public Component
669 {
670 public:
671 Rpm0() : Component("Rpm0","rev/min") {};
672 double get() const
673 {
674 return m_fdm->GetPropulsion()->GetEngine(0)->GetThruster()->GetRPM();
675 }
676 void set(double val)
677 {
678 m_fdm->GetPropulsion()->GetEngine(0)->GetThruster()->SetRPM(val);
679 }
680 };
681
682 class Rpm1 : public Component
683 {
684 public:
685 Rpm1() : Component("Rpm1","rev/min") {};
686 double get() const
687 {
688 return m_fdm->GetPropulsion()->GetEngine(1)->GetThruster()->GetRPM();
689 }
690 void set(double val)
691 {
692 m_fdm->GetPropulsion()->GetEngine(1)->GetThruster()->SetRPM(val);
693 }
694 };
695
696 class Rpm2 : public Component
697 {
698 public:
699 Rpm2() : Component("Rpm2","rev/min") {};
700 double get() const
701 {
702 return m_fdm->GetPropulsion()->GetEngine(2)->GetThruster()->GetRPM();
703 }
704 void set(double val)
705 {
706 m_fdm->GetPropulsion()->GetEngine(2)->GetThruster()->SetRPM(val);
707 }
708 };
709
710 class Rpm3 : public Component
711 {
712 public:
713 Rpm3() : Component("Rpm3","rev/min") {};
714 double get() const
715 {
716 return m_fdm->GetPropulsion()->GetEngine(3)->GetThruster()->GetRPM();
717 }
718 void set(double val)
719 {
720 m_fdm->GetPropulsion()->GetEngine(3)->GetThruster()->SetRPM(val);
721 }
722 };
723
724 class PropPitch : public Component
725 {
726 public:
727 PropPitch() : Component("Prop Pitch","deg") {};
728 double get() const
729 {
730 return m_fdm->GetPropulsion()->GetEngine(0)->GetThruster()->GetPitch();
731 }
732 void set(double val)
733 {
734 for (unsigned int i=0;i<m_fdm->GetPropulsion()->GetNumEngines();i++)
735 m_fdm->GetPropulsion()->GetEngine(i)->GetThruster()->SetPitch(val);
736 }
737 };
738
739 class Longitude : public Component
740 {
741 public:
742 Longitude() : Component("Longitude","rad") {};
743 double get() const
744 {
745 return m_fdm->GetPropagate()->GetLongitude();
746 }
747 void set(double val)
748 {
749 m_fdm->GetIC()->SetLongitudeRadIC(val);
750 }
751 double getDeriv() const
752 {
753 return m_fdm->GetPropagate()->GetVel(2)/(cos(m_fdm->GetPropagate()->GetLatitude())*m_fdm->GetPropagate()->GetRadius());
754 }
755 };
756
757 class Latitude : public Component
758 {
759 public:
760 Latitude() : Component("Latitude","rad") {};
761 double get() const
762 {
763 return m_fdm->GetPropagate()->GetLatitude();
764 }
765 void set(double val)
766 {
767 m_fdm->GetIC()->SetLatitudeRadIC(val);
768 }
769 double getDeriv() const
770 {
771 return m_fdm->GetPropagate()->GetVel(1)/(m_fdm->GetPropagate()->GetRadius());
772 }
773 };
774
775 class Pi : public Component
776 {
777 public:
778 Pi() : Component("P inertial","rad/s") {};
779 double get() const
780 {
781 return m_fdm->GetPropagate()->GetPQRi(1);
782 }
783 void set(double val)
784 {
785 //Set PQR from PQRi
786 //VState.vPQR = VState.vPQRi - Ti2b * vOmegaEarth;
787 m_fdm->GetIC()->SetQRadpsIC(val + \
788 (m_fdm->GetPropagate()->GetPQR(1) - m_fdm->GetPropagate()->GetPQRi(1)));
789 }
790 double getDeriv() const
791 {
792 return m_fdm->GetAccelerations()->GetPQRdot(1);
793 }
794 };
795
796 class Qi : public Component
797 {
798 public:
799 Qi() : Component("Q inertial","rad/s") {};
800 double get() const
801 {
802 return m_fdm->GetPropagate()->GetPQRi(2);
803 }
804 void set(double val)
805 {
806 //Set PQR from PQRi
807 //VState.vPQR = VState.vPQRi - Ti2b * vOmegaEarth;
808 m_fdm->GetIC()->SetQRadpsIC(val + \
809 (m_fdm->GetPropagate()->GetPQR(2) - m_fdm->GetPropagate()->GetPQRi(2)));
810 }
811 double getDeriv() const
812 {
813 return m_fdm->GetAccelerations()->GetPQRdot(2);
814 }
815 };
816
817 class Ri : public Component
818 {
819 public:
820 Ri() : Component("R inertial","rad/s") {};
821 double get() const
822 {
823 return m_fdm->GetPropagate()->GetPQRi(3);
824 }
825 void set(double val)
826 {
827 //Set PQR from PQRi
828 //VState.vPQR = VState.vPQRi - Ti2b * vOmegaEarth;
829 m_fdm->GetIC()->SetQRadpsIC(val + \
830 (m_fdm->GetPropagate()->GetPQR(3) - m_fdm->GetPropagate()->GetPQRi(3)));
831 }
832 double getDeriv() const
833 {
834 return m_fdm->GetAccelerations()->GetPQRdot(3);
835 }
836 };
837
838 class Vn : public Component
839 {
840 public:
841 Vn() : Component("Vel north","feet/s") {};
842 double get() const
843 {
844 return m_fdm->GetPropagate()->GetVel(1);
845 }
846 void set(double val)
847 {
848 m_fdm->GetIC()->SetVNorthFpsIC(val);
849 }
850 double getDeriv() const
851 {
852 //get NED accel from body accel
853 return (m_fdm->GetPropagate()->GetTb2l()*m_fdm->GetAccelerations()->GetUVWdot())(1);
854 }
855 };
856
857 class Ve : public Component
858 {
859 public:
860 Ve() : Component("Vel east","feet/s") {};
861 double get() const
862 {
863 return m_fdm->GetPropagate()->GetVel(2);
864 }
865 void set(double val)
866 {
867 m_fdm->GetIC()->SetVEastFpsIC(val);
868 }
869 double getDeriv() const
870 {
871 //get NED accel from body accel
872 return (m_fdm->GetPropagate()->GetTb2l()*m_fdm->GetAccelerations()->GetUVWdot())(2);
873 }
874 };
875
876 class Vd : public Component
877 {
878 public:
879 Vd() : Component("Vel down","feet/s") {};
880 double get() const
881 {
882 return m_fdm->GetPropagate()->GetVel(3);
883 }
884 void set(double val)
885 {
886 m_fdm->GetIC()->SetVDownFpsIC(val);
887 }
888 double getDeriv() const
889 {
890 //get NED accel from body accel
891 return (m_fdm->GetPropagate()->GetTb2l()*m_fdm->GetAccelerations()->GetUVWdot())(3);
892 }
893 };
894
895 class COG : public Component
896 {
897 public:
898 COG() : Component("Course Over Ground","rad") {};
899 double get() const
900 {
901 //cog = atan2(Ve,Vn)
902 return atan2(m_fdm->GetPropagate()->GetVel(2),m_fdm->GetPropagate()->GetVel(1));
903 }
904 void set(double val)
905 {
906 //set Vn and Ve according to vGround and COG
907 m_fdm->GetIC()->SetVNorthFpsIC(m_fdm->GetAuxiliary()->GetVground()*cos(val));
908 m_fdm->GetIC()->SetVEastFpsIC(m_fdm->GetAuxiliary()->GetVground()*sin(val));
909 }
910 double getDeriv() const
911 {
912 double Vn = m_fdm->GetPropagate()->GetVel(1);
913 double Vndot = (m_fdm->GetPropagate()->GetTb2l()*m_fdm->GetAccelerations()->GetUVWdot())(1);
914 double Ve = m_fdm->GetPropagate()->GetVel(2);
915 double Vedot = (m_fdm->GetPropagate()->GetTb2l()*m_fdm->GetAccelerations()->GetUVWdot())(2);
916
917 //dCOG/dt = dCOG/dVe*dVe/dt + dCOG/dVn*dVn/dt
918 return Vn/(Vn*Vn+Ve*Ve)*Vedot - Ve/(Vn*Vn+Ve*Ve)*Vndot;
919 }
920 };
921
922};
923
924// stream output
925std::ostream &operator<<(std::ostream &out, const FGStateSpace::Component &c );
926std::ostream &operator<<(std::ostream &out, const FGStateSpace::ComponentVector &v );
927std::ostream &operator<<(std::ostream &out, const FGStateSpace &ss );
928std::ostream &operator<<( std::ostream &out, const std::vector< std::vector<double> > &vec2d );
929std::ostream &operator<<( std::ostream &out, const std::vector<double> &vec );
930
931} // JSBSim
932
933#endif
934
935// vim:ts=4:sw=4
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
void EnableOutput(void)
Enables data logging to all outputs.
Definition FGFDMExec.h:486
std::shared_ptr< FGInitialCondition > GetIC(void) const
Returns a pointer to the FGInitialCondition object.
Definition FGFDMExec.h:389
int GetDebugLevel(void) const
Retrieves the current debug level setting.
Definition FGFDMExec.h:601
double GetDeltaT(void) const
Returns the simulation delta T.
Definition FGFDMExec.h:552
double Setsim_time(double cur_time)
Sets the current sim time.
bool Run(void)
This function executes each scheduled model in succession.
void DisableOutput(void)
Disables data logging to all outputs.
Definition FGFDMExec.h:484
std::shared_ptr< FGPropulsion > GetPropulsion(void) const
Returns the FGPropulsion pointer.
double GetSimTime(void) const
Returns the cumulative simulation time in seconds.
Definition FGFDMExec.h:549
void Setdt(double delta_t)
Sets the integration time step for the simulation executive.
Definition FGFDMExec.h:571
void SuspendIntegration(void)
Suspends the simulation and sets the delta T to zero.
Definition FGFDMExec.h:555
void Initialize(const FGInitialCondition *FGIC)
Initializes the simulation with initial conditions.
void ResumeIntegration(void)
Resumes the simulation by resetting delta T to the correct value.
Definition FGFDMExec.h:558