1
0
Fork 0
flightgear/src/FDM/groundcache.cxx
Richard Harrison 88f43a33a3 Remove messages that appear when no tile is loaded because it isn't present in the scenery
These get a little spammy when flying over areas without terrain and don't really serve much of a purpose.
2019-01-20 23:26:58 +01:00

1060 lines
34 KiB
C++

// groundcache.cxx -- carries a small subset of the scenegraph near the vehicle
//
// Written by Mathias Froehlich, started Nov 2004.
//
// Copyright (C) 2004, 2009 Mathias Froehlich - Mathias.Froehlich@web.de
//
// 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.
//
// $Id$
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include "groundcache.hxx"
#include <utility>
#include <osg/Drawable>
#include <osg/Geode>
#include <osg/Geometry>
#include <osg/Camera>
#include <osg/Transform>
#include <osg/MatrixTransform>
#include <osg/PositionAttitudeTransform>
#include <osg/CameraView>
#include <simgear/sg_inlines.h>
#include <simgear/constants.h>
#include <simgear/debug/logstream.hxx>
#include <simgear/math/SGMisc.hxx>
#include <simgear/scene/material/mat.hxx>
#include <simgear/scene/util/SGNodeMasks.hxx>
#include <simgear/scene/util/SGSceneUserData.hxx>
#include <simgear/scene/util/OsgMath.hxx>
#include <simgear/bvh/BVHNode.hxx>
#include <simgear/bvh/BVHGroup.hxx>
#include <simgear/bvh/BVHTransform.hxx>
#include <simgear/bvh/BVHMotionTransform.hxx>
#include <simgear/bvh/BVHLineGeometry.hxx>
#include <simgear/bvh/BVHStaticGeometry.hxx>
#include <simgear/bvh/BVHStaticData.hxx>
#include <simgear/bvh/BVHStaticNode.hxx>
#include <simgear/bvh/BVHStaticTriangle.hxx>
#include <simgear/bvh/BVHStaticBinary.hxx>
#include <simgear/bvh/BVHSubTreeCollector.hxx>
#include <simgear/bvh/BVHLineSegmentVisitor.hxx>
#include <simgear/bvh/BVHNearestPointVisitor.hxx>
#ifdef GROUNDCACHE_DEBUG
#include <simgear/scene/model/BVHDebugCollectVisitor.hxx>
#include <Main/fg_props.hxx>
#endif
#include <Main/globals.hxx>
#include <Scenery/scenery.hxx>
#include "flight.hxx"
using namespace simgear;
class FGGroundCache::CacheFill : public osg::NodeVisitor {
public:
CacheFill(const SGVec3d& center, const SGVec3d& down, const double& radius,
const double& startTime, const double& endTime) :
osg::NodeVisitor(osg::NodeVisitor::TRAVERSE_ACTIVE_CHILDREN),
_center(center),
_down(down),
_radius(radius),
_startTime(startTime),
_endTime(endTime),
_sceneryHit(0, 0, 0),
_maxDown(SGGeod::fromCart(center).getElevationM() + 9999),
_material(0),
_haveHit(false)
{
setTraversalMask(SG_NODEMASK_TERRAIN_BIT);
}
virtual void apply(osg::Node& node)
{
if (!testBoundingSphere(node.getBound()))
return;
addBoundingVolume(node);
}
virtual void apply(osg::Group& group)
{
if (!testBoundingSphere(group.getBound()))
return;
simgear::BVHSubTreeCollector::NodeList parentNodeList;
mSubTreeCollector.pushNodeList(parentNodeList);
traverse(group);
addBoundingVolume(group);
mSubTreeCollector.popNodeList(parentNodeList);
}
virtual void apply(osg::Transform& transform)
{ handleTransform(transform); }
virtual void apply(osg::Camera& camera)
{
if (camera.getRenderOrder() != osg::Camera::NESTED_RENDER)
return;
handleTransform(camera);
}
virtual void apply(osg::CameraView& transform)
{ handleTransform(transform); }
virtual void apply(osg::MatrixTransform& transform)
{ handleTransform(transform); }
virtual void apply(osg::PositionAttitudeTransform& transform)
{ handleTransform(transform); }
void handleTransform(osg::Transform& transform)
{
// Hmm, may be this needs to be refined somehow ...
if (transform.getReferenceFrame() != osg::Transform::RELATIVE_RF)
return;
if (!testBoundingSphere(transform.getBound()))
return;
osg::Matrix inverseMatrix;
if (!transform.computeWorldToLocalMatrix(inverseMatrix, this))
return;
osg::Matrix matrix;
if (!transform.computeLocalToWorldMatrix(matrix, this))
return;
// Look for a velocity note
const SGSceneUserData::Velocity* velocity = getVelocity(transform);
SGVec3d center = _center;
SGVec3d down = _down;
double radius = _radius;
bool haveHit = _haveHit;
const simgear::BVHMaterial* material = _material;
_haveHit = false;
_center = toSG(inverseMatrix.preMult(toOsg(_center)));
_down = toSG(osg::Matrix::transform3x3(toOsg(_down), inverseMatrix));
if (velocity) {
SGVec3d staticCenter(_center);
double dtStart = velocity->referenceTime - _startTime;
SGVec3d startCenter = staticCenter + dtStart*velocity->linear;
SGQuatd startOr(SGQuatd::fromAngleAxis(dtStart*velocity->angular));
startCenter = startOr.transform(startCenter);
double dtEnd = velocity->referenceTime - _endTime;
SGVec3d endCenter = staticCenter + dtEnd*velocity->linear;
SGQuatd endOr(SGQuatd::fromAngleAxis(dtEnd*velocity->angular));
endCenter = endOr.transform(endCenter);
_center = 0.5*(startCenter + endCenter);
_down = startOr.transform(_down);
_radius += 0.5*dist(startCenter, endCenter);
}
simgear::BVHSubTreeCollector::NodeList parentNodeList;
mSubTreeCollector.pushNodeList(parentNodeList);
addBoundingVolume(transform);
traverse(transform);
if (mSubTreeCollector.haveChildren()) {
if (velocity) {
simgear::BVHMotionTransform* bvhTransform;
bvhTransform = new simgear::BVHMotionTransform;
bvhTransform->setToWorldTransform(SGMatrixd(matrix.ptr()));
bvhTransform->setLinearVelocity(velocity->linear);
bvhTransform->setAngularVelocity(velocity->angular);
bvhTransform->setReferenceTime(velocity->referenceTime);
bvhTransform->setStartTime(_startTime);
bvhTransform->setEndTime(_endTime);
bvhTransform->setId(velocity->id);
mSubTreeCollector.popNodeList(parentNodeList, bvhTransform);
} else {
simgear::BVHTransform* bvhTransform;
bvhTransform = new simgear::BVHTransform;
bvhTransform->setToWorldTransform(SGMatrixd(matrix.ptr()));
mSubTreeCollector.popNodeList(parentNodeList, bvhTransform);
}
} else {
mSubTreeCollector.popNodeList(parentNodeList);
}
if (_haveHit) {
if (velocity) {
double dt = _startTime - velocity->referenceTime;
SGQuatd ori(SGQuatd::fromAngleAxis(dt*velocity->angular));
_sceneryHit = ori.transform(_sceneryHit);
_sceneryHit += dt*velocity->linear;
}
_sceneryHit = toSG(matrix.preMult(toOsg(_sceneryHit)));
} else {
_material = material;
_haveHit = haveHit;
}
_center = center;
_down = down;
_radius = radius;
}
const SGSceneUserData::Velocity* getVelocity(osg::Node& node)
{
SGSceneUserData* userData = SGSceneUserData::getSceneUserData(&node);
if (!userData)
return 0;
return userData->getVelocity();
}
simgear::BVHNode* getNodeBoundingVolume(osg::Node& node)
{
SGSceneUserData* userData = SGSceneUserData::getSceneUserData(&node);
if (!userData)
return 0;
return userData->getBVHNode();
}
void addBoundingVolume(osg::Node& node)
{
simgear::BVHNode* bvNode = getNodeBoundingVolume(node);
if (!bvNode)
return;
// Find a croase ground intersection
SGLineSegmentd line(_center + _radius*_down, _center + _maxDown*_down);
simgear::BVHLineSegmentVisitor lineSegmentVisitor(line, _startTime);
bvNode->accept(lineSegmentVisitor);
if (!lineSegmentVisitor.empty()) {
_sceneryHit = lineSegmentVisitor.getPoint();
_material = lineSegmentVisitor.getMaterial();
_maxDown = SGMiscd::max(_radius, dot(_down, _sceneryHit - _center));
_haveHit = true;
}
// Get that part of the local bv tree that intersects our sphere
// of interrest.
mSubTreeCollector.setSphere(SGSphered(_center, _radius));
bvNode->accept(mSubTreeCollector);
}
bool testBoundingSphere(const osg::BoundingSphere& bound) const
{
if (!bound.valid())
return false;
SGLineSegmentd downSeg(_center, _center + _maxDown*_down);
double maxDist = bound._radius + _radius;
SGVec3d boundCenter(toVec3d(toSG(bound._center)));
return distSqr(downSeg, boundCenter) <= maxDist*maxDist;
}
SGSharedPtr<simgear::BVHNode> getBVHNode() const
{ return mSubTreeCollector.getNode(); }
bool getHaveElevationBelowCache() const
{ return _haveHit; }
double getElevationBelowCache() const
{ return SGGeod::fromCart(_sceneryHit).getElevationM(); }
const simgear::BVHMaterial* getMaterialBelowCache() const
{ return _material; }
private:
SGVec3d _center;
SGVec3d _down;
double _radius;
double _startTime;
double _endTime;
simgear::BVHSubTreeCollector mSubTreeCollector;
SGVec3d _sceneryHit;
double _maxDown;
const simgear::BVHMaterial* _material;
bool _haveHit;
};
FGGroundCache::FGGroundCache() :
_altitude(0),
_material(0),
cache_ref_time(0),
cache_time_offset(0),
_wire(0),
reference_wgs84_point(SGVec3d(0, 0, 0)),
reference_vehicle_radius(0),
down(0.0, 0.0, 0.0),
found_ground(false)
{
#ifdef GROUNDCACHE_DEBUG
_lookupTime = SGTimeStamp::fromSec(0.0);
_lookupCount = 0;
_buildTime = SGTimeStamp::fromSec(0.0);
_buildCount = 0;
#endif
}
FGGroundCache::~FGGroundCache()
{
}
bool
FGGroundCache::prepare_ground_cache(double startSimTime, double endSimTime,
const SGVec3d& pt, double rad)
{
if (rad > 10000.0) {
SG_LOG(SG_FLIGHT, SG_DEV_WARN, "FGGroundCache::prepare_ground_cache passed an excessive radius");
rad = 10000.0;
}
#ifdef GROUNDCACHE_DEBUG
SGTimeStamp t0 = SGTimeStamp::now();
#endif
// Empty cache.
found_ground = false;
SGGeod geodPt = SGGeod::fromCart(pt);
// Don't blow away the cache ground_radius and stuff if there's no
// scenery
if (!globals->get_scenery()->schedule_scenery(geodPt, rad, 1.0)) {
SG_LOG(SG_FLIGHT, SG_BULK, "prepare_ground_cache(): scenery_available "
"returns false at " << geodPt << " " << pt << " " << rad);
return false;
}
_material = 0;
// If we have an active wire, get some more area into the groundcache
if (_wire)
rad = SGMiscd::max(200, rad);
// Store the parameters we used to build up that cache.
reference_wgs84_point = pt;
reference_vehicle_radius = rad;
// Store the time reference used to compute movements of moving triangles.
cache_ref_time = startSimTime;
// Get a normalized down vector valid for the whole cache
SGQuatd hlToEc = SGQuatd::fromLonLat(geodPt);
down = hlToEc.rotate(SGVec3d(0, 0, 1));
// Get the ground cache, that is a local collision tree of the environment
startSimTime += cache_time_offset;
endSimTime += cache_time_offset;
CacheFill subtreeCollector(pt, down, rad, startSimTime, endSimTime);
globals->get_scenery()->get_scene_graph()->accept(subtreeCollector);
_localBvhTree = subtreeCollector.getBVHNode();
if (subtreeCollector.getHaveElevationBelowCache()) {
// Use the altitude value below the cache that we gathered during
// cache collection
_altitude = subtreeCollector.getElevationBelowCache();
_material = subtreeCollector.getMaterialBelowCache();
found_ground = true;
} else if (_localBvhTree) {
// We have nothing below us, so try starting with the lowest point
// upwards for a croase altitude value
SGLineSegmentd line(pt + reference_vehicle_radius*down, pt - 1e3*down);
simgear::BVHLineSegmentVisitor lineSegmentVisitor(line, startSimTime);
_localBvhTree->accept(lineSegmentVisitor);
if (!lineSegmentVisitor.empty()) {
SGGeod geodPt = SGGeod::fromCart(lineSegmentVisitor.getPoint());
_altitude = geodPt.getElevationM();
_material = lineSegmentVisitor.getMaterial();
found_ground = true;
}
}
if (!found_ground) {
// Ok, still nothing here?? Last resort ...
double alt = 0;
_material = 0;
found_ground = globals->get_scenery()->
get_elevation_m(SGGeod::fromGeodM(geodPt, 10000), alt, &_material);
if (found_ground)
_altitude = alt;
}
// RJH: 2018-12-31: Remove this message as it happens too frequently when flying over areas of missing terrain
// and realistically it doesn't really give much information to help identify or resolve a problem
// which is evident when looking out of the window.
//if (!found_ground)
// SG_LOG(SG_FLIGHT, SG_WARN, "prepare_ground_cache(): trying to build "
// "cache without any scenery below the aircraft");
#ifdef GROUNDCACHE_DEBUG
t0 = SGTimeStamp::now() - t0;
_buildTime += t0;
_buildCount++;
if (_buildCount > 60) {
double buildTime = 0;
if (_buildCount)
buildTime = _buildTime.toSecs()/_buildCount;
double lookupTime = 0;
if (_lookupCount)
lookupTime = _lookupTime.toSecs()/_lookupCount;
_buildTime = SGTimeStamp::fromSec(0.0);
_buildCount = 0;
_lookupTime = SGTimeStamp::fromSec(0.0);
_lookupCount = 0;
SG_LOG(SG_FLIGHT, SG_ALERT, "build time = " << buildTime
<< ", lookup Time = " << lookupTime);
}
if (!_group.valid()) {
_group = new osg::Group;
globals->get_scenery()->get_scene_graph()->addChild(_group);
fgSetInt("/fdm/groundcache-debug-level", -3);
}
_group->removeChildren(0, _group->getNumChildren());
if (_localBvhTree) {
int level = fgGetInt("/fdm/groundcache-debug-level");
if (-2 <= level) {
simgear::BVHDebugCollectVisitor debug(endSimTime, level);
_localBvhTree->accept(debug);
_group->addChild(debug.getNode());
}
}
#endif
return found_ground;
}
bool
FGGroundCache::is_valid(double& ref_time, SGVec3d& pt, double& rad)
{
pt = reference_wgs84_point;
rad = reference_vehicle_radius;
ref_time = cache_ref_time;
return found_ground;
}
class FGGroundCache::BodyFinder : public BVHVisitor {
public:
BodyFinder(BVHNode::Id id, const double& t) :
_id(id),
_bodyToWorld(SGMatrixd::unit()),
_linearVelocity(0, 0, 0),
_angularVelocity(0, 0, 0),
_foundId(false),
_time(t)
{ }
virtual void apply(BVHGroup& leaf)
{
if (_foundId)
return;
leaf.traverse(*this);
}
virtual void apply(BVHPageNode& leaf)
{
if (_foundId)
return;
leaf.traverse(*this);
}
virtual void apply(BVHTransform& transform)
{
if (_foundId)
return;
transform.traverse(*this);
if (_foundId) {
_linearVelocity = transform.vecToWorld(_linearVelocity);
_angularVelocity = transform.vecToWorld(_angularVelocity);
_bodyToWorld = transform.getToWorldTransform()*_bodyToWorld;
}
}
virtual void apply(BVHMotionTransform& transform)
{
if (_foundId)
return;
if (_id == transform.getId()) {
_foundId = true;
} else {
transform.traverse(*this);
}
if (_foundId) {
SGMatrixd toWorld = transform.getToWorldTransform(_time);
SGVec3d referencePoint = _bodyToWorld.xformPt(SGVec3d::zeros());
_linearVelocity += transform.getLinearVelocityAt(referencePoint);
_angularVelocity += transform.getAngularVelocity();
_linearVelocity = toWorld.xformVec(_linearVelocity);
_angularVelocity = toWorld.xformVec(_angularVelocity);
_bodyToWorld = toWorld*_bodyToWorld;
}
}
virtual void apply(BVHLineGeometry& node) { }
virtual void apply(BVHStaticGeometry& node) { }
virtual void apply(const BVHStaticBinary&, const BVHStaticData&) { }
virtual void apply(const BVHStaticTriangle&, const BVHStaticData&) { }
const SGMatrixd& getBodyToWorld() const
{ return _bodyToWorld; }
const SGVec3d& getLinearVelocity() const
{ return _linearVelocity; }
const SGVec3d& getAngularVelocity() const
{ return _angularVelocity; }
bool empty() const
{ return !_foundId; }
protected:
simgear::BVHNode::Id _id;
SGMatrixd _bodyToWorld;
SGVec3d _linearVelocity;
SGVec3d _angularVelocity;
bool _foundId;
double _time;
};
bool
FGGroundCache::get_body(double t, SGMatrixd& bodyToWorld, SGVec3d& linearVel,
SGVec3d& angularVel, simgear::BVHNode::Id id)
{
// Get the transform matrix and velocities of a moving body with id at t.
if (!_localBvhTree)
return false;
t += cache_time_offset;
BodyFinder bodyFinder(id, t);
_localBvhTree->accept(bodyFinder);
bodyToWorld = bodyFinder.getBodyToWorld();
linearVel = bodyFinder.getLinearVelocity();
angularVel = bodyFinder.getAngularVelocity();
if (bodyFinder.empty())
return false;
return true;
}
class FGGroundCache::CatapultFinder : public BVHVisitor {
public:
CatapultFinder(const SGSphered& sphere, const double& t) :
_haveLineSegment(false),
_sphere(sphere),
_time(t)
{ }
virtual void apply(BVHGroup& leaf)
{
if (!intersects(_sphere, leaf.getBoundingSphere()))
return;
leaf.traverse(*this);
}
virtual void apply(BVHPageNode& leaf)
{
if (!intersects(_sphere, leaf.getBoundingSphere()))
return;
leaf.traverse(*this);
}
virtual void apply(BVHTransform& transform)
{
if (!intersects(_sphere, transform.getBoundingSphere()))
return;
SGSphered sphere = _sphere;
_sphere = transform.sphereToLocal(sphere);
bool haveLineSegment = _haveLineSegment;
_haveLineSegment = false;
transform.traverse(*this);
if (_haveLineSegment) {
_lineSegment = transform.lineSegmentToWorld(_lineSegment);
_linearVelocity = transform.vecToWorld(_linearVelocity);
_angularVelocity = transform.vecToWorld(_angularVelocity);
}
_haveLineSegment |= haveLineSegment;
_sphere.setCenter(sphere.getCenter());
}
virtual void apply(BVHMotionTransform& transform)
{
if (!intersects(_sphere, transform.getBoundingSphere()))
return;
SGSphered sphere = _sphere;
_sphere = transform.sphereToLocal(sphere, _time);
bool haveLineSegment = _haveLineSegment;
_haveLineSegment = false;
transform.traverse(*this);
if (_haveLineSegment) {
SGMatrixd toWorld = transform.getToWorldTransform(_time);
_linearVelocity
+= transform.getLinearVelocityAt(_lineSegment.getStart());
_angularVelocity += transform.getAngularVelocity();
_linearVelocity = toWorld.xformVec(_linearVelocity);
_angularVelocity = toWorld.xformVec(_angularVelocity);
_lineSegment = _lineSegment.transform(toWorld);
}
_haveLineSegment |= haveLineSegment;
_sphere.setCenter(sphere.getCenter());
}
virtual void apply(BVHLineGeometry& node)
{
if (node.getType() != BVHLineGeometry::CarrierCatapult)
return;
SGLineSegmentd lineSegment(node.getLineSegment());
if (!intersects(_sphere, lineSegment))
return;
_lineSegment = lineSegment;
double dist = distSqr(lineSegment, getSphere().getCenter());
_sphere.setRadius(sqrt(dist));
_linearVelocity = SGVec3d::zeros();
_angularVelocity = SGVec3d::zeros();
_haveLineSegment = true;
}
virtual void apply(BVHStaticGeometry& node)
{ }
virtual void apply(const BVHStaticBinary&, const BVHStaticData&) { }
virtual void apply(const BVHStaticTriangle&, const BVHStaticData&) { }
void setSphere(const SGSphered& sphere)
{ _sphere = sphere; }
const SGSphered& getSphere() const
{ return _sphere; }
const SGLineSegmentd& getLineSegment() const
{ return _lineSegment; }
const SGVec3d& getLinearVelocity() const
{ return _linearVelocity; }
const SGVec3d& getAngularVelocity() const
{ return _angularVelocity; }
bool getHaveLineSegment() const
{ return _haveLineSegment; }
protected:
SGLineSegmentd _lineSegment;
SGVec3d _linearVelocity;
SGVec3d _angularVelocity;
bool _haveLineSegment;
SGSphered _sphere;
double _time;
};
double
FGGroundCache::get_cat(double t, const SGVec3d& pt,
SGVec3d end[2], SGVec3d vel[2])
{
double maxDistance = 1000;
// Get the wire in question
t += cache_time_offset;
CatapultFinder catapultFinder(SGSphered(pt, maxDistance), t);
if (_localBvhTree)
_localBvhTree->accept(catapultFinder);
if (!catapultFinder.getHaveLineSegment())
return maxDistance;
// prepare the returns
end[0] = catapultFinder.getLineSegment().getStart();
end[1] = catapultFinder.getLineSegment().getEnd();
// The linear velocity is the one at the start of the line segment ...
vel[0] = catapultFinder.getLinearVelocity();
// ... so the end point has the additional cross product.
vel[1] = catapultFinder.getLinearVelocity();
vel[1] += cross(catapultFinder.getAngularVelocity(),
catapultFinder.getLineSegment().getDirection());
// Return the distance to the cat
return sqrt(distSqr(catapultFinder.getLineSegment(), pt));
}
bool
FGGroundCache::get_agl(double t, const SGVec3d& pt, SGVec3d& contact,
SGVec3d& normal, SGVec3d& linearVel, SGVec3d& angularVel,
simgear::BVHNode::Id& id, const simgear::BVHMaterial*& material)
{
#ifdef GROUNDCACHE_DEBUG
SGTimeStamp t0 = SGTimeStamp::now();
#endif
// Just set up a ground intersection query for the given point
SGLineSegmentd line(pt, pt + 10*reference_vehicle_radius*down);
t += cache_time_offset;
simgear::BVHLineSegmentVisitor lineSegmentVisitor(line, t);
if (_localBvhTree)
_localBvhTree->accept(lineSegmentVisitor);
#ifdef GROUNDCACHE_DEBUG
t0 = SGTimeStamp::now() - t0;
_lookupTime += t0;
_lookupCount++;
#endif
if (!lineSegmentVisitor.empty()) {
// Have an intersection
contact = lineSegmentVisitor.getPoint();
normal = lineSegmentVisitor.getNormal();
if (0 < dot(normal, down))
normal = -normal;
linearVel = lineSegmentVisitor.getLinearVelocity();
angularVel = lineSegmentVisitor.getAngularVelocity();
material = lineSegmentVisitor.getMaterial();
id = lineSegmentVisitor.getId();
return true;
} else {
// Whenever we did not have a ground triangle for the requested point,
// take the ground level we found during the current cache build.
// This is as good as what we had before for agl.
SGGeod geodPt = SGGeod::fromCart(pt);
geodPt.setElevationM(_altitude);
contact = SGVec3d::fromGeod(geodPt);
normal = -down;
linearVel = SGVec3d(0, 0, 0);
angularVel = SGVec3d(0, 0, 0);
material = _material;
id = 0;
return found_ground;
}
}
bool
FGGroundCache::get_nearest(double t, const SGVec3d& pt, double maxDist,
SGVec3d& contact, SGVec3d& linearVel,
SGVec3d& angularVel, simgear::BVHNode::Id& id,
const simgear::BVHMaterial*& material)
{
if (!_localBvhTree)
return false;
#ifdef GROUNDCACHE_DEBUG
SGTimeStamp t0 = SGTimeStamp::now();
#endif
// Just set up a ground intersection query for the given point
SGSphered sphere(pt, maxDist);
t += cache_time_offset;
simgear::BVHNearestPointVisitor nearestPointVisitor(sphere, t);
_localBvhTree->accept(nearestPointVisitor);
#ifdef GROUNDCACHE_DEBUG
t0 = SGTimeStamp::now() - t0;
_lookupTime += t0;
_lookupCount++;
#endif
if (nearestPointVisitor.empty())
return false;
// Have geometry in the range of maxDist
contact = nearestPointVisitor.getPoint();
linearVel = nearestPointVisitor.getLinearVelocity();
angularVel = nearestPointVisitor.getAngularVelocity();
material = nearestPointVisitor.getMaterial();
id = nearestPointVisitor.getId();
return true;
}
class FGGroundCache::WireIntersector : public BVHVisitor {
public:
WireIntersector(const SGVec3d pt[4], const double& t) :
_linearVelocity(SGVec3d::zeros()),
_angularVelocity(SGVec3d::zeros()),
_wire(0),
_time(t)
{
// Build the two triangles spanning the area where the hook has moved
// during the past step.
_triangles[0].set(pt[0], pt[1], pt[2]);
_triangles[1].set(pt[0], pt[2], pt[3]);
}
virtual void apply(BVHGroup& leaf)
{
if (!_intersects(leaf.getBoundingSphere()))
return;
leaf.traverse(*this);
}
virtual void apply(BVHPageNode& leaf)
{
if (!_intersects(leaf.getBoundingSphere()))
return;
leaf.traverse(*this);
}
virtual void apply(BVHTransform& transform)
{
if (!_intersects(transform.getBoundingSphere()))
return;
SGTriangled triangles[2] = { _triangles[0], _triangles[1] };
_triangles[0] = triangles[0].transform(transform.getToLocalTransform());
_triangles[1] = triangles[1].transform(transform.getToLocalTransform());
transform.traverse(*this);
if (_wire) {
_lineSegment = transform.lineSegmentToWorld(_lineSegment);
_linearVelocity = transform.vecToWorld(_linearVelocity);
_angularVelocity = transform.vecToWorld(_angularVelocity);
}
_triangles[0] = triangles[0];
_triangles[1] = triangles[1];
}
virtual void apply(BVHMotionTransform& transform)
{
if (!_intersects(transform.getBoundingSphere()))
return;
SGMatrixd toLocal = transform.getToLocalTransform(_time);
SGTriangled triangles[2] = { _triangles[0], _triangles[1] };
_triangles[0] = triangles[0].transform(toLocal);
_triangles[1] = triangles[1].transform(toLocal);
transform.traverse(*this);
if (_wire) {
SGMatrixd toWorld = transform.getToWorldTransform(_time);
_linearVelocity
+= transform.getLinearVelocityAt(_lineSegment.getStart());
_angularVelocity += transform.getAngularVelocity();
_linearVelocity = toWorld.xformVec(_linearVelocity);
_angularVelocity = toWorld.xformVec(_angularVelocity);
_lineSegment = _lineSegment.transform(toWorld);
}
_triangles[0] = triangles[0];
_triangles[1] = triangles[1];
}
virtual void apply(BVHLineGeometry& node)
{
if (node.getType() != BVHLineGeometry::CarrierWire)
return;
SGLineSegmentd lineSegment(node.getLineSegment());
if (!_intersects(lineSegment))
return;
_lineSegment = lineSegment;
_linearVelocity = SGVec3d::zeros();
_angularVelocity = SGVec3d::zeros();
_wire = &node;
}
virtual void apply(BVHStaticGeometry& node)
{ }
virtual void apply(const BVHStaticBinary&, const BVHStaticData&) { }
virtual void apply(const BVHStaticTriangle&, const BVHStaticData&) { }
bool _intersects(const SGSphered& sphere) const
{
if (_wire)
return false;
if (intersects(_triangles[0], sphere))
return true;
if (intersects(_triangles[1], sphere))
return true;
return false;
}
bool _intersects(const SGLineSegmentd& lineSegment) const
{
if (_wire)
return false;
if (intersects(_triangles[0], lineSegment))
return true;
if (intersects(_triangles[1], lineSegment))
return true;
return false;
}
const SGLineSegmentd& getLineSegment() const
{ return _lineSegment; }
const SGVec3d& getLinearVelocity() const
{ return _linearVelocity; }
const SGVec3d& getAngularVelocity() const
{ return _angularVelocity; }
const BVHLineGeometry* getWire() const
{ return _wire; }
private:
SGLineSegmentd _lineSegment;
SGVec3d _linearVelocity;
SGVec3d _angularVelocity;
const BVHLineGeometry* _wire;
SGTriangled _triangles[2];
double _time;
};
bool FGGroundCache::caught_wire(double t, const SGVec3d pt[4])
{
// Get the wire in question
t += cache_time_offset;
WireIntersector wireIntersector(pt, t);
if (_localBvhTree)
_localBvhTree->accept(wireIntersector);
_wire = wireIntersector.getWire();
return (_wire != NULL);
}
class FGGroundCache::WireFinder : public BVHVisitor {
public:
WireFinder(const BVHLineGeometry* wire, const double& t) :
_wire(wire),
_time(t),
_lineSegment(SGVec3d::zeros(), SGVec3d::zeros()),
_linearVelocity(SGVec3d::zeros()),
_angularVelocity(SGVec3d::zeros()),
_haveLineSegment(false)
{ }
virtual void apply(BVHGroup& leaf)
{
if (_haveLineSegment)
return;
leaf.traverse(*this);
}
virtual void apply(BVHPageNode& leaf)
{
if (_haveLineSegment)
return;
leaf.traverse(*this);
}
virtual void apply(BVHTransform& transform)
{
if (_haveLineSegment)
return;
transform.traverse(*this);
if (_haveLineSegment) {
_linearVelocity = transform.vecToWorld(_linearVelocity);
_angularVelocity = transform.vecToWorld(_angularVelocity);
_lineSegment = transform.lineSegmentToWorld(_lineSegment);
}
}
virtual void apply(BVHMotionTransform& transform)
{
if (_haveLineSegment)
return;
transform.traverse(*this);
if (_haveLineSegment) {
SGMatrixd toWorld = transform.getToWorldTransform(_time);
_linearVelocity
+= transform.getLinearVelocityAt(_lineSegment.getStart());
_angularVelocity += transform.getAngularVelocity();
_linearVelocity = toWorld.xformVec(_linearVelocity);
_angularVelocity = toWorld.xformVec(_angularVelocity);
_lineSegment = _lineSegment.transform(toWorld);
}
}
virtual void apply(BVHLineGeometry& node)
{
if (_haveLineSegment)
return;
if (_wire != &node)
return;
if (node.getType() != BVHLineGeometry::CarrierWire)
return;
_lineSegment = SGLineSegmentd(node.getLineSegment());
_linearVelocity = SGVec3d::zeros();
_angularVelocity = SGVec3d::zeros();
_haveLineSegment = true;
}
virtual void apply(BVHStaticGeometry&) { }
virtual void apply(const BVHStaticBinary&, const BVHStaticData&) { }
virtual void apply(const BVHStaticTriangle&, const BVHStaticData&) { }
const SGLineSegmentd& getLineSegment() const
{ return _lineSegment; }
bool getHaveLineSegment() const
{ return _haveLineSegment; }
const SGVec3d& getLinearVelocity() const
{ return _linearVelocity; }
const SGVec3d& getAngularVelocity() const
{ return _angularVelocity; }
private:
const BVHLineGeometry* _wire;
double _time;
SGLineSegmentd _lineSegment;
SGVec3d _linearVelocity;
SGVec3d _angularVelocity;
bool _haveLineSegment;
};
bool FGGroundCache::get_wire_ends(double t, SGVec3d end[2], SGVec3d vel[2])
{
// Fast return if we do not have an active wire.
if (!_wire)
return false;
// Get the wire in question
t += cache_time_offset;
WireFinder wireFinder(_wire, t);
if (_localBvhTree)
_localBvhTree->accept(wireFinder);
if (!wireFinder.getHaveLineSegment())
return false;
// prepare the returns
end[0] = wireFinder.getLineSegment().getStart();
end[1] = wireFinder.getLineSegment().getEnd();
// The linear velocity is the one at the start of the line segment ...
vel[0] = wireFinder.getLinearVelocity();
// ... so the end point has the additional cross product.
vel[1] = wireFinder.getLinearVelocity();
vel[1] += cross(wireFinder.getAngularVelocity(),
wireFinder.getLineSegment().getDirection());
return true;
}
void FGGroundCache::release_wire(void)
{
_wire = 0;
}