2001-07-02 22:14:09 +00:00
|
|
|
// fg_init.cxx -- Flight Gear top level initialization routines
|
1998-04-25 15:11:10 +00:00
|
|
|
//
|
|
|
|
// Written by Curtis Olson, started August 1997.
|
|
|
|
//
|
2004-11-19 22:10:41 +00:00
|
|
|
// Copyright (C) 1997 Curtis L. Olson - http://www.flightgear.org/~curt
|
1998-04-25 15:11:10 +00:00
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License as
|
|
|
|
// published by the Free Software Foundation; either version 2 of the
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful, but
|
|
|
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
2006-02-21 01:16:04 +00:00
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
1998-04-25 15:11:10 +00:00
|
|
|
//
|
|
|
|
// $Id$
|
1997-08-23 01:46:20 +00:00
|
|
|
|
|
|
|
|
1998-04-24 00:49:17 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
1998-04-22 13:25:39 +00:00
|
|
|
|
2013-11-14 16:48:14 +00:00
|
|
|
#include <simgear/compiler.h>
|
|
|
|
|
1997-08-23 01:46:20 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2002-10-04 13:20:53 +00:00
|
|
|
#include <string.h> // strcmp()
|
1998-07-22 21:40:43 +00:00
|
|
|
|
2013-11-14 16:48:14 +00:00
|
|
|
#if defined(SG_WINDOWS)
|
2008-10-01 15:03:53 +00:00
|
|
|
# include <io.h> // isatty()
|
2013-11-14 16:48:14 +00:00
|
|
|
# include <process.h> // _getpid()
|
|
|
|
# include <Windows.h>
|
2008-10-01 15:03:53 +00:00
|
|
|
# define isatty _isatty
|
2013-11-16 09:31:18 +00:00
|
|
|
#else
|
|
|
|
// for open() and options
|
|
|
|
# include <sys/types.h>
|
|
|
|
# include <sys/stat.h>
|
|
|
|
# include <fcntl.h>
|
2007-06-16 21:35:16 +00:00
|
|
|
#endif
|
2001-03-22 04:02:11 +00:00
|
|
|
|
2008-07-25 18:38:29 +00:00
|
|
|
#include <string>
|
2013-11-16 12:10:32 +00:00
|
|
|
|
2009-12-05 14:25:31 +00:00
|
|
|
#include <boost/algorithm/string/compare.hpp>
|
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
1997-08-23 01:46:20 +00:00
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
#include <osgViewer/Viewer>
|
|
|
|
|
2013-12-09 22:13:45 +00:00
|
|
|
#include <simgear/canvas/Canvas.hxx>
|
2000-02-15 03:30:01 +00:00
|
|
|
#include <simgear/constants.h>
|
2000-02-16 23:01:03 +00:00
|
|
|
#include <simgear/debug/logstream.hxx>
|
2003-09-24 17:20:55 +00:00
|
|
|
#include <simgear/structure/exception.hxx>
|
|
|
|
#include <simgear/structure/event_mgr.hxx>
|
2011-11-19 14:37:49 +00:00
|
|
|
#include <simgear/structure/SGPerfMon.hxx>
|
2001-03-25 14:20:12 +00:00
|
|
|
#include <simgear/misc/sg_path.hxx>
|
2010-07-06 20:06:50 +00:00
|
|
|
#include <simgear/misc/sg_dir.hxx>
|
2010-10-20 16:54:16 +00:00
|
|
|
#include <simgear/misc/sgstream.hxx>
|
2011-07-18 09:09:43 +00:00
|
|
|
#include <simgear/misc/strutils.hxx>
|
2011-07-30 09:47:28 +00:00
|
|
|
#include <simgear/props/props_io.hxx>
|
2013-11-16 12:10:32 +00:00
|
|
|
#include <simgear/scene/tsync/terrasync.hxx>
|
2010-10-20 16:54:16 +00:00
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
#include <simgear/scene/model/modellib.hxx>
|
2003-05-12 21:34:29 +00:00
|
|
|
#include <simgear/scene/material/matlib.hxx>
|
2013-12-04 09:09:47 +00:00
|
|
|
#include <simgear/scene/material/Effect.hxx>
|
2008-02-15 06:45:19 +00:00
|
|
|
#include <simgear/scene/model/particles.hxx>
|
2013-09-28 13:07:45 +00:00
|
|
|
#include <simgear/scene/tsync/terrasync.hxx>
|
2000-02-15 03:30:01 +00:00
|
|
|
|
2014-05-31 17:04:56 +00:00
|
|
|
#include <simgear/package/Root.hxx>
|
2014-06-11 17:19:41 +00:00
|
|
|
#include <simgear/package/Package.hxx>
|
|
|
|
#include <simgear/package/Install.hxx>
|
|
|
|
#include <simgear/package/Catalog.hxx>
|
2014-05-31 17:04:56 +00:00
|
|
|
|
2008-07-29 08:27:48 +00:00
|
|
|
#include <Aircraft/controls.hxx>
|
2005-11-01 13:41:49 +00:00
|
|
|
#include <Aircraft/replay.hxx>
|
2012-12-10 18:11:44 +00:00
|
|
|
#include <Aircraft/FlightHistory.hxx>
|
2000-08-16 00:09:03 +00:00
|
|
|
#include <Airports/runways.hxx>
|
2013-02-21 11:32:02 +00:00
|
|
|
#include <Airports/airport.hxx>
|
2008-08-14 18:13:39 +00:00
|
|
|
#include <Airports/dynamics.hxx>
|
2015-12-05 00:25:29 +00:00
|
|
|
#include <Airports/airportdynamicsmanager.hxx>
|
2008-08-14 18:13:39 +00:00
|
|
|
|
Mathias Frhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
#include <AIModel/AIManager.hxx>
|
2009-12-30 14:11:16 +00:00
|
|
|
|
2011-04-03 15:58:16 +00:00
|
|
|
#include <ATC/atc_mgr.hxx>
|
2009-12-30 14:11:16 +00:00
|
|
|
|
2004-01-31 19:47:45 +00:00
|
|
|
#include <Autopilot/route_mgr.hxx>
|
2010-06-24 15:09:33 +00:00
|
|
|
#include <Autopilot/autopilotgroup.hxx>
|
2009-09-21 13:57:39 +00:00
|
|
|
|
User-visible
- knobs now continue to rotate when you hold down the mouse
- the middle mouse button makes knobs rotate much faster
- there are NAV1, NAV2, and ADF radios that can be tuned using the mouse
- there are standby frequencies for NAV1 and NAV2, and buttons to swap
- there is a crude, rather silly-looking DME, hard-wired to NAV1
- there is a crude, rather silly-looking autopilot that can lock
the heading (to the bug on the gyro), can lock to NAV1, and can lock
the current altitude
- the knobs for changing the radials on NAV1 and NAV2 look much better
and are in the right place
- tuning into an ILS frequency doesn't change the displayed radial for
NAV1
Code
- I've created a new module, sp_panel.[ch]xx, that constructs the
default single-prop panel; this works entirely outside of FGPanel,
so it is possible to construct similar modules for other sorts of
panels; all code specific to the default panel has been removed from
panel.cxx
- current_panel is now a pointer
- radiostack.[ch]xx keeps track both of the actual radial and of the
selected radial (they will differ with ILS); the NAV gauges should
not spin around automatically to show the actual radial (we need to
do something similar with the autopilot)
- the panel is initialized fairly early
- make sure that standby frequencies also get initialized
- I've started combining and clipping small textures to save texture
memory; there's a lot more to do, but at least I've made a start
2000-05-02 18:26:00 +00:00
|
|
|
#include <Cockpit/panel.hxx>
|
2000-09-22 17:20:56 +00:00
|
|
|
#include <Cockpit/panel_io.hxx>
|
2009-12-30 14:11:16 +00:00
|
|
|
|
2012-04-21 13:31:20 +00:00
|
|
|
#include <Canvas/canvas_mgr.hxx>
|
2012-07-27 11:17:42 +00:00
|
|
|
#include <Canvas/gui_mgr.hxx>
|
2013-12-09 22:13:45 +00:00
|
|
|
#include <Canvas/FGCanvasSystemAdapter.hxx>
|
2002-11-07 16:27:47 +00:00
|
|
|
#include <GUI/new_gui.hxx>
|
2013-11-06 23:49:58 +00:00
|
|
|
#include <GUI/MessageBox.hxx>
|
I'm attaching diffs to add a new FGInput module to FlightGear
(src/Input). So far, FGInput replaces most of src/Main/keyboard.cxx
(I've left a tiny stub); in the very near future, it will also take
over control of the joystick, mouse (Norm permitting), and panel
instrument interactions, so that there is a single mechanism for
configuring all input devices.
The new format should be (close to) self-explanatory by looking at the
new base-package file keyboard.xml, which is now included by
preferences.xml (I'll do the same thing for the joystick when I have a
chance). I have not managed to move all keybindings into this file
yet, but I've made a good start. I'm including Tony in the recipient
list so that he can see how bindings can use an external XML file.
This patch also adds support for multiple bindings for a single key,
special keys (i.e. keypad and function keys), and key modifiers
(shift/alt/ctrl); special keys use the PUI convention of adding 256 to
the Glut key code.
Unfortunately, everything comes with a price; in this case, I have not
yet found a general mechanism for the old (hard-coded) modal bindings,
which behaved differently depending on the autopilot state (i.e. left
rudder or move AP heading left); with my patches, this functionality
disappears, but you can still adjust the autopilot using the panel or
the GUI input dialogs.
2001-05-23 23:01:15 +00:00
|
|
|
#include <Input/input.hxx>
|
2002-09-24 14:51:37 +00:00
|
|
|
#include <Instrumentation/instrument_mgr.hxx>
|
2002-04-09 18:58:24 +00:00
|
|
|
#include <Model/acmodel.hxx>
|
2010-10-22 11:22:00 +00:00
|
|
|
#include <Model/modelmgr.hxx>
|
2004-10-22 09:58:24 +00:00
|
|
|
#include <AIModel/submodel.hxx>
|
2003-11-28 15:48:05 +00:00
|
|
|
#include <AIModel/AIManager.hxx>
|
2004-05-28 05:24:54 +00:00
|
|
|
#include <Navaids/navdb.hxx>
|
2000-04-24 23:51:26 +00:00
|
|
|
#include <Navaids/navlist.hxx>
|
1998-04-30 12:34:17 +00:00
|
|
|
#include <Scenery/scenery.hxx>
|
2014-03-13 18:18:23 +00:00
|
|
|
#include <Scenery/SceneryPager.hxx>
|
1998-04-22 13:25:39 +00:00
|
|
|
#include <Scenery/tilemgr.hxx>
|
2003-11-25 21:08:36 +00:00
|
|
|
#include <Scripting/NasalSys.hxx>
|
2006-02-11 12:02:40 +00:00
|
|
|
#include <Sound/voice.hxx>
|
2012-09-17 11:36:38 +00:00
|
|
|
#include <Sound/soundmanager.hxx>
|
2002-09-23 19:55:10 +00:00
|
|
|
#include <Systems/system_mgr.hxx>
|
1998-04-22 13:25:39 +00:00
|
|
|
#include <Time/light.hxx>
|
2013-11-16 12:10:32 +00:00
|
|
|
#include <Time/TimeManager.hxx>
|
|
|
|
|
2004-06-03 17:59:14 +00:00
|
|
|
#include <Traffic/TrafficMgr.hxx>
|
2005-11-01 13:41:49 +00:00
|
|
|
#include <MultiPlayer/multiplaymgr.hxx>
|
2010-06-16 07:31:59 +00:00
|
|
|
#include <FDM/fdm_shell.hxx>
|
2011-11-05 13:18:36 +00:00
|
|
|
#include <Environment/ephemeris.hxx>
|
2004-03-16 20:19:07 +00:00
|
|
|
#include <Environment/environment_mgr.hxx>
|
2012-04-25 21:28:00 +00:00
|
|
|
#include <Viewer/renderer.hxx>
|
|
|
|
#include <Viewer/viewmgr.hxx>
|
2013-11-16 12:10:32 +00:00
|
|
|
#include <Viewer/FGEventHandler.hxx>
|
2012-08-27 23:26:36 +00:00
|
|
|
#include <Navaids/NavDataCache.hxx>
|
2012-09-26 16:02:11 +00:00
|
|
|
#include <Instrumentation/HUD/HUD.hxx>
|
2012-09-26 19:30:24 +00:00
|
|
|
#include <Cockpit/cockpitDisplayManager.hxx>
|
2012-11-19 11:30:12 +00:00
|
|
|
#include <Network/HTTPClient.hxx>
|
2013-05-19 14:29:49 +00:00
|
|
|
#include <Network/fgcom.hxx>
|
2014-03-03 20:59:16 +00:00
|
|
|
#include <Network/http/httpd.hxx>
|
2014-05-31 17:04:56 +00:00
|
|
|
#include <Include/version.h>
|
2015-03-19 17:01:38 +00:00
|
|
|
#include <Viewer/splash.hxx>
|
2013-11-16 12:10:32 +00:00
|
|
|
#include <Viewer/CameraGroup.hxx>
|
|
|
|
|
1998-04-22 13:25:39 +00:00
|
|
|
#include "fg_init.hxx"
|
1999-11-19 02:10:24 +00:00
|
|
|
#include "fg_io.hxx"
|
2001-06-01 17:52:28 +00:00
|
|
|
#include "fg_commands.hxx"
|
2001-06-05 22:12:08 +00:00
|
|
|
#include "fg_props.hxx"
|
2013-03-16 15:43:55 +00:00
|
|
|
#include "FGInterpolator.hxx"
|
2001-01-13 22:06:39 +00:00
|
|
|
#include "options.hxx"
|
2000-07-08 06:29:19 +00:00
|
|
|
#include "globals.hxx"
|
2002-03-12 16:29:00 +00:00
|
|
|
#include "logger.hxx"
|
2005-12-21 13:36:04 +00:00
|
|
|
#include "main.hxx"
|
2012-09-18 19:50:28 +00:00
|
|
|
#include "positioninit.hxx"
|
2013-04-29 20:15:32 +00:00
|
|
|
#include "util.hxx"
|
2013-11-16 14:22:34 +00:00
|
|
|
#include "AircraftDirVisitorBase.hxx"
|
1998-04-22 13:25:39 +00:00
|
|
|
|
2013-11-16 10:05:09 +00:00
|
|
|
#if defined(SG_MAC)
|
|
|
|
#include <GUI/CocoaHelpers.h> // for Mac impl of platformDefaultDataPath()
|
|
|
|
#endif
|
|
|
|
|
2008-07-27 16:25:13 +00:00
|
|
|
using std::string;
|
2012-11-23 20:00:20 +00:00
|
|
|
using std::endl;
|
|
|
|
using std::cerr;
|
|
|
|
using std::cout;
|
2014-05-31 17:04:56 +00:00
|
|
|
|
2009-12-05 14:25:31 +00:00
|
|
|
using namespace boost::algorithm;
|
2014-05-31 17:04:56 +00:00
|
|
|
using namespace simgear::pkg;
|
2003-05-09 18:44:33 +00:00
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
extern osg::ref_ptr<osgViewer::Viewer> viewer;
|
2002-11-16 20:17:11 +00:00
|
|
|
|
2001-07-16 19:17:58 +00:00
|
|
|
// Return the current base package version
|
2015-03-06 18:52:06 +00:00
|
|
|
string fgBasePackageVersion(const SGPath& base_path) {
|
|
|
|
SGPath p(base_path);
|
|
|
|
p.append("version");
|
|
|
|
if (!p.exists()) {
|
2013-11-06 23:49:58 +00:00
|
|
|
return string();
|
|
|
|
}
|
|
|
|
|
2015-03-06 18:52:06 +00:00
|
|
|
sg_gzifstream in( p.str() );
|
2013-11-06 23:49:58 +00:00
|
|
|
if (!in.is_open()) {
|
|
|
|
return string();
|
2001-07-16 19:17:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
string version;
|
|
|
|
in >> version;
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
2013-11-16 14:22:34 +00:00
|
|
|
class FindAndCacheAircraft : public AircraftDirVistorBase
|
2010-07-06 20:06:50 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
FindAndCacheAircraft(SGPropertyNode* autoSave)
|
|
|
|
{
|
|
|
|
_cache = autoSave->getNode("sim/startup/path-cache", true);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool loadAircraft()
|
|
|
|
{
|
2010-11-29 21:48:37 +00:00
|
|
|
std::string aircraft = fgGetString( "/sim/aircraft", "");
|
2010-07-06 20:06:50 +00:00
|
|
|
if (aircraft.empty()) {
|
2013-11-06 23:49:58 +00:00
|
|
|
flightgear::fatalMessageBox("No aircraft", "No aircraft was specified");
|
2010-07-06 20:06:50 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "no aircraft specified");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_searchAircraft = aircraft + "-set.xml";
|
2011-10-16 20:15:54 +00:00
|
|
|
std::string aircraftDir = fgGetString("/sim/aircraft-dir", "");
|
|
|
|
if (!aircraftDir.empty()) {
|
|
|
|
// aircraft-dir was set, skip any searching at all, if it's valid
|
|
|
|
simgear::Dir acPath(aircraftDir);
|
|
|
|
SGPath setFile = acPath.file(_searchAircraft);
|
|
|
|
if (setFile.exists()) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "found aircraft in dir: " << aircraftDir );
|
|
|
|
|
|
|
|
try {
|
|
|
|
readProperties(setFile.str(), globals->get_props());
|
|
|
|
} catch ( const sg_exception &e ) {
|
|
|
|
SG_LOG(SG_INPUT, SG_ALERT, "Error reading aircraft: " << e.getFormattedMessage());
|
2013-11-06 23:49:58 +00:00
|
|
|
flightgear::fatalMessageBox("Error reading aircraft",
|
|
|
|
"An error occured reading the requested aircraft (" + aircraft + ")",
|
|
|
|
e.getFormattedMessage());
|
2011-10-16 20:15:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "aircraft '" << _searchAircraft <<
|
|
|
|
"' not found in specified dir:" << aircraftDir);
|
2013-11-06 23:49:58 +00:00
|
|
|
flightgear::fatalMessageBox("Aircraft not found",
|
|
|
|
"The requested aircraft '" + aircraft + "' could not be found in the specified location.",
|
|
|
|
aircraftDir);
|
2011-10-16 20:15:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-06 20:06:50 +00:00
|
|
|
if (!checkCache()) {
|
|
|
|
// prepare cache for re-scan
|
|
|
|
SGPropertyNode *n = _cache->getNode("fg-root", true);
|
|
|
|
n->setStringValue(globals->get_fg_root().c_str());
|
|
|
|
n->setAttribute(SGPropertyNode::USERARCHIVE, true);
|
2010-11-29 21:48:37 +00:00
|
|
|
n = _cache->getNode("fg-aircraft", true);
|
|
|
|
n->setStringValue(getAircraftPaths().c_str());
|
|
|
|
n->setAttribute(SGPropertyNode::USERARCHIVE, true);
|
2010-07-06 20:06:50 +00:00
|
|
|
_cache->removeChildren("aircraft");
|
2010-07-18 12:21:30 +00:00
|
|
|
|
2013-11-16 14:22:34 +00:00
|
|
|
visitAircraftPaths();
|
2010-07-06 20:06:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_foundPath.str().empty()) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Cannot find specified aircraft: " << aircraft );
|
2013-11-06 23:49:58 +00:00
|
|
|
flightgear::fatalMessageBox("Aircraft not found",
|
|
|
|
"The requested aircraft '" + aircraft + "' could not be found in any of the search paths");
|
|
|
|
|
2010-07-06 20:06:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Loading aircraft -set file from:" << _foundPath.str());
|
|
|
|
fgSetString( "/sim/aircraft-dir", _foundPath.dir().c_str());
|
|
|
|
if (!_foundPath.exists()) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Unable to find -set file:" << _foundPath.str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
readProperties(_foundPath.str(), globals->get_props());
|
|
|
|
} catch ( const sg_exception &e ) {
|
|
|
|
SG_LOG(SG_INPUT, SG_ALERT, "Error reading aircraft: " << e.getFormattedMessage());
|
2013-11-06 23:49:58 +00:00
|
|
|
flightgear::fatalMessageBox("Error reading aircraft",
|
|
|
|
"An error occured reading the requested aircraft (" + aircraft + ")",
|
|
|
|
e.getFormattedMessage());
|
2010-07-06 20:06:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2010-11-29 21:48:37 +00:00
|
|
|
SGPath getAircraftPaths() {
|
|
|
|
string_list pathList = globals->get_aircraft_paths();
|
|
|
|
SGPath aircraftPaths;
|
|
|
|
string_list::const_iterator it = pathList.begin();
|
|
|
|
if (it != pathList.end()) {
|
|
|
|
aircraftPaths.set(*it);
|
|
|
|
it++;
|
|
|
|
}
|
|
|
|
for (; it != pathList.end(); ++it) {
|
|
|
|
aircraftPaths.add(*it);
|
|
|
|
}
|
|
|
|
return aircraftPaths;
|
|
|
|
}
|
|
|
|
|
2010-07-06 20:06:50 +00:00
|
|
|
bool checkCache()
|
|
|
|
{
|
|
|
|
if (globals->get_fg_root() != _cache->getStringValue("fg-root", "")) {
|
|
|
|
return false; // cache mismatch
|
|
|
|
}
|
2010-11-29 21:48:37 +00:00
|
|
|
|
|
|
|
if (getAircraftPaths().str() != _cache->getStringValue("fg-aircraft", "")) {
|
|
|
|
return false; // cache mismatch
|
|
|
|
}
|
2010-07-06 20:06:50 +00:00
|
|
|
|
|
|
|
vector<SGPropertyNode_ptr> cache = _cache->getChildren("aircraft");
|
|
|
|
for (unsigned int i = 0; i < cache.size(); i++) {
|
|
|
|
const char *name = cache[i]->getStringValue("file", "");
|
|
|
|
if (!boost::equals(_searchAircraft, name, is_iequal())) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
SGPath xml(cache[i]->getStringValue("path", ""));
|
|
|
|
xml.append(name);
|
|
|
|
if (xml.exists()) {
|
|
|
|
_foundPath = xml;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
} // of aircraft in cache iteration
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-11-16 14:22:34 +00:00
|
|
|
virtual VisitResult visit(const SGPath& p)
|
2010-07-06 20:06:50 +00:00
|
|
|
{
|
|
|
|
// create cache node
|
|
|
|
int i = 0;
|
|
|
|
while (1) {
|
|
|
|
if (!_cache->getChild("aircraft", i++, false))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
SGPropertyNode *n, *entry = _cache->getChild("aircraft", --i, true);
|
2007-04-02 23:19:36 +00:00
|
|
|
|
2010-07-06 20:06:50 +00:00
|
|
|
std::string fileName(p.file());
|
|
|
|
n = entry->getNode("file", true);
|
|
|
|
n->setStringValue(fileName);
|
|
|
|
n->setAttribute(SGPropertyNode::USERARCHIVE, true);
|
2007-04-02 23:19:36 +00:00
|
|
|
|
2010-07-06 20:06:50 +00:00
|
|
|
n = entry->getNode("path", true);
|
|
|
|
n->setStringValue(p.dir());
|
|
|
|
n->setAttribute(SGPropertyNode::USERARCHIVE, true);
|
2007-04-02 23:19:36 +00:00
|
|
|
|
2010-07-06 20:06:50 +00:00
|
|
|
if ( boost::equals(fileName, _searchAircraft.c_str(), is_iequal()) ) {
|
|
|
|
_foundPath = p;
|
2013-11-16 14:22:34 +00:00
|
|
|
return VISIT_DONE;
|
2003-09-19 20:06:27 +00:00
|
|
|
}
|
|
|
|
|
2013-11-16 14:22:34 +00:00
|
|
|
return VISIT_CONTINUE;
|
2010-07-06 20:06:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string _searchAircraft;
|
|
|
|
SGPath _foundPath;
|
|
|
|
SGPropertyNode* _cache;
|
|
|
|
};
|
2003-09-19 20:06:27 +00:00
|
|
|
|
2011-10-16 18:55:04 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
static SGPath platformDefaultDataPath()
|
|
|
|
{
|
|
|
|
char *envp = ::getenv( "APPDATA" );
|
|
|
|
SGPath config( envp );
|
|
|
|
config.append( "flightgear.org" );
|
2011-10-16 18:58:22 +00:00
|
|
|
return config;
|
2011-10-16 18:55:04 +00:00
|
|
|
}
|
2013-05-04 09:58:22 +00:00
|
|
|
|
2013-11-16 10:05:09 +00:00
|
|
|
#elif defined(SG_MAC)
|
2013-06-10 19:42:53 +00:00
|
|
|
|
2013-11-16 10:05:09 +00:00
|
|
|
// platformDefaultDataPath defined in GUI/CocoaHelpers.h
|
2013-06-10 19:42:53 +00:00
|
|
|
|
2011-10-16 18:55:04 +00:00
|
|
|
#else
|
|
|
|
static SGPath platformDefaultDataPath()
|
|
|
|
{
|
2013-06-10 19:42:53 +00:00
|
|
|
return SGPath::home() / ".fgfs";
|
2011-10-16 18:55:04 +00:00
|
|
|
}
|
2013-06-10 19:42:53 +00:00
|
|
|
#endif
|
2011-10-16 18:55:04 +00:00
|
|
|
|
2013-11-14 16:48:14 +00:00
|
|
|
bool fgInitHome()
|
2011-10-21 15:49:15 +00:00
|
|
|
{
|
2013-06-10 19:42:53 +00:00
|
|
|
SGPath dataPath = SGPath::fromEnv("FG_HOME", platformDefaultDataPath());
|
|
|
|
globals->set_fg_home(dataPath.c_str());
|
2013-06-16 16:44:08 +00:00
|
|
|
|
|
|
|
simgear::Dir fgHome(dataPath);
|
|
|
|
if (!fgHome.exists()) {
|
|
|
|
fgHome.create(0755);
|
|
|
|
}
|
2013-11-14 16:48:14 +00:00
|
|
|
|
|
|
|
if (!fgHome.exists()) {
|
|
|
|
flightgear::fatalMessageBox("Problem setting up user data",
|
|
|
|
"Unable to create the user-data storage folder at: '"
|
|
|
|
+ dataPath.str() + "'");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fgGetBool("/sim/fghome-readonly", false)) {
|
|
|
|
// user / config forced us into readonly mode, fine
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Running with FG_HOME readonly");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// write our PID, and check writeability
|
|
|
|
SGPath pidPath(dataPath, "fgfs.pid");
|
|
|
|
if (pidPath.exists()) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "flightgear instance already running, switching to FG_HOME read-only.");
|
|
|
|
// set a marker property so terrasync/navcache don't try to write
|
|
|
|
// from secondary instances
|
|
|
|
fgSetBool("/sim/fghome-readonly", true);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
char buf[16];
|
|
|
|
bool result = false;
|
|
|
|
#if defined(SG_WINDOWS)
|
|
|
|
size_t len = snprintf(buf, 16, "%d", _getpid());
|
|
|
|
|
|
|
|
HANDLE f = CreateFileA(pidPath.c_str(), GENERIC_READ | GENERIC_WRITE,
|
|
|
|
FILE_SHARE_READ, /* sharing */
|
|
|
|
NULL, /* security attributes */
|
|
|
|
CREATE_NEW, /* error if already exists */
|
|
|
|
FILE_FLAG_DELETE_ON_CLOSE,
|
|
|
|
NULL /* template */);
|
|
|
|
|
|
|
|
result = (f != INVALID_HANDLE_VALUE);
|
|
|
|
if (result) {
|
|
|
|
DWORD written;
|
|
|
|
WriteFile(f, buf, len, &written, NULL /* overlapped */);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// POSIX, do open+unlink trick to the file is deleted on exit, even if we
|
|
|
|
// crash or exit(-1)
|
2013-12-09 22:15:35 +00:00
|
|
|
ssize_t len = snprintf(buf, 16, "%d", getpid());
|
2013-11-14 16:48:14 +00:00
|
|
|
int fd = ::open(pidPath.c_str(), O_WRONLY | O_CREAT | O_TRUNC | O_EXCL, 0644);
|
|
|
|
if (fd >= 0) {
|
2013-12-09 22:15:35 +00:00
|
|
|
result = ::write(fd, buf, len) == len;
|
|
|
|
if( ::unlink(pidPath.c_str()) != 0 ) // delete file when app quits
|
|
|
|
result = false;
|
2013-11-14 16:48:14 +00:00
|
|
|
}
|
|
|
|
#endif
|
2013-12-09 22:16:25 +00:00
|
|
|
|
|
|
|
fgSetBool("/sim/fghome-readonly", false);
|
|
|
|
|
2013-11-14 16:48:14 +00:00
|
|
|
if (!result) {
|
|
|
|
flightgear::fatalMessageBox("File permissions problem",
|
|
|
|
"Can't write to user-data storage folder, check file permissions and FG_HOME.",
|
|
|
|
"User-data at:" + dataPath.str());
|
|
|
|
}
|
|
|
|
return result;
|
2013-02-08 11:43:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read in configuration (file and command line)
|
2013-11-16 12:10:32 +00:00
|
|
|
int fgInitConfig ( int argc, char **argv, bool reinit )
|
2013-02-08 11:43:51 +00:00
|
|
|
{
|
|
|
|
SGPath dataPath = globals->get_fg_home();
|
2013-06-16 16:44:08 +00:00
|
|
|
|
2011-10-16 18:55:04 +00:00
|
|
|
simgear::Dir exportDir(simgear::Dir(dataPath).file("Export"));
|
|
|
|
if (!exportDir.exists()) {
|
2013-12-19 14:53:32 +00:00
|
|
|
exportDir.create(0755);
|
2011-10-16 18:55:04 +00:00
|
|
|
}
|
2011-10-21 15:49:15 +00:00
|
|
|
|
2015-03-13 18:11:29 +00:00
|
|
|
// Set /sim/fg-home. Use FG_HOME if necessary.
|
|
|
|
// deliberately not a tied property, for fgValidatePath security
|
|
|
|
// write-protect to avoid accidents
|
2011-10-16 18:55:04 +00:00
|
|
|
SGPropertyNode *home = fgGetNode("/sim", true);
|
2014-03-05 23:41:41 +00:00
|
|
|
home->removeChild("fg-home", 0);
|
2011-10-16 18:55:04 +00:00
|
|
|
home = home->getChild("fg-home", 0, true);
|
|
|
|
home->setStringValue(dataPath.c_str());
|
|
|
|
home->setAttribute(SGPropertyNode::WRITE, false);
|
2011-10-21 15:49:15 +00:00
|
|
|
|
2013-11-16 12:00:00 +00:00
|
|
|
fgSetDefaults();
|
2012-09-29 18:02:03 +00:00
|
|
|
flightgear::Options* options = flightgear::Options::sharedInstance();
|
2013-11-16 12:10:32 +00:00
|
|
|
if (!reinit) {
|
|
|
|
options->init(argc, argv, dataPath);
|
|
|
|
}
|
|
|
|
|
2013-11-23 19:58:45 +00:00
|
|
|
bool loadDefaults = options->shouldLoadDefaultConfig();
|
2012-09-29 18:02:03 +00:00
|
|
|
if (loadDefaults) {
|
|
|
|
// Read global preferences from $FG_ROOT/preferences.xml
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "Reading global preferences");
|
|
|
|
fgLoadProps("preferences.xml", globals->get_props());
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "Finished Reading global preferences");
|
2013-10-26 17:29:23 +00:00
|
|
|
|
|
|
|
// do not load user settings when reset to default is requested, or if
|
|
|
|
// told to explicitly ignore
|
|
|
|
if (options->isOptionSet("restore-defaults") || options->isOptionSet("ignore-autosave"))
|
2012-09-29 18:02:03 +00:00
|
|
|
{
|
2014-05-20 21:12:34 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Ignoring user settings. Restoring defaults.");
|
2012-09-29 18:02:03 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
globals->loadUserSettings(dataPath);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "not reading default configuration files");
|
|
|
|
}// of no-default-config selected
|
2013-11-23 19:58:45 +00:00
|
|
|
|
|
|
|
return flightgear::FG_OPTIONS_OK;
|
|
|
|
}
|
2011-12-29 23:39:08 +00:00
|
|
|
|
2014-06-11 17:19:41 +00:00
|
|
|
static void initAircraftDirsNasalSecurity()
|
|
|
|
{
|
2015-03-13 18:11:29 +00:00
|
|
|
// deliberately not a tied property, for fgValidatePath security
|
|
|
|
// write-protect to avoid accidents
|
2014-06-11 17:19:41 +00:00
|
|
|
SGPropertyNode* sim = fgGetNode("/sim", true);
|
|
|
|
sim->removeChildren("fg-aircraft");
|
2014-06-12 14:38:21 +00:00
|
|
|
|
2014-06-11 17:19:41 +00:00
|
|
|
int index = 0;
|
2014-06-12 14:38:21 +00:00
|
|
|
string_list const aircraft_paths = globals->get_aircraft_paths();
|
|
|
|
for( string_list::const_iterator it = aircraft_paths.begin();
|
|
|
|
it != aircraft_paths.end();
|
|
|
|
++it, ++index )
|
2014-06-11 17:19:41 +00:00
|
|
|
{
|
|
|
|
SGPropertyNode* n = sim->getChild("fg-aircraft", index, true);
|
|
|
|
n->setStringValue(*it);
|
|
|
|
n->setAttribute(SGPropertyNode::WRITE, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-26 12:20:51 +00:00
|
|
|
void fgInitAircraftPaths(bool reinit)
|
2013-11-23 19:58:45 +00:00
|
|
|
{
|
2015-03-10 08:30:55 +00:00
|
|
|
if (!globals->packageRoot()) {
|
|
|
|
fgInitPackageRoot();
|
2014-12-26 12:20:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SGSharedPtr<Root> pkgRoot(globals->packageRoot());
|
|
|
|
SGPropertyNode* aircraftProp = fgGetNode("/sim/aircraft", true);
|
|
|
|
aircraftProp->setAttribute(SGPropertyNode::PRESERVE, true);
|
|
|
|
|
|
|
|
if (!reinit) {
|
|
|
|
flightgear::Options::sharedInstance()->initPaths();
|
|
|
|
}
|
|
|
|
}
|
2014-06-11 17:19:41 +00:00
|
|
|
|
2014-12-26 12:20:51 +00:00
|
|
|
int fgInitAircraft(bool reinit)
|
|
|
|
{
|
2014-06-11 17:19:41 +00:00
|
|
|
if (!reinit) {
|
2013-11-23 19:58:45 +00:00
|
|
|
flightgear::Options::sharedInstance()->initAircraft();
|
|
|
|
}
|
2014-12-26 12:20:51 +00:00
|
|
|
|
|
|
|
SGSharedPtr<Root> pkgRoot(globals->packageRoot());
|
|
|
|
SGPropertyNode* aircraftProp = fgGetNode("/sim/aircraft", true);
|
2014-06-11 17:19:41 +00:00
|
|
|
|
2014-06-15 18:10:24 +00:00
|
|
|
string aircraftId(aircraftProp->getStringValue());
|
|
|
|
PackageRef acftPackage = pkgRoot->getPackageById(aircraftId);
|
2014-06-11 17:19:41 +00:00
|
|
|
if (acftPackage) {
|
2014-06-12 12:06:01 +00:00
|
|
|
if (acftPackage->isInstalled()) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Loading aircraft from package:" << acftPackage->qualifiedId());
|
|
|
|
|
|
|
|
// set catalog path so intra-package dependencies within the catalog
|
|
|
|
// are resolved correctly.
|
|
|
|
globals->set_catalog_aircraft_path(acftPackage->catalog()->installRoot());
|
|
|
|
|
|
|
|
// set aircraft-dir to short circuit the search process
|
|
|
|
InstallRef acftInstall = acftPackage->install();
|
|
|
|
fgSetString("/sim/aircraft-dir", acftInstall->path().c_str());
|
|
|
|
|
|
|
|
// overwrite the fully qualified ID with the aircraft one, so the
|
|
|
|
// code in FindAndCacheAircraft works as normal
|
2014-06-15 18:10:24 +00:00
|
|
|
// note since we may be using a variant, we can't use the package ID
|
|
|
|
size_t lastDot = aircraftId.rfind('.');
|
2015-11-19 21:37:03 +00:00
|
|
|
if (lastDot != std::string::npos) {
|
2015-10-16 01:05:35 +00:00
|
|
|
aircraftId = aircraftId.substr(lastDot + 1);
|
|
|
|
}
|
2015-08-03 20:53:56 +00:00
|
|
|
aircraftProp->setStringValue(aircraftId);
|
2015-10-16 01:05:35 +00:00
|
|
|
|
2014-06-12 12:06:01 +00:00
|
|
|
// run the traditional-code path below
|
|
|
|
} else {
|
|
|
|
#if 0
|
2014-06-11 17:19:41 +00:00
|
|
|
// naturally the better option would be to on-demand install it!
|
|
|
|
flightgear::fatalMessageBox("Aircraft not installed",
|
|
|
|
"Requested aircraft is not currently installed.",
|
2014-06-15 18:10:24 +00:00
|
|
|
aircraftId);
|
2014-06-11 17:19:41 +00:00
|
|
|
|
|
|
|
return flightgear::FG_OPTIONS_ERROR;
|
2014-06-12 12:06:01 +00:00
|
|
|
#endif
|
2015-10-16 01:05:35 +00:00
|
|
|
// fall back the default aircraft instead
|
2014-06-11 17:19:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
initAircraftDirsNasalSecurity();
|
|
|
|
|
2011-12-30 00:01:15 +00:00
|
|
|
FindAndCacheAircraft f(globals->get_props());
|
2010-07-06 20:06:50 +00:00
|
|
|
if (!f.loadAircraft()) {
|
2013-11-23 19:58:45 +00:00
|
|
|
return flightgear::FG_OPTIONS_ERROR;
|
2002-10-10 15:02:50 +00:00
|
|
|
}
|
2013-11-23 19:58:45 +00:00
|
|
|
|
|
|
|
return flightgear::FG_OPTIONS_OK;
|
2002-10-10 15:02:50 +00:00
|
|
|
}
|
|
|
|
|
2003-02-21 02:45:47 +00:00
|
|
|
/**
|
2003-08-28 20:53:08 +00:00
|
|
|
* Initialize vor/ndb/ils/fix list management and query systems (as
|
|
|
|
* well as simple airport db list)
|
2012-09-25 16:24:12 +00:00
|
|
|
* This is called multiple times in the case of a cache rebuild,
|
2013-10-20 19:46:40 +00:00
|
|
|
* to allow lengthy caching to take place in the background, without
|
2012-09-25 16:24:12 +00:00
|
|
|
* blocking the main/UI thread.
|
2003-02-21 02:45:47 +00:00
|
|
|
*/
|
|
|
|
bool
|
|
|
|
fgInitNav ()
|
|
|
|
{
|
2015-03-08 00:40:22 +00:00
|
|
|
flightgear::NavDataCache* cache = flightgear::NavDataCache::instance();
|
|
|
|
static bool doingRebuild = false;
|
|
|
|
|
|
|
|
if (!cache) {
|
|
|
|
cache = flightgear::NavDataCache::createInstance();
|
|
|
|
doingRebuild = cache->isRebuildRequired();
|
|
|
|
}
|
|
|
|
|
2015-03-19 17:01:38 +00:00
|
|
|
static const char* splashIdentsByRebuildPhase[] = {
|
|
|
|
"loading-nav-dat",
|
|
|
|
"navdata-airports",
|
|
|
|
"navdata-navaids",
|
|
|
|
"navdata-fixes",
|
|
|
|
"navdata-pois"
|
|
|
|
};
|
|
|
|
|
2015-03-08 00:40:22 +00:00
|
|
|
if (doingRebuild) {
|
2015-03-19 17:01:38 +00:00
|
|
|
flightgear::NavDataCache::RebuildPhase phase;
|
|
|
|
phase = cache->rebuild();
|
|
|
|
if (phase != flightgear::NavDataCache::REBUILD_DONE) {
|
|
|
|
// update the splash text based on percentage, phase
|
|
|
|
|
|
|
|
fgSplashProgress(splashIdentsByRebuildPhase[phase],
|
|
|
|
cache->rebuildPhaseCompletionPercentage());
|
|
|
|
|
2015-03-08 00:40:22 +00:00
|
|
|
// sleep to give the rebuild thread more time
|
|
|
|
SGTimeStamp::sleepForMSec(50);
|
|
|
|
return false;
|
|
|
|
}
|
2012-09-25 16:24:12 +00:00
|
|
|
}
|
2014-12-26 12:20:51 +00:00
|
|
|
|
2012-09-18 19:50:28 +00:00
|
|
|
FGTACANList *channellist = new FGTACANList;
|
|
|
|
globals->set_channellist( channellist );
|
|
|
|
|
2012-08-27 23:26:36 +00:00
|
|
|
SGPath path(globals->get_fg_root());
|
|
|
|
path.append( "Navaids/TACAN_freq.dat" );
|
|
|
|
flightgear::loadTacan(path, channellist);
|
2012-09-18 19:50:28 +00:00
|
|
|
|
|
|
|
return true;
|
2002-11-16 21:34:51 +00:00
|
|
|
}
|
|
|
|
|
1998-04-25 15:11:10 +00:00
|
|
|
// General house keeping initializations
|
2002-11-16 21:34:51 +00:00
|
|
|
bool fgInitGeneral() {
|
1998-08-27 17:01:55 +00:00
|
|
|
string root;
|
1999-11-19 02:10:24 +00:00
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "General Initialization" );
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "======= ==============" );
|
1998-04-25 20:24:00 +00:00
|
|
|
|
2001-01-13 22:06:39 +00:00
|
|
|
root = globals->get_fg_root();
|
1998-09-08 21:40:08 +00:00
|
|
|
if ( ! root.length() ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
// No root path set? Then bail ...
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT,
|
|
|
|
"Cannot continue without a path to the base package "
|
|
|
|
<< "being defined." );
|
2013-10-15 21:16:50 +00:00
|
|
|
return false;
|
1998-04-25 15:11:10 +00:00
|
|
|
}
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "FG_ROOT = " << '"' << root << '"' << endl );
|
1998-04-25 15:11:10 +00:00
|
|
|
|
2012-08-18 12:11:31 +00:00
|
|
|
// Note: browser command is hard-coded for Mac/Windows, so this only affects other platforms
|
|
|
|
globals->set_browser(fgGetString("/sim/startup/browser-app", WEB_BROWSER));
|
|
|
|
fgSetString("/sim/startup/browser-app", globals->get_browser());
|
2008-07-09 19:35:53 +00:00
|
|
|
|
2012-05-31 07:57:22 +00:00
|
|
|
simgear::Dir cwd(simgear::Dir::current());
|
2008-06-22 11:29:32 +00:00
|
|
|
SGPropertyNode *curr = fgGetNode("/sim", true);
|
2014-03-05 23:41:41 +00:00
|
|
|
curr->removeChild("fg-current", 0);
|
2008-06-22 11:29:32 +00:00
|
|
|
curr = curr->getChild("fg-current", 0, true);
|
2012-05-31 07:57:22 +00:00
|
|
|
curr->setStringValue(cwd.path().str());
|
2008-06-22 11:29:32 +00:00
|
|
|
curr->setAttribute(SGPropertyNode::WRITE, false);
|
2008-10-01 15:03:53 +00:00
|
|
|
|
2010-08-18 16:50:07 +00:00
|
|
|
fgSetBool("/sim/startup/stdout-to-terminal", isatty(1) != 0 );
|
|
|
|
fgSetBool("/sim/startup/stderr-to-terminal", isatty(2) != 0 );
|
1999-06-18 03:42:54 +00:00
|
|
|
return true;
|
1998-04-25 15:11:10 +00:00
|
|
|
}
|
|
|
|
|
2013-02-08 13:15:20 +00:00
|
|
|
// Write various configuraton values out to the logs
|
|
|
|
void fgOutputSettings()
|
|
|
|
{
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Configuration State" );
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "======= ==============" );
|
|
|
|
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "aircraft-dir = " << '"' << fgGetString("/sim/aircraft-dir") << '"' );
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "fghome-dir = " << '"' << globals->get_fg_home() << '"');
|
2015-03-17 22:10:49 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "download-dir = " << '"' << fgGetString("/sim/paths/download-dir") << '"' );
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "terrasync-dir = " << '"' << fgGetString("/sim/terrasync/scenery-dir") << '"' );
|
|
|
|
|
2013-02-08 13:15:20 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "aircraft-search-paths = \n\t" << simgear::strutils::join(globals->get_aircraft_paths(), "\n\t") );
|
2013-02-11 12:51:54 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "scenery-search-paths = \n\t" << simgear::strutils::join(globals->get_fg_scenery(), "\n\t") );
|
2013-02-08 13:15:20 +00:00
|
|
|
}
|
|
|
|
|
1998-04-25 15:11:10 +00:00
|
|
|
// This is the top level init routine which calls all the other
|
|
|
|
// initialization routines. If you are adding a subsystem to flight
|
|
|
|
// gear, its initialization call should located in this routine.
|
|
|
|
// Returns non-zero if a problem encountered.
|
2013-11-16 12:10:32 +00:00
|
|
|
void fgCreateSubsystems(bool duringReset) {
|
2009-10-24 09:22:20 +00:00
|
|
|
|
2012-09-18 19:29:36 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Creating Subsystems");
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "========== ==========");
|
1998-04-25 15:11:10 +00:00
|
|
|
|
2011-11-20 15:20:05 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the sound subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Sound manager uses an own subsystem group "SOUND" which is the last
|
|
|
|
// to be updated in every loop.
|
|
|
|
// Sound manager is updated last so it can use the CPU while the GPU
|
|
|
|
// is processing the scenery (doubled the frame-rate for me) -EMH-
|
2015-12-10 22:48:02 +00:00
|
|
|
globals->add_new_subsystem<FGSoundManager>(SGSubsystemMgr::SOUND);
|
2011-11-20 15:20:05 +00:00
|
|
|
|
2003-09-24 17:20:55 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the event manager subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2004-11-18 05:00:44 +00:00
|
|
|
globals->get_event_mgr()->init();
|
2005-11-09 20:34:46 +00:00
|
|
|
globals->get_event_mgr()->setRealtimeProperty(fgGetNode("/sim/time/delta-realtime-sec", true));
|
2001-04-05 20:20:44 +00:00
|
|
|
|
2003-12-05 01:52:34 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
2007-05-01 18:06:48 +00:00
|
|
|
// Initialize the property interpolator subsystem. Put into the INIT
|
|
|
|
// group because the "nasal" subsystem may need it at GENERAL take-down.
|
2003-12-05 01:52:34 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
2014-07-11 23:17:35 +00:00
|
|
|
globals->add_subsystem("prop-interpolator", new FGInterpolator, SGSubsystemMgr::INIT);
|
|
|
|
|
2004-02-20 17:35:33 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Add the FlightGear property utilities.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem("properties", new FGProperties);
|
2003-12-05 01:52:34 +00:00
|
|
|
|
2011-11-19 14:37:49 +00:00
|
|
|
|
2015-12-05 00:25:29 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Add the FlightGear property utilities.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
2015-12-10 22:48:02 +00:00
|
|
|
globals->add_new_subsystem<flightgear::AirportDynamicsManager>();
|
2015-12-05 00:25:29 +00:00
|
|
|
|
2011-11-19 14:37:49 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Add the performance monitoring system.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem("performance-mon",
|
|
|
|
new SGPerformanceMonitor(globals->get_subsystem_mgr(),
|
2011-11-19 21:37:13 +00:00
|
|
|
fgGetNode("/sim/performance-monitor", true)));
|
2011-11-19 14:37:49 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the material property subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2013-11-21 17:38:22 +00:00
|
|
|
SGPath mpath( globals->get_fg_root() );
|
|
|
|
mpath.append( fgGetString("/sim/rendering/materials-file") );
|
|
|
|
if ( ! globals->get_matlib()->load(globals->get_fg_root(), mpath.str(),
|
|
|
|
globals->get_props()) ) {
|
|
|
|
throw sg_io_exception("Error loading materials file", mpath);
|
2000-06-23 00:30:04 +00:00
|
|
|
}
|
2015-03-10 08:30:55 +00:00
|
|
|
|
|
|
|
// may exist already due to GUI startup
|
|
|
|
if (!globals->get_subsystem("http")) {
|
|
|
|
globals->add_subsystem( "http", new FGHTTPClient );
|
|
|
|
}
|
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the scenery management subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2008-02-15 06:45:38 +00:00
|
|
|
globals->get_scenery()->get_scene_graph()
|
|
|
|
->addChild(simgear::Particles::getCommonRoot());
|
2008-10-17 21:25:03 +00:00
|
|
|
simgear::GlobalParticleCallback::setSwitch(fgGetNode("/sim/rendering/particles", true));
|
2008-02-15 06:45:19 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the flight model subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2010-06-16 07:31:59 +00:00
|
|
|
globals->add_subsystem("flight", new FDMShell, SGSubsystemMgr::FDM);
|
2007-11-03 21:02:28 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the weather subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// Initialize the weather modeling subsystem
|
|
|
|
globals->add_subsystem("environment", new FGEnvironmentMgr);
|
2011-11-05 13:18:36 +00:00
|
|
|
globals->add_subsystem("ephemeris", new Ephemeris);
|
|
|
|
|
2006-06-24 03:42:30 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the aircraft systems and instrumentation (before the
|
|
|
|
// autopilot.)
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2010-06-16 07:31:59 +00:00
|
|
|
globals->add_subsystem("systems", new FGSystemMgr, SGSubsystemMgr::FDM);
|
2012-09-19 18:53:54 +00:00
|
|
|
globals->add_subsystem("instrumentation", new FGInstrumentMgr, SGSubsystemMgr::FDM);
|
2012-09-26 16:02:11 +00:00
|
|
|
globals->add_subsystem("hud", new HUD, SGSubsystemMgr::DISPLAY);
|
2012-09-26 19:30:24 +00:00
|
|
|
globals->add_subsystem("cockpit-displays", new flightgear::CockpitDisplayManager, SGSubsystemMgr::DISPLAY);
|
2012-09-26 16:02:11 +00:00
|
|
|
|
2004-01-31 19:47:45 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the XML Autopilot subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2012-04-05 18:40:07 +00:00
|
|
|
globals->add_subsystem( "xml-autopilot", FGXMLAutopilotGroup::createInstance("autopilot"), SGSubsystemMgr::FDM );
|
|
|
|
globals->add_subsystem( "xml-proprules", FGXMLAutopilotGroup::createInstance("property-rule"), SGSubsystemMgr::GENERAL );
|
2015-12-10 22:30:26 +00:00
|
|
|
globals->add_new_subsystem<FGRouteMgr>();
|
2002-04-13 13:11:34 +00:00
|
|
|
|
2009-09-07 07:27:38 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the Input-Output subsystem
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem( "io", new FGIO );
|
2012-11-19 11:30:12 +00:00
|
|
|
|
2002-03-12 16:29:00 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
2002-11-07 16:27:47 +00:00
|
|
|
// Create and register the logger.
|
2002-03-12 16:29:00 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2003-01-16 16:01:26 +00:00
|
|
|
globals->add_subsystem("logger", new FGLogger);
|
2002-03-12 16:29:00 +00:00
|
|
|
|
2002-11-07 16:27:47 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Create and register the XML GUI.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2003-09-24 17:20:55 +00:00
|
|
|
globals->add_subsystem("gui", new NewGUI, SGSubsystemMgr::INIT);
|
2002-11-07 16:27:47 +00:00
|
|
|
|
2004-05-20 13:27:40 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the 2D cloud subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
2004-10-20 08:18:29 +00:00
|
|
|
fgGetBool("/sim/rendering/bump-mapping", false);
|
2003-09-24 17:20:55 +00:00
|
|
|
|
2012-04-21 13:31:20 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the canvas 2d drawing subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
2013-12-09 22:13:45 +00:00
|
|
|
simgear::canvas::Canvas::setSystemAdapter(
|
|
|
|
simgear::canvas::SystemAdapterPtr(new canvas::FGCanvasSystemAdapter)
|
|
|
|
);
|
2012-07-04 11:15:12 +00:00
|
|
|
globals->add_subsystem("Canvas", new CanvasMgr, SGSubsystemMgr::DISPLAY);
|
2012-07-27 11:17:42 +00:00
|
|
|
globals->add_subsystem("CanvasGUI", new GUIMgr, SGSubsystemMgr::DISPLAY);
|
2012-04-21 13:31:20 +00:00
|
|
|
|
2010-08-01 14:16:33 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
2011-04-03 15:58:16 +00:00
|
|
|
// Initialize the ATC subsystem
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem("ATC", new FGATCManager, SGSubsystemMgr::POST_FDM);
|
2010-10-02 22:40:38 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize multiplayer subsystem
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
globals->add_subsystem("mp", new FGMultiplayMgr, SGSubsystemMgr::POST_FDM);
|
|
|
|
|
2003-11-28 15:48:05 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialise the AI Model Manager
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, " AI Model Manager");
|
2011-11-19 14:37:49 +00:00
|
|
|
globals->add_subsystem("ai-model", new FGAIManager, SGSubsystemMgr::POST_FDM);
|
|
|
|
globals->add_subsystem("submodel-mgr", new FGSubmodelMgr, SGSubsystemMgr::POST_FDM);
|
2003-11-28 15:48:05 +00:00
|
|
|
|
|
|
|
|
2007-04-04 19:05:59 +00:00
|
|
|
// It's probably a good idea to initialize the top level traffic manager
|
|
|
|
// After the AI and ATC systems have been initialized properly.
|
|
|
|
// AI Traffic manager
|
2011-11-19 14:37:49 +00:00
|
|
|
globals->add_subsystem("traffic-manager", new FGTrafficManager, SGSubsystemMgr::POST_FDM);
|
2004-06-03 17:59:14 +00:00
|
|
|
|
2003-03-30 12:46:08 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Add a new 2D panel.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2012-05-16 10:43:09 +00:00
|
|
|
fgSetArchivable("/sim/panel/visibility");
|
|
|
|
fgSetArchivable("/sim/panel/x-offset");
|
|
|
|
fgSetArchivable("/sim/panel/y-offset");
|
|
|
|
fgSetArchivable("/sim/panel/jitter");
|
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the controls subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
2012-09-24 23:48:48 +00:00
|
|
|
|
2015-12-10 21:05:54 +00:00
|
|
|
globals->add_new_subsystem<FGControls>(SGSubsystemMgr::GENERAL);
|
2001-04-05 20:20:44 +00:00
|
|
|
|
I'm attaching diffs to add a new FGInput module to FlightGear
(src/Input). So far, FGInput replaces most of src/Main/keyboard.cxx
(I've left a tiny stub); in the very near future, it will also take
over control of the joystick, mouse (Norm permitting), and panel
instrument interactions, so that there is a single mechanism for
configuring all input devices.
The new format should be (close to) self-explanatory by looking at the
new base-package file keyboard.xml, which is now included by
preferences.xml (I'll do the same thing for the joystick when I have a
chance). I have not managed to move all keybindings into this file
yet, but I've made a good start. I'm including Tony in the recipient
list so that he can see how bindings can use an external XML file.
This patch also adds support for multiple bindings for a single key,
special keys (i.e. keypad and function keys), and key modifiers
(shift/alt/ctrl); special keys use the PUI convention of adding 256 to
the Glut key code.
Unfortunately, everything comes with a price; in this case, I have not
yet found a general mechanism for the old (hard-coded) modal bindings,
which behaved differently depending on the autopilot state (i.e. left
rudder or move AP heading left); with my patches, this functionality
disappears, but you can still adjust the autopilot using the panel or
the GUI input dialogs.
2001-05-23 23:01:15 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the input subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2012-09-27 14:33:12 +00:00
|
|
|
globals->add_subsystem("input", new FGInput, SGSubsystemMgr::GENERAL);
|
I'm attaching diffs to add a new FGInput module to FlightGear
(src/Input). So far, FGInput replaces most of src/Main/keyboard.cxx
(I've left a tiny stub); in the very near future, it will also take
over control of the joystick, mouse (Norm permitting), and panel
instrument interactions, so that there is a single mechanism for
configuring all input devices.
The new format should be (close to) self-explanatory by looking at the
new base-package file keyboard.xml, which is now included by
preferences.xml (I'll do the same thing for the joystick when I have a
chance). I have not managed to move all keybindings into this file
yet, but I've made a good start. I'm including Tony in the recipient
list so that he can see how bindings can use an external XML file.
This patch also adds support for multiple bindings for a single key,
special keys (i.e. keypad and function keys), and key modifiers
(shift/alt/ctrl); special keys use the PUI convention of adding 256 to
the Glut key code.
Unfortunately, everything comes with a price; in this case, I have not
yet found a general mechanism for the old (hard-coded) modal bindings,
which behaved differently depending on the autopilot state (i.e. left
rudder or move AP heading left); with my patches, this functionality
disappears, but you can still adjust the autopilot using the panel or
the GUI input dialogs.
2001-05-23 23:01:15 +00:00
|
|
|
|
|
|
|
|
2003-07-17 18:24:17 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the replay subsystem
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem("replay", new FGReplay);
|
2012-12-10 18:11:44 +00:00
|
|
|
globals->add_subsystem("history", new FGFlightHistory);
|
|
|
|
|
2010-10-22 11:22:00 +00:00
|
|
|
#ifdef ENABLE_AUDIO_SUPPORT
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the sound-effects subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem("voice", new FGVoiceMgr, SGSubsystemMgr::DISPLAY);
|
|
|
|
#endif
|
2009-10-04 13:52:53 +00:00
|
|
|
|
2013-05-19 14:29:49 +00:00
|
|
|
#ifdef ENABLE_IAX
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the FGCom subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem("fgcom", new FGCom);
|
|
|
|
#endif
|
|
|
|
|
2014-03-03 20:59:16 +00:00
|
|
|
{
|
|
|
|
SGSubsystem * httpd = flightgear::http::FGHttpd::createInstance( fgGetNode(flightgear::http::PROPERTY_ROOT) );
|
|
|
|
if( NULL != httpd )
|
|
|
|
globals->add_subsystem("httpd", httpd );
|
|
|
|
}
|
|
|
|
|
2002-10-03 21:20:56 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
2010-10-22 11:22:00 +00:00
|
|
|
// Initialize the lighting subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// ordering here is important : Nasal (via events), then models, then views
|
2013-11-16 12:10:32 +00:00
|
|
|
if (!duringReset) {
|
|
|
|
globals->add_subsystem("lighting", new FGLight, SGSubsystemMgr::DISPLAY);
|
|
|
|
globals->add_subsystem("events", globals->get_event_mgr(), SGSubsystemMgr::DISPLAY);
|
|
|
|
}
|
2010-10-22 11:22:00 +00:00
|
|
|
|
2015-12-19 04:13:23 +00:00
|
|
|
globals->add_new_subsystem<FGAircraftModel>(SGSubsystemMgr::DISPLAY);
|
2015-12-19 03:56:56 +00:00
|
|
|
globals->add_new_subsystem<FGModelMgr>(SGSubsystemMgr::DISPLAY);
|
2010-10-22 11:22:00 +00:00
|
|
|
|
2015-12-10 22:48:02 +00:00
|
|
|
globals->add_new_subsystem<FGViewMgr>(SGSubsystemMgr::DISPLAY);
|
2012-09-18 19:29:36 +00:00
|
|
|
}
|
2002-10-03 21:20:56 +00:00
|
|
|
|
2012-09-18 19:29:36 +00:00
|
|
|
void fgPostInitSubsystems()
|
|
|
|
{
|
|
|
|
SGTimeStamp st;
|
|
|
|
st.stamp();
|
|
|
|
|
2003-11-25 21:08:36 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the Nasal interpreter.
|
|
|
|
// Do this last, so that the loaded scripts see initialized state
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
FGNasalSys* nasal = new FGNasalSys();
|
2006-06-10 22:24:05 +00:00
|
|
|
globals->add_subsystem("nasal", nasal, SGSubsystemMgr::INIT);
|
2003-11-25 21:08:36 +00:00
|
|
|
nasal->init();
|
2012-09-18 19:29:36 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Nasal init took:" << st.elapsedMSec());
|
2013-04-29 20:15:32 +00:00
|
|
|
|
2005-06-11 09:13:44 +00:00
|
|
|
// initialize methods that depend on other subsystems.
|
2012-09-18 19:29:36 +00:00
|
|
|
st.stamp();
|
2005-06-11 09:13:44 +00:00
|
|
|
globals->get_subsystem_mgr()->postinit();
|
2012-09-18 19:29:36 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Subsystems postinit took:" << st.elapsedMSec());
|
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// End of subsystem initialization.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2012-01-29 09:34:11 +00:00
|
|
|
fgSetBool("/sim/crashed", false);
|
2010-11-02 23:19:09 +00:00
|
|
|
fgSetBool("/sim/initialized", true);
|
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, endl);
|
1997-08-23 01:46:20 +00:00
|
|
|
}
|
1999-04-27 15:52:32 +00:00
|
|
|
|
2014-02-24 06:10:29 +00:00
|
|
|
// re-position is a simplified version of the traditional (legacy)
|
|
|
|
// reset procedure. We only need to poke systems which will be upset by
|
|
|
|
// a sudden change in aircraft position. Since this potentially includes
|
|
|
|
// Nasal, we trigger the 'reinit' signal.
|
|
|
|
void fgStartReposition()
|
|
|
|
{
|
|
|
|
SGPropertyNode *master_freeze = fgGetNode("/sim/freeze/master");
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "fgStartReposition()");
|
|
|
|
|
|
|
|
// ensure we are frozen
|
|
|
|
bool freeze = master_freeze->getBoolValue();
|
|
|
|
if ( !freeze ) {
|
|
|
|
master_freeze->setBoolValue(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// set this signal so Nasal scripts can take action.
|
|
|
|
fgSetBool("/sim/signals/reinit", true);
|
|
|
|
fgSetBool("/sim/crashed", false);
|
|
|
|
|
2014-03-10 12:00:51 +00:00
|
|
|
FDMShell* fdm = static_cast<FDMShell*>(globals->get_subsystem("flight"));
|
|
|
|
fdm->unbind();
|
2014-02-24 06:10:29 +00:00
|
|
|
|
|
|
|
// update our position based on current presets
|
|
|
|
// this will mark position as needed finalized which we'll do in the
|
|
|
|
// main-loop
|
|
|
|
flightgear::initPosition();
|
|
|
|
|
|
|
|
simgear::SGTerraSync* terraSync =
|
|
|
|
static_cast<simgear::SGTerraSync*>(globals->get_subsystem("terrasync"));
|
|
|
|
if (terraSync) {
|
|
|
|
terraSync->reposition();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize the FDM
|
|
|
|
globals->get_subsystem("flight")->reinit();
|
|
|
|
|
|
|
|
// reset replay buffers
|
|
|
|
globals->get_subsystem("replay")->reinit();
|
|
|
|
|
|
|
|
// ugly: finalizePosition waits for METAR to arrive for the new airport.
|
|
|
|
// we don't re-init the environment manager here, since historically we did
|
|
|
|
// not, and doing so seems to have other issues. All that's needed is to
|
|
|
|
// schedule METAR fetch immediately, so it's available for finalizePosition.
|
|
|
|
// So we manually extract the METAR-fetching component inside the environment
|
|
|
|
// manager, and re-init that.
|
|
|
|
SGSubsystemGroup* envMgr = static_cast<SGSubsystemGroup*>(globals->get_subsystem("environment"));
|
|
|
|
if (envMgr) {
|
|
|
|
envMgr->get_subsystem("realwx")->reinit();
|
|
|
|
}
|
|
|
|
|
|
|
|
// need to bind FDMshell again, since we manually unbound it above...
|
2014-03-10 12:02:10 +00:00
|
|
|
fdm->bind();
|
2014-02-24 06:10:29 +00:00
|
|
|
|
2014-03-10 12:02:10 +00:00
|
|
|
// need to reset aircraft (systems/instruments/autopilot)
|
|
|
|
// so they can adapt to current environment
|
2014-02-24 06:10:29 +00:00
|
|
|
globals->get_subsystem("systems")->reinit();
|
|
|
|
globals->get_subsystem("instrumentation")->reinit();
|
2014-03-10 12:02:10 +00:00
|
|
|
globals->get_subsystem("xml-autopilot")->reinit();
|
2014-03-05 15:16:07 +00:00
|
|
|
|
2014-02-24 06:10:29 +00:00
|
|
|
// setup state to end re-init
|
|
|
|
fgSetBool("/sim/signals/reinit", false);
|
|
|
|
if ( !freeze ) {
|
|
|
|
master_freeze->setBoolValue(false);
|
|
|
|
}
|
|
|
|
fgSetBool("/sim/sceneryloaded",false);
|
|
|
|
}
|
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
void fgStartNewReset()
|
|
|
|
{
|
2013-12-09 23:27:25 +00:00
|
|
|
SGPropertyNode_ptr preserved(new SGPropertyNode);
|
2014-02-24 06:10:29 +00:00
|
|
|
|
2013-12-15 20:45:08 +00:00
|
|
|
if (!copyPropertiesWithAttribute(globals->get_props(), preserved, SGPropertyNode::PRESERVE))
|
2013-12-09 23:27:25 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Error saving preserved state");
|
2013-11-16 12:10:32 +00:00
|
|
|
|
|
|
|
fgSetBool("/sim/signals/reinit", true);
|
|
|
|
fgSetBool("/sim/freeze/master", true);
|
|
|
|
|
|
|
|
SGSubsystemMgr* subsystemManger = globals->get_subsystem_mgr();
|
2014-01-13 16:41:31 +00:00
|
|
|
// Nasal is manually inited in fgPostInit, ensure it's already shutdown
|
|
|
|
// before other subsystems, so Nasal listeners don't fire during shutdonw
|
|
|
|
SGSubsystem* nasal = subsystemManger->get_subsystem("nasal");
|
|
|
|
nasal->shutdown();
|
|
|
|
nasal->unbind();
|
|
|
|
subsystemManger->remove("nasal");
|
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
subsystemManger->shutdown();
|
|
|
|
subsystemManger->unbind();
|
|
|
|
|
2013-12-15 20:45:08 +00:00
|
|
|
// remove most subsystems, with a few exceptions.
|
2013-11-16 12:10:32 +00:00
|
|
|
for (int g=0; g<SGSubsystemMgr::MAX_GROUPS; ++g) {
|
|
|
|
SGSubsystemGroup* grp = subsystemManger->get_group(static_cast<SGSubsystemMgr::GroupType>(g));
|
|
|
|
const string_list& names(grp->member_names());
|
|
|
|
string_list::const_iterator it;
|
|
|
|
for (it = names.begin(); it != names.end(); ++it) {
|
|
|
|
if ((*it == "time") || (*it == "terrasync") || (*it == "events")
|
2015-12-11 18:11:46 +00:00
|
|
|
|| (*it == "lighting") || (*it == FGTileMgr::subsystemName()) || (*it == FGScenery::subsystemName()))
|
2013-11-16 12:10:32 +00:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
subsystemManger->remove(it->c_str());
|
|
|
|
} catch (std::exception& e) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "caught std::exception shutting down:" << *it);
|
|
|
|
} catch (...) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "caught generic exception shutting down:" << *it);
|
|
|
|
}
|
|
|
|
|
|
|
|
// don't delete here, dropping the ref should be sufficient
|
|
|
|
}
|
|
|
|
} // of top-level groups iteration
|
|
|
|
|
2014-03-11 16:45:31 +00:00
|
|
|
FGRenderer* render = globals->get_renderer();
|
|
|
|
// needed or we crash in multi-threaded OSG mode
|
|
|
|
render->getViewer()->stopThreading();
|
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
// order is important here since tile-manager shutdown needs to
|
|
|
|
// access the scenery object
|
2015-12-10 22:48:02 +00:00
|
|
|
subsystemManger->remove(FGTileMgr::subsystemName());
|
|
|
|
subsystemManger->remove(FGScenery::subsystemName());
|
|
|
|
|
2014-03-13 18:18:23 +00:00
|
|
|
FGScenery::getPagerSingleton()->clearRequests();
|
2013-11-16 12:10:32 +00:00
|
|
|
flightgear::CameraGroup::setDefault(NULL);
|
|
|
|
|
|
|
|
// don't cancel the pager until after shutdown, since AIModels (and
|
|
|
|
// potentially others) can queue delete requests on the pager.
|
|
|
|
render->getViewer()->getDatabasePager()->cancel();
|
2013-12-09 23:27:25 +00:00
|
|
|
render->getViewer()->getDatabasePager()->clear();
|
2013-11-16 12:10:32 +00:00
|
|
|
|
2013-12-06 17:33:13 +00:00
|
|
|
osgDB::Registry::instance()->clearObjectCache();
|
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
// preserve the event handler; re-creating it would entail fixing the
|
|
|
|
// idle handler
|
|
|
|
osg::ref_ptr<flightgear::FGEventHandler> eventHandler = render->getEventHandler();
|
|
|
|
|
|
|
|
globals->set_renderer(NULL);
|
2013-11-21 17:38:22 +00:00
|
|
|
globals->set_matlib(NULL);
|
2013-11-25 23:53:58 +00:00
|
|
|
globals->set_chatter_queue(NULL);
|
2013-11-16 12:10:32 +00:00
|
|
|
|
2013-12-04 09:09:47 +00:00
|
|
|
simgear::clearEffectCache();
|
2013-11-25 23:53:58 +00:00
|
|
|
simgear::SGModelLib::resetPropertyRoot();
|
|
|
|
|
2013-12-15 20:45:08 +00:00
|
|
|
simgear::GlobalParticleCallback::setSwitch(NULL);
|
2013-11-16 12:10:32 +00:00
|
|
|
|
2013-12-15 20:45:08 +00:00
|
|
|
globals->resetPropertyRoot();
|
2014-03-26 19:23:34 +00:00
|
|
|
// otherwise channels are duplicated
|
|
|
|
globals->get_channel_options_list()->clear();
|
|
|
|
|
2013-11-22 22:42:04 +00:00
|
|
|
fgInitConfig(0, NULL, true);
|
|
|
|
fgInitGeneral(); // all of this?
|
|
|
|
|
2014-03-11 22:43:25 +00:00
|
|
|
flightgear::Options::sharedInstance()->processOptions();
|
|
|
|
|
|
|
|
// PRESERVED properties over-write state from options, intentionally
|
2013-12-15 20:45:08 +00:00
|
|
|
if ( copyProperties(preserved, globals->get_props()) ) {
|
2013-12-09 23:27:25 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Preserved state restored successfully" );
|
|
|
|
} else {
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
|
|
|
"Some errors restoring preserved state (read-only props?)" );
|
|
|
|
}
|
|
|
|
|
|
|
|
fgGetNode("/sim")->removeChild("aircraft-dir");
|
2014-12-26 12:20:51 +00:00
|
|
|
fgInitAircraftPaths(true);
|
2013-11-23 19:58:45 +00:00
|
|
|
fgInitAircraft(true);
|
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
render = new FGRenderer;
|
|
|
|
render->setEventHandler(eventHandler);
|
2014-03-11 16:45:47 +00:00
|
|
|
eventHandler->reset();
|
2013-11-16 12:10:32 +00:00
|
|
|
globals->set_renderer(render);
|
|
|
|
render->init();
|
|
|
|
render->setViewer(viewer.get());
|
2014-01-13 14:46:38 +00:00
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
viewer->getDatabasePager()->setUpThreads(1, 1);
|
|
|
|
|
2014-01-13 14:46:38 +00:00
|
|
|
// must do this before splashinit for Rembrandt
|
2013-11-16 12:10:32 +00:00
|
|
|
flightgear::CameraGroup::buildDefaultGroup(viewer.get());
|
2014-01-13 14:46:38 +00:00
|
|
|
render->splashinit();
|
2014-03-11 16:45:31 +00:00
|
|
|
viewer->startThreading();
|
2014-01-13 14:46:38 +00:00
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
fgOSResetProperties();
|
2013-11-23 19:58:45 +00:00
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
// init some things manually
|
|
|
|
// which do not follow the regular init pattern
|
|
|
|
|
|
|
|
globals->get_event_mgr()->init();
|
|
|
|
globals->get_event_mgr()->setRealtimeProperty(fgGetNode("/sim/time/delta-realtime-sec", true));
|
|
|
|
|
2013-11-21 17:38:22 +00:00
|
|
|
globals->set_matlib( new SGMaterialLib );
|
|
|
|
|
2013-11-16 12:10:32 +00:00
|
|
|
// terra-sync needs the property tree root, pass it back in
|
|
|
|
simgear::SGTerraSync* terra_sync = static_cast<simgear::SGTerraSync*>(subsystemManger->get_subsystem("terrasync"));
|
|
|
|
terra_sync->setRoot(globals->get_props());
|
|
|
|
|
|
|
|
fgSetBool("/sim/signals/reinit", false);
|
|
|
|
fgSetBool("/sim/freeze/master", false);
|
|
|
|
fgSetBool("/sim/sceneryloaded",false);
|
|
|
|
}
|
2008-07-09 12:22:33 +00:00
|
|
|
|
2015-03-10 08:30:55 +00:00
|
|
|
void fgInitPackageRoot()
|
|
|
|
{
|
|
|
|
if (globals->packageRoot()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-03-17 22:10:49 +00:00
|
|
|
SGPath packageAircraftDir = flightgear::defaultDownloadDir();
|
2015-03-10 08:30:55 +00:00
|
|
|
packageAircraftDir.append("Aircraft");
|
|
|
|
|
|
|
|
SGSharedPtr<Root> pkgRoot(new Root(packageAircraftDir, FLIGHTGEAR_VERSION));
|
|
|
|
// set the http client later (too early in startup right now)
|
|
|
|
globals->setPackageRoot(pkgRoot);
|
|
|
|
|
|
|
|
}
|