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
JSBSim.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: JSBSim.cpp
4 Author: Jon S. Berndt
5 Date started: 08/17/99
6 Purpose: Standalone version of JSBSim.
7 Called by: The USER.
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--------------------------------------------------------------------------------
30
31This class implements the JSBSim standalone application. It is set up for
32compilation under gnu C++, MSVC++, or other compiler.
33
34HISTORY
35--------------------------------------------------------------------------------
3608/17/99 JSB Created
37
38%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
39INCLUDES
40%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
41
42#include "initialization/FGTrim.h"
43#include "initialization/FGInitialCondition.h"
44#include "FGFDMExec.h"
45#include "input_output/FGXMLFileRead.h"
46#include "input_output/string_utilities.h"
47
48#if !defined(__GNUC__) && !defined(sgi) && !defined(_MSC_VER)
49# include <time>
50#else
51# include <time.h>
52#endif
53
54#if defined(_MSC_VER)
55# include <float.h>
56#elif defined(__GNUC__) && !defined(sgi)
57# define __GNU_VISIBLE 1
58# include <fenv.h>
59#endif
60
61#if defined(__BORLANDC__) || defined(_MSC_VER) || defined(__MINGW32__)
62# define WIN32_LEAN_AND_MEAN
63# include <windows.h>
64# include <mmsystem.h>
65# include <regstr.h>
66# include <sys/types.h>
67# include <sys/timeb.h>
68#else
69# include <sys/time.h>
70#endif
71
72// The flag ENABLE_VIRTUAL_TERMINAL_PROCESSING is not defined for MinGW < 7.0.0
73#if defined(__MINGW64_VERSION_MAJOR) && __MINGW64_VERSION_MAJOR < 7
74#define ENABLE_VIRTUAL_TERMINAL_PROCESSING 0x4
75#endif
76
77#include <iostream>
78#include <cstdlib>
79
80using namespace std;
82using JSBSim::Element;
83
84/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
85GLOBAL DATA
86%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
87
88SGPath RootDir;
89SGPath ScriptName;
90string AircraftName;
91SGPath ResetName;
92SGPath PlanetName;
93vector <string> LogOutputName;
94vector <SGPath> LogDirectiveName;
95vector <string> CommandLineProperties;
96vector <double> CommandLinePropertyValues;
97JSBSim::FGFDMExec* FDMExec;
98JSBSim::FGTrim* trimmer;
99
100bool realtime;
101bool play_nice;
102bool suspend;
103bool catalog;
104bool nohighlight;
105
106double end_time = 1e99;
107double simulation_rate = 1./120.;
108bool override_sim_rate = false;
109double sleep_period=0.01;
110
111/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
112FORWARD DECLARATIONS
113%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
114
115bool options(int, char**);
116int real_main(int argc, char* argv[]);
117void PrintHelp(void);
118
119#if defined(__BORLANDC__) || defined(_MSC_VER) || defined(__MINGW32__)
120 double getcurrentseconds(void)
121 {
122 struct timeb tm_ptr;
123 ftime(&tm_ptr);
124 return tm_ptr.time + tm_ptr.millitm*0.001;
125 }
126#else
127 double getcurrentseconds(void)
128 {
129 struct timeval tval;
130 struct timezone tz;
131
132 gettimeofday(&tval, &tz);
133 return (tval.tv_sec + tval.tv_usec*1e-6);
134 }
135#endif
136
137#if defined(__BORLANDC__) || defined(_MSC_VER) || defined(__MINGW32__)
138 void sim_nsleep(long nanosec)
139 {
140 Sleep((DWORD)(nanosec*1e-6)); // convert nanoseconds (passed in) to milliseconds for Win32.
141 }
142#else
143 void sim_nsleep(long nanosec)
144 {
145 struct timespec ts, ts1;
146
147 ts.tv_sec = 0;
148 ts.tv_nsec = nanosec;
149 nanosleep(&ts, &ts1);
150 }
151#endif
152
155class XMLFile : public FGXMLFileRead {
156public:
157 bool IsScriptFile(const SGPath& filename) {
158 bool result=false;
159 Element *document = LoadXMLDocument(filename, false);
160 if (document && document->GetName() == "runscript") result = true;
161 ResetParser();
162 return result;
163 }
164 bool IsLogDirectiveFile(const SGPath& filename) {
165 bool result=false;
166 Element *document = LoadXMLDocument(filename, false);
167 if (document && document->GetName() == "output") result = true;
168 ResetParser();
169 return result;
170 }
171 bool IsAircraftFile(const SGPath& filename) {
172 bool result=false;
173 Element* document = LoadXMLDocument(filename, false);
174 if (document && document->GetName() == "fdm_config") result = true;
175 ResetParser();
176 return result;
177 }
178 bool IsInitFile(const SGPath& filename) {
179 bool result=false;
180 Element *document = LoadXMLDocument(filename, false);
181 if (document && document->GetName() == "initialize") result = true;
182 ResetParser();
183 return result;
184 }
185};
186
190class Timer : public SGPropertyChangeListener {
191public:
192 Timer() : SGPropertyChangeListener(), isPaused(false) { start(); }
193 void start(void) { initial_seconds = getcurrentseconds(); }
194
196 void valueChanged(SGPropertyNode* prop) override {
197 start();
198 if (isPaused) pause_start_seconds = initial_seconds;
199 }
201 double getElapsedTime(void) { return getcurrentseconds() - initial_seconds; }
202
205 void pause(bool paused) {
206 if (paused) {
207 if (!isPaused) {
208 isPaused = true;
209 pause_start_seconds = getcurrentseconds();
210 }
211 } else {
212 if (isPaused) {
213 isPaused = false;
214 double pause_duration = getcurrentseconds() - pause_start_seconds;
215 initial_seconds += pause_duration; // Shift the initial time to account for the pause duration.
216 }
217 }
218 }
219private:
220 double initial_seconds = 0.0;
221 bool isPaused = false;
222 double pause_start_seconds = 0.0;
223};
224
225/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
226CLASS DOCUMENTATION
227%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
228
323/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
324IMPLEMENTATION
325%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
326
327int main(int argc, char* argv[])
328{
329#if defined(_MSC_VER) || defined(__MINGW32__)
330 _clearfp();
331 _controlfp(_controlfp(0, 0) & ~(_EM_INVALID | _EM_ZERODIVIDE | _EM_OVERFLOW),
332 _MCW_EM);
333#elif defined(__GNUC__) && !defined(sgi) && !defined(__APPLE__)
334 feenableexcept(FE_DIVBYZERO | FE_INVALID | FE_OVERFLOW);
335#endif
336
337 try {
338 real_main(argc, argv);
339 } catch (string& msg) {
340 std::cerr << "FATAL ERROR: JSBSim terminated with an exception."
341 << std::endl << "The message was: " << msg << std::endl;
342 return 1;
343 } catch (const char* msg) {
344 std::cerr << "FATAL ERROR: JSBSim terminated with an exception."
345 << std::endl << "The message was: " << msg << std::endl;
346 return 1;
347 } catch (const JSBSim::BaseException& e) {
348 std::cerr << "FATAL ERROR: JSBSim terminated with an exception."
349 << std::endl << "The message was: " << e.what() << std::endl;
350 return 1;
351 } catch (...) {
352 std::cerr << "FATAL ERROR: JSBSim terminated with an unknown exception."
353 << std::endl;
354 return 1;
355 }
356 return 0;
357}
358
359int real_main(int argc, char* argv[])
360{
361 // *** INITIALIZATIONS *** //
362
363 ScriptName = "";
364 AircraftName = "";
365 ResetName = "";
366 PlanetName = "";
367 LogOutputName.clear();
368 LogDirectiveName.clear();
369 bool result = false, success;
370 double frame_duration;
371
372 double new_five_second_value = 0.0;
373 double actual_elapsed_time = 0;
374 double cycle_duration = 0.0;
375 double override_sim_rate_value = 0.0;
376 long sleep_nseconds = 0;
377
378 realtime = false;
379 play_nice = false;
380 suspend = false;
381 catalog = false;
382 nohighlight = false;
383
384 // *** PARSE OPTIONS PASSED INTO THIS SPECIFIC APPLICATION: JSBSim *** //
385 success = options(argc, argv);
386 if (!success) {
387 PrintHelp();
388 exit(-1);
389 }
390
391 // *** SET UP JSBSIM *** //
392 FDMExec = new JSBSim::FGFDMExec();
393 FDMExec->SetRootDir(RootDir);
394 FDMExec->SetAircraftPath(SGPath("aircraft"));
395 FDMExec->SetEnginePath(SGPath("engine"));
396 FDMExec->SetSystemsPath(SGPath("systems"));
397 FDMExec->SetOutputPath(SGPath("."));
398 FDMExec->GetPropertyManager()->Tie("simulation/frame_start_time", &actual_elapsed_time);
399 FDMExec->GetPropertyManager()->Tie("simulation/cycle_duration", &cycle_duration);
400
401 Timer timer;
402 SGPropertyNode_ptr reset_node = FDMExec->GetPropertyManager()->GetNode("simulation/reset");
403 reset_node->addChangeListener(&timer);
404
405 // Check whether to disable console highlighting output on Windows.
406 // Support was added to Windows for Virtual Terminal codes by a particular
407 // Windows 10 release.
408#ifdef _WIN32
409 HANDLE hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
410 DWORD dwMode = 0;
411 GetConsoleMode(hStdOut, &dwMode);
412 if ((dwMode & ENABLE_VIRTUAL_TERMINAL_PROCESSING) == 0)
413 nohighlight = true;
414#endif
415
416 if (nohighlight) FDMExec->disableHighLighting();
417
418 if (simulation_rate < 1.0 )
419 FDMExec->Setdt(simulation_rate);
420 else
421 FDMExec->Setdt(1.0/simulation_rate);
422
423 if (override_sim_rate) override_sim_rate_value = FDMExec->GetDeltaT();
424
425 // SET PROPERTY VALUES THAT ARE GIVEN ON THE COMMAND LINE and which are for the simulation only.
426
427 for (unsigned int i=0; i<CommandLineProperties.size(); i++) {
428
429 if (CommandLineProperties[i].find("simulation") != std::string::npos) {
430 if (FDMExec->GetPropertyManager()->GetNode(CommandLineProperties[i])) {
431 FDMExec->SetPropertyValue(CommandLineProperties[i], CommandLinePropertyValues[i]);
432 }
433 }
434 }
435
436 if (!PlanetName.isNull()) {
437 result = FDMExec->LoadPlanet(PlanetName, false);
438
439 if (!result) {
440 cerr << "Planet file " << PlanetName << " was not successfully loaded" << endl;
441 delete FDMExec;
442 exit(-1);
443 }
444 }
445
446 // *** OPTION A: LOAD A SCRIPT, WHICH LOADS EVERYTHING ELSE *** //
447 if (!ScriptName.isNull()) {
448
449 result = FDMExec->LoadScript(ScriptName, override_sim_rate_value, ResetName);
450
451 if (!result) {
452 cerr << "Script file " << ScriptName << " was not successfully loaded" << endl;
453 delete FDMExec;
454 exit(-1);
455 }
456
457 // *** OPTION B: LOAD AN AIRCRAFT AND A SET OF INITIAL CONDITIONS *** //
458 } else if (!AircraftName.empty() || !ResetName.isNull()) {
459
460 if (catalog) FDMExec->SetDebugLevel(0);
461
462 if ( ! FDMExec->LoadModel(SGPath("aircraft"),
463 SGPath("engine"),
464 SGPath("systems"),
465 AircraftName)) {
466 cerr << " JSBSim could not be started" << endl << endl;
467 delete FDMExec;
468 exit(-1);
469 }
470
471 if (catalog) {
472 FDMExec->PrintPropertyCatalog();
473 delete FDMExec;
474 return 0;
475 }
476
477 auto IC = FDMExec->GetIC();
478 if ( ! IC->Load(ResetName)) {
479 delete FDMExec;
480 cerr << "Initialization unsuccessful" << endl;
481 exit(-1);
482 }
483
484 } else {
485 cout << " No Aircraft, Script, or Reset information given" << endl << endl;
486 delete FDMExec;
487 exit(-1);
488 }
489
490 // Load output directives file[s], if given
491 for (unsigned int i=0; i<LogDirectiveName.size(); i++) {
492 if (!LogDirectiveName[i].isNull()) {
493 if (!FDMExec->SetOutputDirectives(LogDirectiveName[i])) {
494 cout << "Output directives not properly set in file " << LogDirectiveName[i] << endl;
495 delete FDMExec;
496 exit(-1);
497 }
498 }
499 }
500
501 // OVERRIDE OUTPUT FILE NAME. THIS IS USEFUL FOR CASES WHERE MULTIPLE
502 // RUNS ARE BEING MADE (SUCH AS IN A MONTE CARLO STUDY) AND THE OUTPUT FILE
503 // NAME MUST BE SET EACH TIME TO AVOID THE PREVIOUS RUN DATA FROM BEING OVER-
504 // WRITTEN.
505 for (unsigned int i=0; i<LogOutputName.size(); i++) {
506 string old_filename = FDMExec->GetOutputFileName(i);
507 if (!FDMExec->SetOutputFileName(i, LogOutputName[i])) {
508 cout << "Output filename could not be set" << endl;
509 } else {
510 cout << "Output filename change from " << old_filename << " from aircraft"
511 " configuration file to " << LogOutputName[i] << " specified on"
512 " command line" << endl;
513 }
514 }
515
516 // SET PROPERTY VALUES THAT ARE GIVEN ON THE COMMAND LINE
517
518 for (unsigned int i=0; i<CommandLineProperties.size(); i++) {
519
520 if (!FDMExec->GetPropertyManager()->GetNode(CommandLineProperties[i])) {
521 cerr << endl << " No property by the name " << CommandLineProperties[i] << endl;
522 delete FDMExec;
523 exit(-1);
524 } else {
525 FDMExec->SetPropertyValue(CommandLineProperties[i], CommandLinePropertyValues[i]);
526 }
527 }
528
529 FDMExec->RunIC();
530
531 // PRINT SIMULATION CONFIGURATION
532 FDMExec->PrintSimulationConfiguration();
533
534 // Dump the simulation state (position, orientation, etc.)
535 FDMExec->GetPropagate()->DumpState();
536
537 // Perform trim if requested via the initialization file
538 JSBSim::TrimMode icTrimRequested = (JSBSim::TrimMode)FDMExec->GetIC()->TrimRequested();
539 if (icTrimRequested != JSBSim::TrimMode::tNone) {
540 trimmer = new JSBSim::FGTrim( FDMExec, icTrimRequested );
541 try {
542 trimmer->DoTrim();
543
544 if (FDMExec->GetDebugLevel() > 0)
545 trimmer->Report();
546
547 delete trimmer;
548 } catch (string& msg) {
549 cerr << endl << msg << endl << endl;
550 exit(1);
551 }
552 }
553
555 << "---- JSBSim Execution beginning ... --------------------------------------------"
556 << JSBSim::FGFDMExec::reset << endl << endl;
557
558 result = FDMExec->Run(); // MAKE AN INITIAL RUN
559
560 if (suspend) FDMExec->Hold();
561
562 // Print actual time at start
563 char s[100];
564 time_t tod;
565 time(&tod);
566 struct tm local;
567#if defined(_MSC_VER) || defined(__MINGW32__)
568 localtime_s(&local, &tod);
569#else
570 localtime_r(&tod, &local);
571#endif
572 strftime(s, 99, "%A %B %d %Y %X", &local);
573 cout << "Start: " << s << " (HH:MM:SS)" << endl;
574
575 frame_duration = FDMExec->GetDeltaT();
576 if (realtime) sleep_nseconds = (long)(frame_duration*1e9);
577 else sleep_nseconds = (sleep_period )*1e9; // 0.01 seconds
578
579 tzset();
580 timer.start();
581
582 // *** CYCLIC EXECUTION LOOP, AND MESSAGE READING *** //
583 while (result && FDMExec->GetSimTime() <= end_time) {
584 // Check if increment then hold is on and take appropriate actions if it is
585 // Iterate is not supported in realtime - only in batch and playnice modes
586 FDMExec->CheckIncrementalHold();
587
588 // if running realtime, throttle the execution, else just run flat-out fast
589 // unless "playing nice", in which case sleep for a while (0.01 seconds) each frame.
590 // If suspended, then don't increment cumulative realtime "stopwatch".
591
592 if ( ! FDMExec->Holding()) {
593 if ( ! realtime ) { // ------------ RUNNING IN BATCH MODE
594
595 result = FDMExec->Run();
596
597 if (play_nice) sim_nsleep(sleep_nseconds);
598
599 } else { // ------------ RUNNING IN REALTIME MODE
600 timer.pause(false);
601 actual_elapsed_time = timer.getElapsedTime();
602
603 double sim_lag_time = actual_elapsed_time - FDMExec->GetSimTime(); // How far behind sim-time is from actual elapsed time.
604 double cycle_start = getcurrentseconds();
605
606 for (int i=0; i<(int)(sim_lag_time/frame_duration); i++) { // catch up sim time to actual elapsed time.
607 result = FDMExec->Run();
608 cycle_duration = getcurrentseconds() - cycle_start; // Calculate cycle duration
609 cycle_start = getcurrentseconds(); // Get new current_seconds
610 if (FDMExec->Holding()) break;
611 }
612
613 if (play_nice) sim_nsleep(sleep_nseconds);
614
615 if (FDMExec->GetSimTime() >= new_five_second_value) { // Print out elapsed time every five seconds.
616 cout << "Simulation elapsed time: " << FDMExec->GetSimTime() << endl;
617 new_five_second_value += 5.0;
618 }
619 }
620 } else { // Suspended
621 timer.pause(true);
622 sim_nsleep(sleep_nseconds);
623 result = FDMExec->Run();
624 }
625
626 }
627
628 // PRINT ENDING CLOCK TIME
629 time(&tod);
630#if defined(_MSC_VER) || defined(__MINGW32__)
631 localtime_s(&local, &tod);
632#else
633 localtime_r(&tod, &local);
634#endif
635 strftime(s, 99, "%A %B %d %Y %X", &local);
636 cout << "End: " << s << " (HH:MM:SS)" << endl;
637
638 // CLEAN UP
639 delete FDMExec;
640
641 return 0;
642}
643
644//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
645
646#define gripe cerr << "Option '" << keyword \
647 << "' requires a value, as in '" \
648 << keyword << "=something'" << endl << endl;
649
650bool options(int count, char **arg)
651{
652 int i;
653 bool result = true;
654
655 if (count == 1) {
656 PrintHelp();
657 exit(0);
658 }
659
660 cout.setf(ios_base::fixed);
661
662 for (i=1; i<count; i++) {
663 string argument = string(arg[i]);
664 string keyword(argument);
665 string value("");
666 string::size_type n=argument.find("=");
667
668 if (n != string::npos && n > 0) {
669 keyword = argument.substr(0, n);
670 value = argument.substr(n+1);
671 }
672
673 if (keyword == "--help") {
674 PrintHelp();
675 exit(0);
676 } else if (keyword == "--version") {
677 cout << endl << " JSBSim Version: " << FDMExec->GetVersion() << endl << endl;
678 exit (0);
679 } else if (keyword == "--realtime") {
680 realtime = true;
681 } else if (keyword == "--nice") {
682 play_nice = true;
683 if (n != string::npos) {
684 try {
685 sleep_period = JSBSim::atof_locale_c( value.c_str() );
686 } catch (...) {
687 cerr << endl << " Invalid sleep period given!" << endl << endl;
688 result = false;
689 }
690 } else {
691 sleep_period = 0.01;
692 }
693 } else if (keyword == "--suspend") {
694 suspend = true;
695 } else if (keyword == "--nohighlight") {
696 nohighlight = true;
697 } else if (keyword == "--outputlogfile") {
698 if (n != string::npos) {
699 LogOutputName.push_back(value);
700 }
701 } else if (keyword == "--logdirectivefile") {
702 if (n != string::npos) {
703 LogDirectiveName.push_back(SGPath::fromLocal8Bit(value.c_str()));
704 } else {
705 gripe;
706 exit(1);
707 }
708 } else if (keyword == "--root") {
709 if (n != string::npos) {
710 RootDir = SGPath::fromLocal8Bit(value.c_str());
711 } else {
712 gripe;
713 exit(1);
714 }
715 } else if (keyword == "--aircraft") {
716 if (n != string::npos) {
717 AircraftName = value;
718 } else {
719 gripe;
720 exit(1);
721 }
722 } else if (keyword == "--script") {
723 if (n != string::npos) {
724 ScriptName = SGPath::fromLocal8Bit(value.c_str());
725 } else {
726 gripe;
727 exit(1);
728 }
729 } else if (keyword == "--initfile") {
730 if (n != string::npos) {
731 ResetName = SGPath::fromLocal8Bit(value.c_str());
732 } else {
733 gripe;
734 exit(1);
735 }
736 } else if (keyword == "--planet") {
737 if (n != string::npos) {
738 PlanetName = SGPath::fromLocal8Bit(value.c_str());
739 } else {
740 gripe;
741 exit(1);
742 }
743 } else if (keyword == "--property") {
744 if (n != string::npos) {
745 string propName = value.substr(0,value.find("="));
746 string propValueString = value.substr(value.find("=")+1);
747 double propValue;
748 try {
749 propValue = JSBSim::atof_locale_c(propValueString.c_str());
750 } catch (JSBSim::InvalidNumber&) {
751 gripe;
752 exit(1);
753 }
754 CommandLineProperties.push_back(propName);
755 CommandLinePropertyValues.push_back(propValue);
756 } else {
757 gripe;
758 exit(1);
759 }
760
761 } else if (keyword.substr(0,5) == "--end") {
762 if (n != string::npos) {
763 try {
764 end_time = JSBSim::atof_locale_c( value.c_str() );
765 } catch (...) {
766 cerr << endl << " Invalid end time given!" << endl << endl;
767 result = false;
768 }
769 } else {
770 gripe;
771 exit(1);
772 }
773
774 } else if (keyword == "--simulation-rate") {
775 if (n != string::npos) {
776 try {
777 simulation_rate = JSBSim::atof_locale_c( value.c_str() );
778 override_sim_rate = true;
779 } catch (...) {
780 cerr << endl << " Invalid simulation rate given!" << endl << endl;
781 result = false;
782 }
783 } else {
784 gripe;
785 exit(1);
786 }
787
788 } else if (keyword == "--catalog") {
789 catalog = true;
790 if (!value.empty()) AircraftName=value;
791 } else if (keyword.substr(0,2) != "--" && value.empty() ) {
792 // See what kind of files we are specifying on the command line
793
794 XMLFile xmlFile;
795 SGPath path = SGPath::fromLocal8Bit(keyword.c_str());
796
797 if (xmlFile.IsScriptFile(path)) ScriptName = path;
798 else if (xmlFile.IsLogDirectiveFile(path)) LogDirectiveName.push_back(path);
799 else if (xmlFile.IsAircraftFile(SGPath("aircraft")/keyword/keyword)) AircraftName = keyword;
800 else if (xmlFile.IsInitFile(path)) ResetName = path;
801 else if (xmlFile.IsInitFile(SGPath("aircraft")/AircraftName/keyword)) ResetName = SGPath("aircraft")/AircraftName/keyword;
802 else {
803 cerr << "The argument \"" << keyword << "\" cannot be interpreted as a file name or option." << endl;
804 exit(1);
805 }
806
807 }
808 else //Unknown keyword so print the help file, the bad keyword and abort
809 {
810 PrintHelp();
811 cerr << "The argument \"" << keyword << "\" cannot be interpreted as a file name or option." << endl;
812 exit(1);
813 }
814
815 }
816
817 // Post-processing for script options. check for incompatible options.
818
819 if (catalog && !ScriptName.isNull()) {
820 cerr << "Cannot specify catalog with script option" << endl << endl;
821 result = false;
822 }
823 if (!AircraftName.empty() && ResetName.isNull() && !catalog) {
824 cerr << "You must specify an initialization file with the aircraft name." << endl << endl;
825 result = false;
826 }
827 if (!ScriptName.isNull() && !AircraftName.empty()) {
828 cerr << "You cannot specify an aircraft file with a script." << endl;
829 result = false;
830 }
831
832 return result;
833
834}
835
836//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
837
838void PrintHelp(void)
839{
840 cout << endl << " JSBSim version " << FDMExec->GetVersion() << endl << endl;
841 cout << " Usage: jsbsim [script file name] [output file names] <options>" << endl << endl;
842 cout << " options:" << endl;
843 cout << " --help returns this message" << endl;
844 cout << " --version returns the version number" << endl;
845 cout << " --outputlogfile=<filename> sets (overrides) the name of a data output file" << endl;
846 cout << " --logdirectivefile=<filename> specifies the name of a data logging directives file" << endl;
847 cout << " (can appear multiple times)" << endl;
848 cout << " --root=<path> specifies the JSBSim root directory (where aircraft/, engine/, etc. reside)" << endl;
849 cout << " --aircraft=<filename> specifies the name of the aircraft to be modeled" << endl;
850 cout << " --script=<filename> specifies a script to run" << endl;
851 cout << " --realtime specifies to run in actual real world time" << endl;
852 cout << " --nice specifies to run at lower CPU usage" << endl;
853 cout << " --nohighlight specifies that console output should be pure text only (no color)" << endl;
854 cout << " --suspend specifies to suspend the simulation after initialization" << endl;
855 cout << " --initfile=<filename> specifies an initialization file" << endl;
856 cout << " --planet=<filename> specifies a planet definition file" << endl;
857 cout << " --catalog specifies that all properties for this aircraft model should be printed" << endl;
858 cout << " (catalog=aircraftname is an optional format)" << endl;
859 cout << " --property=<name=value> e.g. --property=simulation/integrator/rate/rotational=1" << endl;
860 cout << " --simulation-rate=<rate (double)> specifies the sim dT time or frequency" << endl;
861 cout << " If rate specified is less than 1, it is interpreted as" << endl;
862 cout << " a time step size, otherwise it is assumed to be a rate in Hertz." << endl;
863 cout << " --end=<time (double)> specifies the sim end time" << endl << endl;
864
865 cout << " NOTE: There can be no spaces around the = sign when" << endl;
866 cout << " an option is followed by a filename" << endl << endl;
867}
const std::string & GetName(void) const
Retrieves the element name.
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
void SetPropertyValue(const std::string &property, double value)
Sets a property value.
Definition FGFDMExec.h:414
std::shared_ptr< FGInitialCondition > GetIC(void) const
Returns a pointer to the FGInitialCondition object.
Definition FGFDMExec.h:389
bool SetSystemsPath(const SGPath &path)
Set the path to the systems config file directories.
Definition FGFDMExec.h:336
bool SetOutputFileName(const int n, const std::string &fname)
Sets (or overrides) the output filename.
Definition FGFDMExec.h:459
void SetDebugLevel(int level)
Sets the debug level.
Definition FGFDMExec.h:509
bool LoadScript(const SGPath &Script, double deltaT=0.0, const SGPath &initfile=SGPath())
Load a script.
bool SetEnginePath(const SGPath &path)
Set the path to the engine config file directories.
Definition FGFDMExec.h:313
int GetDebugLevel(void) const
Retrieves the current debug level setting.
Definition FGFDMExec.h:601
bool SetOutputPath(const SGPath &path)
Set the directory where the output files will be written.
Definition FGFDMExec.h:347
void SetRootDir(const SGPath &rootDir)
Set the root directory that is used to obtain absolute paths from relative paths.
Definition FGFDMExec.h:585
std::string GetOutputFileName(int n) const
Retrieves the current output filename.
Definition FGFDMExec.h:465
double GetDeltaT(void) const
Returns the simulation delta T.
Definition FGFDMExec.h:552
std::shared_ptr< FGPropagate > GetPropagate(void) const
Returns the FGPropagate pointer.
void CheckIncrementalHold(void)
Checks if required to hold afer increment.
bool LoadPlanet(const SGPath &PlanetPath, bool useAircraftPath=true)
Loads the planet.
bool Run(void)
This function executes each scheduled model in succession.
bool LoadModel(const SGPath &AircraftPath, const SGPath &EnginePath, const SGPath &SystemsPath, const std::string &model, bool addModelToPath=true)
Loads an aircraft model.
bool SetOutputDirectives(const SGPath &fname)
Sets the output (logging) mechanism for this run.
Definition FGFDMExec.h:446
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
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Definition FGFDMExec.h:421
void Hold(void)
Pauses execution by preventing time from incrementing.
Definition FGFDMExec.h:488
bool SetAircraftPath(const SGPath &path)
Set the path to the aircraft config file directories.
Definition FGFDMExec.h:325
bool RunIC(void)
Initializes the sim from the initial condition object and executes each scheduled model without integ...
bool Holding(void)
Returns true if the simulation is Holding (i.e. simulation time is not moving).
Definition FGFDMExec.h:496
static const std::string & GetVersion(void)
Returns the version number of JSBSim.
Definition FGJSBBase.h:175
static char fggreen[6]
green text
Definition FGJSBBase.h:168
void disableHighLighting(void)
Disables highlighting in the console output.
Definition FGJSBBase.cpp:68
static char reset[5]
resets text properties
Definition FGJSBBase.h:156
static char highint[5]
highlights text
Definition FGJSBBase.h:150
The trimming routine for JSBSim.
Definition FGTrim.h:126
void Report(void)
Print the results of the trim.
Definition FGTrim.cpp:113
bool DoTrim(void)
Execute the trim.
Definition FGTrim.cpp:186
The Timer class measures the elapsed real time and can be paused and resumed.
Definition JSBSim.cpp:190
void pause(bool paused)
Pause the timer if the paused parameter is true and resume it if the paused parameter is false.
Definition JSBSim.cpp:205
void valueChanged(SGPropertyNode *prop) override
Restart the timer when the listened property is modified.
Definition JSBSim.cpp:196
double getElapsedTime(void)
Get the elapsed real time in seconds since the timer was started.
Definition JSBSim.cpp:201
This class is solely for the purpose of determining what type of file is given on the command line.
Definition JSBSim.cpp:155