2012-08-18 13:00:08 +02:00
|
|
|
// fgelev.cxx -- compute scenery elevation
|
|
|
|
//
|
|
|
|
// Copyright (C) 2009 - 2012 Mathias Froehlich
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include <config.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <iomanip>
|
|
|
|
|
|
|
|
#include <osg/ArgumentParser>
|
2012-08-25 12:04:50 +02:00
|
|
|
#include <osg/Image>
|
2012-08-18 13:00:08 +02:00
|
|
|
|
|
|
|
#include <simgear/props/props.hxx>
|
|
|
|
#include <simgear/props/props_io.hxx>
|
2012-08-24 21:20:44 +02:00
|
|
|
#include <simgear/misc/sg_path.hxx>
|
2012-08-18 13:00:08 +02:00
|
|
|
#include <simgear/misc/ResourceManager.hxx>
|
2012-08-24 21:20:44 +02:00
|
|
|
#include <simgear/bvh/BVHNode.hxx>
|
|
|
|
#include <simgear/bvh/BVHLineSegmentVisitor.hxx>
|
2012-08-24 21:20:44 +02:00
|
|
|
#include <simgear/bvh/BVHPager.hxx>
|
|
|
|
#include <simgear/bvh/BVHPageNode.hxx>
|
|
|
|
#include <simgear/scene/material/matlib.hxx>
|
|
|
|
#include <simgear/scene/model/BVHPageNodeOSG.hxx>
|
|
|
|
#include <simgear/scene/model/ModelRegistry.hxx>
|
|
|
|
#include <simgear/scene/util/SGReaderWriterOptions.hxx>
|
2012-08-25 12:04:50 +02:00
|
|
|
#include <simgear/scene/util/OptionsReadFileCallback.hxx>
|
2012-08-24 21:20:44 +02:00
|
|
|
#include <simgear/scene/tgdb/userdata.hxx>
|
|
|
|
|
|
|
|
namespace sg = simgear;
|
2012-08-18 13:00:08 +02:00
|
|
|
|
2012-08-24 21:20:44 +02:00
|
|
|
class Visitor : public sg::BVHLineSegmentVisitor {
|
2012-08-18 13:00:08 +02:00
|
|
|
public:
|
2012-08-24 21:20:44 +02:00
|
|
|
Visitor(const SGLineSegmentd& lineSegment, sg::BVHPager& pager) :
|
|
|
|
BVHLineSegmentVisitor(lineSegment, 0),
|
|
|
|
_pager(pager)
|
2012-08-18 13:00:08 +02:00
|
|
|
{ }
|
2012-08-24 21:20:44 +02:00
|
|
|
virtual ~Visitor()
|
|
|
|
{ }
|
|
|
|
virtual void apply(sg::BVHPageNode& node)
|
2012-08-18 13:00:08 +02:00
|
|
|
{
|
2012-08-24 21:20:44 +02:00
|
|
|
// we have a non threaded pager so load just right here.
|
|
|
|
_pager.use(node);
|
|
|
|
BVHLineSegmentVisitor::apply(node);
|
2012-08-18 13:00:08 +02:00
|
|
|
}
|
|
|
|
private:
|
2012-08-24 21:20:44 +02:00
|
|
|
sg::BVHPager& _pager;
|
2012-08-18 13:00:08 +02:00
|
|
|
};
|
|
|
|
|
2012-08-25 12:04:50 +02:00
|
|
|
// Short circuit reading image files.
|
|
|
|
class ReadFileCallback : public sg::OptionsReadFileCallback {
|
|
|
|
public:
|
|
|
|
virtual ~ReadFileCallback()
|
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual osgDB::ReaderWriter::ReadResult readImage(const std::string& name, const osgDB::Options*)
|
|
|
|
{ return new osg::Image; }
|
|
|
|
};
|
|
|
|
|
2012-08-24 21:20:44 +02:00
|
|
|
static bool
|
|
|
|
intersect(sg::BVHNode& node, sg::BVHPager& pager,
|
|
|
|
const SGVec3d& start, SGVec3d& end, double offset)
|
|
|
|
{
|
|
|
|
SGVec3d perp = offset*perpendicular(start - end);
|
|
|
|
Visitor visitor(SGLineSegmentd(start + perp, end + perp), pager);
|
|
|
|
node.accept(visitor);
|
|
|
|
if (visitor.empty())
|
|
|
|
return false;
|
|
|
|
end = visitor.getLineSegment().getEnd();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-08-18 13:00:08 +02:00
|
|
|
int
|
|
|
|
main(int argc, char** argv)
|
|
|
|
{
|
|
|
|
/// Read arguments and environment variables.
|
|
|
|
|
|
|
|
// use an ArgumentParser object to manage the program arguments.
|
|
|
|
osg::ArgumentParser arguments(&argc, argv);
|
|
|
|
|
|
|
|
std::string fg_root;
|
|
|
|
if (arguments.read("--fg-root", fg_root)) {
|
|
|
|
} else if (const char *fg_root_env = std::getenv("FG_ROOT")) {
|
|
|
|
fg_root = fg_root_env;
|
|
|
|
} else {
|
|
|
|
fg_root = PKGLIBDIR;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string fg_scenery;
|
|
|
|
if (arguments.read("--fg-scenery", fg_scenery)) {
|
|
|
|
} else if (const char *fg_scenery_env = std::getenv("FG_SCENERY")) {
|
|
|
|
fg_scenery = fg_scenery_env;
|
|
|
|
} else {
|
|
|
|
SGPath path(fg_root);
|
|
|
|
path.append("Scenery");
|
|
|
|
fg_scenery = path.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
SGSharedPtr<SGPropertyNode> props = new SGPropertyNode;
|
|
|
|
try {
|
|
|
|
SGPath preferencesFile = fg_root;
|
|
|
|
preferencesFile.append("preferences.xml");
|
|
|
|
readProperties(preferencesFile.str(), props);
|
|
|
|
} catch (...) {
|
|
|
|
// In case of an error, at least make summer :)
|
|
|
|
props->getNode("sim/startup/season", true)->setStringValue("summer");
|
|
|
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Problems loading FlightGear preferences.\n"
|
|
|
|
<< "Probably FG_ROOT is not properly set.");
|
|
|
|
}
|
|
|
|
|
|
|
|
/// now set up the simgears required model stuff
|
|
|
|
|
|
|
|
simgear::ResourceManager::instance()->addBasePath(fg_root, simgear::ResourceManager::PRIORITY_DEFAULT);
|
|
|
|
// Just reference simgears reader writer stuff so that the globals get
|
|
|
|
// pulled in by the linker ...
|
|
|
|
simgear::ModelRegistry::instance();
|
|
|
|
|
|
|
|
sgUserDataInit(props.get());
|
|
|
|
SGMaterialLib* ml = new SGMaterialLib;
|
|
|
|
SGPath mpath(fg_root);
|
|
|
|
mpath.append("Materials/default/materials.xml");
|
|
|
|
try {
|
|
|
|
ml->load(fg_root, mpath.str(), props);
|
|
|
|
} catch (...) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Problems loading FlightGear materials.\n"
|
|
|
|
<< "Probably FG_ROOT is not properly set.");
|
|
|
|
}
|
|
|
|
simgear::SGModelLib::init(fg_root, props);
|
|
|
|
|
|
|
|
// Set up the reader/writer options
|
|
|
|
osg::ref_ptr<simgear::SGReaderWriterOptions> options;
|
|
|
|
if (osgDB::Options* ropt = osgDB::Registry::instance()->getOptions())
|
|
|
|
options = new simgear::SGReaderWriterOptions(*ropt);
|
|
|
|
else
|
|
|
|
options = new simgear::SGReaderWriterOptions;
|
|
|
|
osgDB::convertStringPathIntoFilePathList(fg_scenery,
|
|
|
|
options->getDatabasePathList());
|
|
|
|
options->setMaterialLib(ml);
|
|
|
|
options->setPropertyNode(props);
|
2012-08-25 12:04:50 +02:00
|
|
|
options->setReadFileCallback(new ReadFileCallback);
|
2012-08-18 13:00:08 +02:00
|
|
|
options->setPluginStringData("SimGear::FG_ROOT", fg_root);
|
2012-08-24 21:20:44 +02:00
|
|
|
// we do not need the builtin boundingvolumes
|
|
|
|
options->setPluginStringData("SimGear::BOUNDINGVOLUMES", "OFF");
|
2012-08-25 08:43:25 +02:00
|
|
|
// And we only want terrain, no objects on top.
|
|
|
|
options->setPluginStringData("SimGear::FG_ONLY_TERRAIN", "ON");
|
|
|
|
props->getNode("sim/rendering/random-objects", true)->setBoolValue(false);
|
|
|
|
props->getNode("sim/rendering/random-vegetation", true)->setBoolValue(false);
|
2012-08-18 13:00:08 +02:00
|
|
|
|
|
|
|
// Here, all arguments are processed
|
|
|
|
arguments.reportRemainingOptionsAsUnrecognized();
|
|
|
|
arguments.writeErrorMessages(std::cerr);
|
|
|
|
|
2012-08-24 21:20:44 +02:00
|
|
|
// Get the whole world bvh tree
|
|
|
|
SGSharedPtr<sg::BVHNode> node;
|
|
|
|
node = sg::BVHPageNodeOSG::load("w180s90-360x180.spt", options);
|
2012-08-18 13:00:08 +02:00
|
|
|
|
|
|
|
// if no model has been successfully loaded report failure.
|
2012-08-24 21:20:44 +02:00
|
|
|
if (!node.valid()) {
|
2012-08-18 13:00:08 +02:00
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, arguments.getApplicationName()
|
|
|
|
<< ": No data loaded");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2012-08-24 21:20:44 +02:00
|
|
|
// We assume that the above is a paged database.
|
|
|
|
sg::BVHPager pager;
|
|
|
|
|
2012-08-24 21:20:44 +02:00
|
|
|
while (std::cin.good()) {
|
2012-08-24 21:20:44 +02:00
|
|
|
// Increment the paging relevant number
|
|
|
|
pager.setUseStamp(1 + pager.getUseStamp());
|
|
|
|
// and expire everything not accessed for the past 30 requests
|
|
|
|
pager.update(10);
|
|
|
|
|
2012-08-24 21:20:44 +02:00
|
|
|
std::string id;
|
|
|
|
std::cin >> id;
|
2012-08-18 13:00:08 +02:00
|
|
|
double lon, lat;
|
2012-08-24 21:20:44 +02:00
|
|
|
std::cin >> lon >> lat;
|
|
|
|
if (std::cin.fail())
|
2012-08-18 13:00:08 +02:00
|
|
|
return EXIT_FAILURE;
|
2012-08-24 21:20:44 +02:00
|
|
|
std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
|
2012-08-18 13:00:08 +02:00
|
|
|
|
|
|
|
SGVec3d start = SGVec3d::fromGeod(SGGeod::fromDegM(lon, lat, 10000));
|
|
|
|
SGVec3d end = SGVec3d::fromGeod(SGGeod::fromDegM(lon, lat, -1000));
|
2012-08-24 21:20:44 +02:00
|
|
|
|
|
|
|
// Try to find an intersection
|
|
|
|
bool found = intersect(*node, pager, start, end, 0);
|
|
|
|
double scale = 1e-5;
|
|
|
|
while (!found && scale <= 1) {
|
|
|
|
found = intersect(*node, pager, start, end, scale);
|
|
|
|
scale *= 2;
|
|
|
|
}
|
|
|
|
if (1e-5 < scale)
|
|
|
|
std::cerr << "Found hole of minimum diameter "
|
|
|
|
<< scale << "m at lon = " << lon
|
|
|
|
<< "deg lat = " << lat << "deg" << std::endl;
|
2012-08-18 13:00:08 +02:00
|
|
|
|
2012-08-24 21:20:44 +02:00
|
|
|
std::cout << id << ": ";
|
2012-08-24 21:20:44 +02:00
|
|
|
if (!found) {
|
2012-08-18 13:00:08 +02:00
|
|
|
std::cout << "-1000" << std::endl;
|
|
|
|
} else {
|
2012-08-24 21:20:44 +02:00
|
|
|
SGGeod geod = SGGeod::fromCart(end);
|
2012-08-18 13:00:08 +02:00
|
|
|
std::cout << std::fixed << std::setprecision(3) << geod.getElevationM() << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|