42#include "initialization/FGTrim.h"
43#include "initialization/FGInitialCondition.h"
45#include "input_output/FGXMLFileRead.h"
46#include "input_output/string_utilities.h"
48#if !defined(__GNUC__) && !defined(sgi) && !defined(_MSC_VER)
56#elif defined(__GNUC__) && !defined(sgi)
57# define __GNU_VISIBLE 1
61#if defined(__BORLANDC__) || defined(_MSC_VER) || defined(__MINGW32__)
62# define WIN32_LEAN_AND_MEAN
66# include <sys/types.h>
67# include <sys/timeb.h>
73#if defined(__MINGW64_VERSION_MAJOR) && __MINGW64_VERSION_MAJOR < 7
74#define ENABLE_VIRTUAL_TERMINAL_PROCESSING 0x4
93vector <string> LogOutputName;
94vector <SGPath> LogDirectiveName;
95vector <string> CommandLineProperties;
96vector <double> CommandLinePropertyValues;
106double end_time = 1e99;
107double simulation_rate = 1./120.;
108bool override_sim_rate =
false;
109double sleep_period=0.01;
115bool options(
int,
char**);
116int real_main(
int argc,
char* argv[]);
119#if defined(__BORLANDC__) || defined(_MSC_VER) || defined(__MINGW32__)
120 double getcurrentseconds(
void)
124 return tm_ptr.time + tm_ptr.millitm*0.001;
127 double getcurrentseconds(
void)
132 gettimeofday(&tval, &tz);
133 return (tval.tv_sec + tval.tv_usec*1e-6);
137#if defined(__BORLANDC__) || defined(_MSC_VER) || defined(__MINGW32__)
138 void sim_nsleep(
long nanosec)
140 Sleep((DWORD)(nanosec*1e-6));
143 void sim_nsleep(
long nanosec)
145 struct timespec ts, ts1;
148 ts.tv_nsec = nanosec;
149 nanosleep(&ts, &ts1);
157 bool IsScriptFile(
const SGPath& filename) {
159 Element *document = LoadXMLDocument(filename,
false);
160 if (document && document->
GetName() ==
"runscript") result =
true;
164 bool IsLogDirectiveFile(
const SGPath& filename) {
166 Element *document = LoadXMLDocument(filename,
false);
167 if (document && document->
GetName() ==
"output") result =
true;
171 bool IsAircraftFile(
const SGPath& filename) {
173 Element* document = LoadXMLDocument(filename,
false);
174 if (document && document->
GetName() ==
"fdm_config") result =
true;
178 bool IsInitFile(
const SGPath& filename) {
180 Element *document = LoadXMLDocument(filename,
false);
181 if (document && document->
GetName() ==
"initialize") result =
true;
190class Timer :
public SGPropertyChangeListener {
192 Timer() : SGPropertyChangeListener(), isPaused(
false) { start(); }
193 void start(
void) { initial_seconds = getcurrentseconds(); }
198 if (isPaused) pause_start_seconds = initial_seconds;
209 pause_start_seconds = getcurrentseconds();
214 double pause_duration = getcurrentseconds() - pause_start_seconds;
215 initial_seconds += pause_duration;
220 double initial_seconds = 0.0;
221 bool isPaused =
false;
222 double pause_start_seconds = 0.0;
327int main(
int argc,
char* argv[])
329#if defined(_MSC_VER) || defined(__MINGW32__)
331 _controlfp(_controlfp(0, 0) & ~(_EM_INVALID | _EM_ZERODIVIDE | _EM_OVERFLOW),
333#elif defined(__GNUC__) && !defined(sgi) && !defined(__APPLE__)
334 feenableexcept(FE_DIVBYZERO | FE_INVALID | FE_OVERFLOW);
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;
343 }
catch (
const char* msg) {
344 std::cerr <<
"FATAL ERROR: JSBSim terminated with an exception."
345 << std::endl <<
"The message was: " << msg << std::endl;
348 std::cerr <<
"FATAL ERROR: JSBSim terminated with an exception."
349 << std::endl <<
"The message was: " << e.what() << std::endl;
352 std::cerr <<
"FATAL ERROR: JSBSim terminated with an unknown exception."
359int real_main(
int argc,
char* argv[])
367 LogOutputName.clear();
368 LogDirectiveName.clear();
369 bool result =
false, success;
370 double frame_duration;
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;
385 success = options(argc, argv);
398 FDMExec->
GetPropertyManager()->Tie(
"simulation/frame_start_time", &actual_elapsed_time);
402 SGPropertyNode_ptr reset_node = FDMExec->
GetPropertyManager()->GetNode(
"simulation/reset");
403 reset_node->addChangeListener(&timer);
409 HANDLE hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
411 GetConsoleMode(hStdOut, &dwMode);
412 if ((dwMode & ENABLE_VIRTUAL_TERMINAL_PROCESSING) == 0)
418 if (simulation_rate < 1.0 )
419 FDMExec->
Setdt(simulation_rate);
421 FDMExec->
Setdt(1.0/simulation_rate);
423 if (override_sim_rate) override_sim_rate_value = FDMExec->
GetDeltaT();
427 for (
unsigned int i=0; i<CommandLineProperties.size(); i++) {
429 if (CommandLineProperties[i].find(
"simulation") != std::string::npos) {
431 FDMExec->
SetPropertyValue(CommandLineProperties[i], CommandLinePropertyValues[i]);
436 if (!PlanetName.isNull()) {
437 result = FDMExec->
LoadPlanet(PlanetName,
false);
440 cerr <<
"Planet file " << PlanetName <<
" was not successfully loaded" << endl;
447 if (!ScriptName.isNull()) {
449 result = FDMExec->
LoadScript(ScriptName, override_sim_rate_value, ResetName);
452 cerr <<
"Script file " << ScriptName <<
" was not successfully loaded" << endl;
458 }
else if (!AircraftName.empty() || !ResetName.isNull()) {
462 if ( ! FDMExec->
LoadModel(SGPath(
"aircraft"),
466 cerr <<
" JSBSim could not be started" << endl << endl;
472 FDMExec->PrintPropertyCatalog();
477 auto IC = FDMExec->
GetIC();
478 if ( ! IC->Load(ResetName)) {
480 cerr <<
"Initialization unsuccessful" << endl;
485 cout <<
" No Aircraft, Script, or Reset information given" << endl << endl;
491 for (
unsigned int i=0; i<LogDirectiveName.size(); i++) {
492 if (!LogDirectiveName[i].isNull()) {
494 cout <<
"Output directives not properly set in file " << LogDirectiveName[i] << endl;
505 for (
unsigned int i=0; i<LogOutputName.size(); i++) {
508 cout <<
"Output filename could not be set" << endl;
510 cout <<
"Output filename change from " << old_filename <<
" from aircraft"
511 " configuration file to " << LogOutputName[i] <<
" specified on"
512 " command line" << endl;
518 for (
unsigned int i=0; i<CommandLineProperties.size(); i++) {
521 cerr << endl <<
" No property by the name " << CommandLineProperties[i] << endl;
525 FDMExec->
SetPropertyValue(CommandLineProperties[i], CommandLinePropertyValues[i]);
532 FDMExec->PrintSimulationConfiguration();
538 JSBSim::TrimMode icTrimRequested = (JSBSim::TrimMode)FDMExec->
GetIC()->TrimRequested();
539 if (icTrimRequested != JSBSim::TrimMode::tNone) {
548 }
catch (
string& msg) {
549 cerr << endl << msg << endl << endl;
555 <<
"---- JSBSim Execution beginning ... --------------------------------------------"
558 result = FDMExec->
Run();
560 if (suspend) FDMExec->
Hold();
567#if defined(_MSC_VER) || defined(__MINGW32__)
568 localtime_s(&local, &tod);
570 localtime_r(&tod, &local);
572 strftime(s, 99,
"%A %B %d %Y %X", &local);
573 cout <<
"Start: " << s <<
" (HH:MM:SS)" << endl;
576 if (realtime) sleep_nseconds = (long)(frame_duration*1e9);
577 else sleep_nseconds = (sleep_period )*1e9;
583 while (result && FDMExec->
GetSimTime() <= end_time) {
595 result = FDMExec->
Run();
597 if (play_nice) sim_nsleep(sleep_nseconds);
603 double sim_lag_time = actual_elapsed_time - FDMExec->
GetSimTime();
604 double cycle_start = getcurrentseconds();
606 for (
int i=0; i<(int)(sim_lag_time/frame_duration); i++) {
607 result = FDMExec->
Run();
608 cycle_duration = getcurrentseconds() - cycle_start;
609 cycle_start = getcurrentseconds();
613 if (play_nice) sim_nsleep(sleep_nseconds);
615 if (FDMExec->
GetSimTime() >= new_five_second_value) {
616 cout <<
"Simulation elapsed time: " << FDMExec->
GetSimTime() << endl;
617 new_five_second_value += 5.0;
622 sim_nsleep(sleep_nseconds);
623 result = FDMExec->
Run();
630#if defined(_MSC_VER) || defined(__MINGW32__)
631 localtime_s(&local, &tod);
633 localtime_r(&tod, &local);
635 strftime(s, 99,
"%A %B %d %Y %X", &local);
636 cout <<
"End: " << s <<
" (HH:MM:SS)" << endl;
646#define gripe cerr << "Option '" << keyword \
647 << "' requires a value, as in '" \
648 << keyword << "=something'" << endl << endl;
650bool options(
int count,
char **arg)
660 cout.setf(ios_base::fixed);
662 for (i=1; i<count; i++) {
663 string argument = string(arg[i]);
664 string keyword(argument);
666 string::size_type n=argument.find(
"=");
668 if (n != string::npos && n > 0) {
669 keyword = argument.substr(0, n);
670 value = argument.substr(n+1);
673 if (keyword ==
"--help") {
676 }
else if (keyword ==
"--version") {
677 cout << endl <<
" JSBSim Version: " << FDMExec->
GetVersion() << endl << endl;
679 }
else if (keyword ==
"--realtime") {
681 }
else if (keyword ==
"--nice") {
683 if (n != string::npos) {
685 sleep_period = JSBSim::atof_locale_c( value.c_str() );
687 cerr << endl <<
" Invalid sleep period given!" << endl << endl;
693 }
else if (keyword ==
"--suspend") {
695 }
else if (keyword ==
"--nohighlight") {
697 }
else if (keyword ==
"--outputlogfile") {
698 if (n != string::npos) {
699 LogOutputName.push_back(value);
701 }
else if (keyword ==
"--logdirectivefile") {
702 if (n != string::npos) {
703 LogDirectiveName.push_back(SGPath::fromLocal8Bit(value.c_str()));
708 }
else if (keyword ==
"--root") {
709 if (n != string::npos) {
710 RootDir = SGPath::fromLocal8Bit(value.c_str());
715 }
else if (keyword ==
"--aircraft") {
716 if (n != string::npos) {
717 AircraftName = value;
722 }
else if (keyword ==
"--script") {
723 if (n != string::npos) {
724 ScriptName = SGPath::fromLocal8Bit(value.c_str());
729 }
else if (keyword ==
"--initfile") {
730 if (n != string::npos) {
731 ResetName = SGPath::fromLocal8Bit(value.c_str());
736 }
else if (keyword ==
"--planet") {
737 if (n != string::npos) {
738 PlanetName = SGPath::fromLocal8Bit(value.c_str());
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);
749 propValue = JSBSim::atof_locale_c(propValueString.c_str());
754 CommandLineProperties.push_back(propName);
755 CommandLinePropertyValues.push_back(propValue);
761 }
else if (keyword.substr(0,5) ==
"--end") {
762 if (n != string::npos) {
764 end_time = JSBSim::atof_locale_c( value.c_str() );
766 cerr << endl <<
" Invalid end time given!" << endl << endl;
774 }
else if (keyword ==
"--simulation-rate") {
775 if (n != string::npos) {
777 simulation_rate = JSBSim::atof_locale_c( value.c_str() );
778 override_sim_rate =
true;
780 cerr << endl <<
" Invalid simulation rate given!" << endl << endl;
788 }
else if (keyword ==
"--catalog") {
790 if (!value.empty()) AircraftName=value;
791 }
else if (keyword.substr(0,2) !=
"--" && value.empty() ) {
795 SGPath path = SGPath::fromLocal8Bit(keyword.c_str());
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;
803 cerr <<
"The argument \"" << keyword <<
"\" cannot be interpreted as a file name or option." << endl;
811 cerr <<
"The argument \"" << keyword <<
"\" cannot be interpreted as a file name or option." << endl;
819 if (catalog && !ScriptName.isNull()) {
820 cerr <<
"Cannot specify catalog with script option" << endl << endl;
823 if (!AircraftName.empty() && ResetName.isNull() && !catalog) {
824 cerr <<
"You must specify an initialization file with the aircraft name." << endl << endl;
827 if (!ScriptName.isNull() && !AircraftName.empty()) {
828 cerr <<
"You cannot specify an aircraft file with a script." << endl;
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;
865 cout <<
" NOTE: There can be no spaces around the = sign when" << endl;
866 cout <<
" an option is followed by a filename" << endl << endl;
const std::string & GetName(void) const
Retrieves the element name.
Encapsulates the JSBSim simulation executive.
void SetPropertyValue(const std::string &property, double value)
Sets a property value.
std::shared_ptr< FGInitialCondition > GetIC(void) const
Returns a pointer to the FGInitialCondition object.
bool SetSystemsPath(const SGPath &path)
Set the path to the systems config file directories.
bool SetOutputFileName(const int n, const std::string &fname)
Sets (or overrides) the output filename.
void SetDebugLevel(int level)
Sets the debug level.
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.
int GetDebugLevel(void) const
Retrieves the current debug level setting.
bool SetOutputPath(const SGPath &path)
Set the directory where the output files will be written.
void SetRootDir(const SGPath &rootDir)
Set the root directory that is used to obtain absolute paths from relative paths.
std::string GetOutputFileName(int n) const
Retrieves the current output filename.
double GetDeltaT(void) const
Returns the simulation delta T.
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.
double GetSimTime(void) const
Returns the cumulative simulation time in seconds.
void Setdt(double delta_t)
Sets the integration time step for the simulation executive.
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
void Hold(void)
Pauses execution by preventing time from incrementing.
bool SetAircraftPath(const SGPath &path)
Set the path to the aircraft config file directories.
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).
static const std::string & GetVersion(void)
Returns the version number of JSBSim.
static char fggreen[6]
green text
void disableHighLighting(void)
Disables highlighting in the console output.
static char reset[5]
resets text properties
static char highint[5]
highlights text
The trimming routine for JSBSim.
void Report(void)
Print the results of the trim.
bool DoTrim(void)
Execute the trim.
The Timer class measures the elapsed real time and can be paused and resumed.
void pause(bool paused)
Pause the timer if the paused parameter is true and resume it if the paused parameter is false.
void valueChanged(SGPropertyNode *prop) override
Restart the timer when the listened property is modified.
double getElapsedTime(void)
Get the elapsed real time in seconds since the timer was started.
This class is solely for the purpose of determining what type of file is given on the command line.