2004-05-28 05:24:54 +00:00
|
|
|
// navdb.cxx -- top level navaids management routines
|
|
|
|
//
|
|
|
|
// Written by Curtis Olson, started May 2004.
|
|
|
|
//
|
2004-11-19 22:10:41 +00:00
|
|
|
// Copyright (C) 2004 Curtis L. Olson - http://www.flightgear.org/~curt
|
2004-05-28 05:24:54 +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.
|
2004-05-28 05:24:54 +00:00
|
|
|
//
|
|
|
|
// $Id$
|
|
|
|
|
2006-02-18 13:58:09 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
2004-05-28 05:24:54 +00:00
|
|
|
|
2004-05-28 16:24:43 +00:00
|
|
|
#include <simgear/compiler.h>
|
|
|
|
|
2008-07-25 18:38:29 +00:00
|
|
|
#include <string>
|
2004-05-28 16:24:43 +00:00
|
|
|
|
2004-05-28 05:24:54 +00:00
|
|
|
#include <simgear/debug/logstream.hxx>
|
2008-08-03 14:34:42 +00:00
|
|
|
#include <simgear/math/sg_geodesy.hxx>
|
2008-08-14 18:13:39 +00:00
|
|
|
#include <simgear/misc/strutils.hxx>
|
2008-12-25 23:11:43 +00:00
|
|
|
#include <simgear/misc/sg_path.hxx>
|
2008-09-12 08:46:15 +00:00
|
|
|
#include <simgear/structure/exception.hxx>
|
|
|
|
#include <simgear/misc/sgstream.hxx>
|
2010-12-07 17:55:45 +00:00
|
|
|
#include <simgear/props/props_io.hxx>
|
2004-05-28 05:24:54 +00:00
|
|
|
|
|
|
|
#include "navrecord.hxx"
|
2008-12-25 23:11:43 +00:00
|
|
|
#include "navlist.hxx"
|
2004-05-28 05:24:54 +00:00
|
|
|
#include "navdb.hxx"
|
2009-05-14 20:55:09 +00:00
|
|
|
#include <Main/globals.hxx>
|
|
|
|
#include <Navaids/markerbeacon.hxx>
|
|
|
|
#include <Airports/simple.hxx>
|
2010-12-07 17:55:45 +00:00
|
|
|
#include <Airports/runways.hxx>
|
|
|
|
#include <Airports/xmlloader.hxx>
|
|
|
|
#include <Main/fg_props.hxx>
|
2004-05-28 05:24:54 +00:00
|
|
|
|
2008-07-29 08:27:48 +00:00
|
|
|
using std::string;
|
2004-05-28 16:24:43 +00:00
|
|
|
|
2010-12-07 17:55:45 +00:00
|
|
|
typedef std::map<FGAirport*, SGPropertyNode_ptr> AirportPropertyMap;
|
|
|
|
|
|
|
|
static AirportPropertyMap static_airportIlsData;
|
|
|
|
|
2009-01-03 15:54:03 +00:00
|
|
|
static FGPositioned::Type
|
|
|
|
mapRobinTypeToFGPType(int aTy)
|
|
|
|
{
|
|
|
|
switch (aTy) {
|
|
|
|
// case 1:
|
|
|
|
case 2: return FGPositioned::NDB;
|
|
|
|
case 3: return FGPositioned::VOR;
|
2009-06-11 22:54:51 +00:00
|
|
|
case 4: return FGPositioned::ILS;
|
|
|
|
case 5: return FGPositioned::LOC;
|
2009-01-03 15:54:03 +00:00
|
|
|
case 6: return FGPositioned::GS;
|
|
|
|
case 12:
|
|
|
|
case 13: return FGPositioned::DME;
|
|
|
|
case 99: return FGPositioned::INVALID; // end-of-file code
|
|
|
|
default:
|
|
|
|
throw sg_range_exception("Got a nav.dat type we don't recognize", "FGNavRecord::createFromStream");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static FGNavRecord* createNavFromStream(std::istream& aStream)
|
|
|
|
{
|
|
|
|
int rawType;
|
|
|
|
aStream >> rawType;
|
2009-01-04 09:41:52 +00:00
|
|
|
if (aStream.eof() || (rawType == 99)) {
|
2009-01-03 15:54:03 +00:00
|
|
|
return NULL; // happens with, eg, carrier_nav.dat
|
|
|
|
}
|
|
|
|
|
|
|
|
double lat, lon, elev_ft, multiuse;
|
|
|
|
int freq, range;
|
|
|
|
std::string name, ident;
|
|
|
|
aStream >> lat >> lon >> elev_ft >> freq >> range >> multiuse >> ident;
|
|
|
|
getline(aStream, name);
|
|
|
|
|
|
|
|
SGGeod pos(SGGeod::fromDegFt(lon, lat, elev_ft));
|
|
|
|
name = simgear::strutils::strip(name);
|
|
|
|
|
|
|
|
if ((rawType >= 7) && (rawType <= 9)) {
|
|
|
|
// marker beacons use a different run-time class now
|
2009-01-04 20:12:43 +00:00
|
|
|
FGMarkerBeaconRecord::create(rawType, name, pos);
|
2009-01-03 15:54:03 +00:00
|
|
|
return NULL; // not a nav-record, but that's okay
|
|
|
|
}
|
|
|
|
|
|
|
|
FGPositioned::Type type = mapRobinTypeToFGPType(rawType);
|
|
|
|
if (type == FGPositioned::INVALID) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// silently multiply adf frequencies by 100 so that adf
|
|
|
|
// vs. nav/loc frequency lookups can use the same code.
|
|
|
|
if (type == FGPositioned::NDB) {
|
|
|
|
freq *= 100;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new FGNavRecord(type, ident, name, pos,
|
|
|
|
freq, range, multiuse);
|
|
|
|
}
|
2004-05-28 05:24:54 +00:00
|
|
|
|
|
|
|
// load and initialize the navigational databases
|
2008-09-12 08:46:15 +00:00
|
|
|
bool fgNavDBInit( FGNavList *navlist, FGNavList *loclist, FGNavList *gslist,
|
2008-12-25 23:11:43 +00:00
|
|
|
FGNavList *dmelist,
|
2005-10-01 09:56:53 +00:00
|
|
|
FGNavList *tacanlist, FGNavList *carrierlist,
|
|
|
|
FGTACANList *channellist)
|
2004-05-28 05:24:54 +00:00
|
|
|
{
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Loading Navaid Databases");
|
|
|
|
|
|
|
|
SGPath path( globals->get_fg_root() );
|
|
|
|
path.append( "Navaids/nav.dat" );
|
|
|
|
|
|
|
|
sg_gzifstream in( path.str() );
|
|
|
|
if ( !in.is_open() ) {
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Cannot open file: " << path.str() );
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// skip first two lines
|
|
|
|
in >> skipeol;
|
|
|
|
in >> skipeol;
|
|
|
|
|
2008-09-12 08:46:15 +00:00
|
|
|
while (!in.eof()) {
|
2009-01-03 15:54:03 +00:00
|
|
|
FGNavRecord *r = createNavFromStream(in);
|
2008-09-12 08:46:15 +00:00
|
|
|
if (!r) {
|
2009-01-03 15:54:03 +00:00
|
|
|
continue;
|
2008-09-12 08:46:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (r->type()) {
|
|
|
|
case FGPositioned::NDB:
|
|
|
|
case FGPositioned::VOR:
|
|
|
|
navlist->add(r);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FGPositioned::ILS:
|
|
|
|
case FGPositioned::LOC:
|
|
|
|
loclist->add(r);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FGPositioned::GS:
|
|
|
|
gslist->add(r);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FGPositioned::DME:
|
|
|
|
{
|
|
|
|
dmelist->add(r);
|
2008-12-25 23:11:43 +00:00
|
|
|
string::size_type loc1= r->name().find( "TACAN", 0 );
|
|
|
|
string::size_type loc2 = r->name().find( "VORTAC", 0 );
|
2008-09-12 08:46:15 +00:00
|
|
|
|
|
|
|
if( loc1 != string::npos || loc2 != string::npos) {
|
|
|
|
tacanlist->add(r);
|
2004-05-28 05:24:54 +00:00
|
|
|
}
|
|
|
|
|
2008-09-12 08:46:15 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw sg_range_exception("got unsupported NavRecord type", "fgNavDBInit");
|
|
|
|
}
|
2004-05-28 05:24:54 +00:00
|
|
|
|
2008-09-12 08:46:15 +00:00
|
|
|
in >> skipcomment;
|
|
|
|
} // of stream data loop
|
2004-05-28 05:24:54 +00:00
|
|
|
|
2005-10-01 09:56:53 +00:00
|
|
|
// load the carrier navaids file
|
|
|
|
|
|
|
|
string file, name;
|
|
|
|
path = globals->get_fg_root() ;
|
|
|
|
path.append( "Navaids/carrier_nav.dat" );
|
|
|
|
|
|
|
|
file = path.str();
|
2005-12-06 18:48:56 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "opening file: " << path.str() );
|
2005-10-01 09:56:53 +00:00
|
|
|
|
|
|
|
sg_gzifstream incarrier( path.str() );
|
|
|
|
|
|
|
|
if ( !incarrier.is_open() ) {
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Cannot open file: " << path.str() );
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// skip first two lines
|
|
|
|
//incarrier >> skipeol;
|
|
|
|
//incarrier >> skipeol;
|
|
|
|
|
|
|
|
while ( ! incarrier.eof() ) {
|
2009-01-03 15:54:03 +00:00
|
|
|
FGNavRecord *r = createNavFromStream(incarrier);
|
2008-09-12 08:46:15 +00:00
|
|
|
if (!r) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
carrierlist->add (r);
|
2005-10-01 09:56:53 +00:00
|
|
|
} // end while
|
|
|
|
|
|
|
|
// end loading the carrier navaids file
|
|
|
|
|
|
|
|
// load the channel/freqency file
|
|
|
|
string channel, freq;
|
|
|
|
path="";
|
|
|
|
path = globals->get_fg_root();
|
|
|
|
path.append( "Navaids/TACAN_freq.dat" );
|
|
|
|
|
2005-12-06 18:48:56 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "opening file: " << path.str() );
|
|
|
|
|
2005-10-01 09:56:53 +00:00
|
|
|
sg_gzifstream inchannel( path.str() );
|
|
|
|
|
|
|
|
if ( !inchannel.is_open() ) {
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Cannot open file: " << path.str() );
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
2005-10-02 17:57:16 +00:00
|
|
|
// skip first line
|
2005-10-01 09:56:53 +00:00
|
|
|
inchannel >> skipeol;
|
|
|
|
while ( ! inchannel.eof() ) {
|
|
|
|
FGTACANRecord *r = new FGTACANRecord;
|
|
|
|
inchannel >> (*r);
|
|
|
|
channellist->add ( r );
|
|
|
|
//cout << "channel = " << r->get_channel() ;
|
|
|
|
//cout << " freq = " << r->get_freq() << endl;
|
|
|
|
|
|
|
|
} // end while
|
|
|
|
|
|
|
|
|
|
|
|
// end ReadChanFile
|
|
|
|
|
2004-05-28 05:24:54 +00:00
|
|
|
|
2010-12-07 17:55:45 +00:00
|
|
|
// flush all the parsed ils.xml data, we don't need it anymore,
|
|
|
|
// since it's been meregd into the FGNavRecords
|
|
|
|
static_airportIlsData.clear();
|
|
|
|
|
2004-05-28 05:24:54 +00:00
|
|
|
return true;
|
|
|
|
}
|
2009-01-03 15:54:03 +00:00
|
|
|
|
2010-12-07 17:55:45 +00:00
|
|
|
SGPropertyNode* ilsDataForRunwayAndNavaid(FGRunway* aRunway, const std::string& aNavIdent)
|
|
|
|
{
|
|
|
|
if (!fgGetBool("/sim/paths/use-custom-scenery-data")) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!aRunway) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
FGAirport* apt = aRunway->airport();
|
|
|
|
// find (or load) the airprot ILS data
|
|
|
|
AirportPropertyMap::iterator it = static_airportIlsData.find(apt);
|
|
|
|
if (it == static_airportIlsData.end()) {
|
|
|
|
SGPath path;
|
|
|
|
if (!XMLLoader::findAirportData(apt->ident(), "ils", path)) {
|
|
|
|
// no ils.xml file for this airpot, insert a NULL entry so we don't
|
|
|
|
// check again
|
|
|
|
static_airportIlsData.insert(it, std::make_pair(apt, SGPropertyNode_ptr()));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
SGPropertyNode_ptr rootNode = new SGPropertyNode;
|
|
|
|
readProperties(path.str(), rootNode);
|
|
|
|
it = static_airportIlsData.insert(it, std::make_pair(apt, rootNode));
|
|
|
|
} // of ils.xml file not loaded
|
|
|
|
|
|
|
|
if (!it->second) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// find the entry matching the runway
|
|
|
|
SGPropertyNode* runwayNode, *ilsNode;
|
|
|
|
for (int i=0; (runwayNode = it->second->getChild("runway", i)) != NULL; ++i) {
|
|
|
|
for (int j=0; (ilsNode = runwayNode->getChild("ils", j)) != NULL; ++j) {
|
|
|
|
// must match on both nav-ident and runway ident, to support the following:
|
|
|
|
// - runways with multiple distinct ILS installations (KEWD, for example)
|
|
|
|
// - runways where both ends share the same nav ident (LFAT, for example)
|
|
|
|
if ((ilsNode->getStringValue("nav-id") == aNavIdent) &&
|
|
|
|
(ilsNode->getStringValue("rwy") == aRunway->ident()))
|
|
|
|
{
|
|
|
|
return ilsNode;
|
|
|
|
}
|
|
|
|
} // of ILS iteration
|
|
|
|
} // of runway iteration
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-01-03 15:54:03 +00:00
|
|
|
FGRunway* getRunwayFromName(const std::string& aName)
|
|
|
|
{
|
|
|
|
vector<string> parts = simgear::strutils::split(aName);
|
|
|
|
if (parts.size() < 2) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_WARN, "getRunwayFromName: malformed name:" << aName);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
const FGAirport* apt = fgFindAirportID(parts[0]);
|
|
|
|
if (!apt) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_WARN, "navaid " << aName << " associated with bogus airport ID:" << parts[0]);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-12-19 17:55:22 +00:00
|
|
|
if (!apt->hasRunwayWithIdent(parts[1])) {
|
2009-01-03 15:54:03 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_WARN, "navaid " << aName << " associated with bogus runway ID:" << parts[1]);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-12-19 17:55:22 +00:00
|
|
|
return apt->getRunwayByIdent(parts[1]);
|
2009-01-03 15:54:03 +00:00
|
|
|
}
|