2018-03-18 11:32:45 +00:00
|
|
|
#include "object.hpp"
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
#include "mtphysics.hpp"
|
2018-03-18 11:32:45 +00:00
|
|
|
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
#include <components/nifosg/particle.hpp>
|
|
|
|
#include <components/resource/bulletshape.hpp>
|
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
2019-02-28 20:03:42 +00:00
|
|
|
#include <components/misc/convert.hpp>
|
2018-03-18 11:32:45 +00:00
|
|
|
|
|
|
|
#include <BulletCollision/CollisionShapes/btCompoundShape.h>
|
|
|
|
#include <BulletCollision/CollisionDispatch/btCollisionObject.h>
|
|
|
|
|
|
|
|
#include <LinearMath/btTransform.h>
|
|
|
|
|
|
|
|
namespace MWPhysics
|
|
|
|
{
|
2021-01-29 12:51:13 +00:00
|
|
|
Object::Object(const MWWorld::Ptr& ptr, osg::ref_ptr<Resource::BulletShapeInstance> shapeInstance, int collisionType, PhysicsTaskScheduler* scheduler)
|
2018-03-18 11:32:45 +00:00
|
|
|
: mShapeInstance(shapeInstance)
|
|
|
|
, mSolid(true)
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
, mTaskScheduler(scheduler)
|
2018-03-18 11:32:45 +00:00
|
|
|
{
|
|
|
|
mPtr = ptr;
|
|
|
|
|
2020-12-25 17:20:42 +00:00
|
|
|
mCollisionObject = std::make_unique<btCollisionObject>();
|
2018-03-18 11:32:45 +00:00
|
|
|
mCollisionObject->setCollisionShape(shapeInstance->getCollisionShape());
|
|
|
|
|
2019-02-13 07:30:16 +00:00
|
|
|
mCollisionObject->setUserPointer(this);
|
2018-03-18 11:32:45 +00:00
|
|
|
|
|
|
|
setScale(ptr.getCellRef().getScale());
|
2021-07-17 06:55:39 +00:00
|
|
|
setRotation(ptr.getRefData().getBaseNode()->getAttitude());
|
|
|
|
updatePosition();
|
2020-10-14 13:55:15 +00:00
|
|
|
commitPositionChange();
|
2020-12-25 17:20:42 +00:00
|
|
|
|
|
|
|
mTaskScheduler->addCollisionObject(mCollisionObject.get(), collisionType, CollisionType_Actor|CollisionType_HeightMap|CollisionType_Projectile);
|
2018-03-18 11:32:45 +00:00
|
|
|
}
|
|
|
|
|
2020-10-14 09:32:12 +00:00
|
|
|
Object::~Object()
|
|
|
|
{
|
2020-12-25 17:20:42 +00:00
|
|
|
mTaskScheduler->removeCollisionObject(mCollisionObject.get());
|
2020-10-14 09:32:12 +00:00
|
|
|
}
|
|
|
|
|
2018-03-18 11:32:45 +00:00
|
|
|
const Resource::BulletShapeInstance* Object::getShapeInstance() const
|
|
|
|
{
|
|
|
|
return mShapeInstance.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Object::setScale(float scale)
|
|
|
|
{
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
std::unique_lock<std::mutex> lock(mPositionMutex);
|
2020-10-14 13:55:15 +00:00
|
|
|
mScale = { scale,scale,scale };
|
|
|
|
mScaleUpdatePending = true;
|
2018-03-18 11:32:45 +00:00
|
|
|
}
|
|
|
|
|
2021-07-17 06:55:39 +00:00
|
|
|
void Object::setRotation(const osg::Quat& quat)
|
2018-03-18 11:32:45 +00:00
|
|
|
{
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
std::unique_lock<std::mutex> lock(mPositionMutex);
|
2021-07-17 06:55:39 +00:00
|
|
|
mRotation = quat;
|
2020-10-14 13:55:15 +00:00
|
|
|
mTransformUpdatePending = true;
|
2018-03-18 11:32:45 +00:00
|
|
|
}
|
|
|
|
|
2021-07-17 06:55:39 +00:00
|
|
|
void Object::updatePosition()
|
2018-03-18 11:32:45 +00:00
|
|
|
{
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
std::unique_lock<std::mutex> lock(mPositionMutex);
|
2021-07-17 06:55:39 +00:00
|
|
|
mPosition = mPtr.getRefData().getPosition().asVec3();
|
2020-10-14 13:55:15 +00:00
|
|
|
mTransformUpdatePending = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Object::commitPositionChange()
|
|
|
|
{
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
std::unique_lock<std::mutex> lock(mPositionMutex);
|
2020-10-14 13:55:15 +00:00
|
|
|
if (mScaleUpdatePending)
|
|
|
|
{
|
|
|
|
mShapeInstance->setLocalScaling(mScale);
|
|
|
|
mScaleUpdatePending = false;
|
|
|
|
}
|
|
|
|
if (mTransformUpdatePending)
|
|
|
|
{
|
2021-07-17 06:55:39 +00:00
|
|
|
btTransform trans;
|
|
|
|
trans.setOrigin(Misc::Convert::toBullet(mPosition));
|
|
|
|
trans.setRotation(Misc::Convert::toBullet(mRotation));
|
|
|
|
mCollisionObject->setWorldTransform(trans);
|
2020-10-14 13:55:15 +00:00
|
|
|
mTransformUpdatePending = false;
|
|
|
|
}
|
2018-03-18 11:32:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
btCollisionObject* Object::getCollisionObject()
|
|
|
|
{
|
|
|
|
return mCollisionObject.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
const btCollisionObject* Object::getCollisionObject() const
|
|
|
|
{
|
|
|
|
return mCollisionObject.get();
|
|
|
|
}
|
|
|
|
|
2020-10-14 13:55:15 +00:00
|
|
|
btTransform Object::getTransform() const
|
|
|
|
{
|
Process movement queue in one or several background threads
Before movement calculation, the main thread prepare a
vector of ActorFrameData, which contains all data necessary to perform
the simulation, and feed it to the solver. At the same time it fetches
the result from the previous background simulation, which in turn is
used by the game mechanics.
Other functions of the physics system (weapon hit for instance)
interrupt the background simulation, with some exceptions described
below.
The number of threads is controlled by the numeric setting
[Physics]
async num threads
In case 'async num threads' > 1 and Bullet doesn't support multiple threads,
1 async thread will be used. 0 means synchronous solver.
Additional settings (will be silently switched off if async num threads = 0)
[Physics]
defer aabb update
Update AABBs of actors and objects in the background thread(s). It is not an especially
costly operation, but it needs exclusive access to the collision world, which blocks
other operations. Since AABB needs to be updated for collision detection, one can queue
them to defer update before start of the movement solver. Extensive tests on as much
as one installation (mine) show no drawback having that switched on.
[Physics]
lineofsight keep inactive cache
Control for how long (how many frames) the line of sight (LOS) request will be kept updated.
When a request for LOS is made for the first time, the background threads are stopped to
service it. From now on, the LOS will be refreshed preemptively as part of the background
routine until it is not required for lineofsight keep inactive cache frames. This mean
that subsequent request will not interrupt the background computation.
2020-10-15 04:11:44 +00:00
|
|
|
std::unique_lock<std::mutex> lock(mPositionMutex);
|
2021-07-17 06:55:39 +00:00
|
|
|
btTransform trans;
|
|
|
|
trans.setOrigin(Misc::Convert::toBullet(mPosition));
|
|
|
|
trans.setRotation(Misc::Convert::toBullet(mRotation));
|
|
|
|
return trans;
|
2020-10-14 13:55:15 +00:00
|
|
|
}
|
|
|
|
|
2018-03-18 11:32:45 +00:00
|
|
|
bool Object::isSolid() const
|
|
|
|
{
|
|
|
|
return mSolid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Object::setSolid(bool solid)
|
|
|
|
{
|
|
|
|
mSolid = solid;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Object::isAnimated() const
|
|
|
|
{
|
|
|
|
return !mShapeInstance->mAnimatedShapes.empty();
|
|
|
|
}
|
|
|
|
|
2020-10-14 13:55:15 +00:00
|
|
|
bool Object::animateCollisionShapes()
|
2018-03-18 11:32:45 +00:00
|
|
|
{
|
|
|
|
if (mShapeInstance->mAnimatedShapes.empty())
|
2020-10-14 13:55:15 +00:00
|
|
|
return false;
|
2018-03-18 11:32:45 +00:00
|
|
|
|
|
|
|
assert (mShapeInstance->getCollisionShape()->isCompound());
|
|
|
|
|
|
|
|
btCompoundShape* compound = static_cast<btCompoundShape*>(mShapeInstance->getCollisionShape());
|
2021-05-16 07:40:41 +00:00
|
|
|
for (const auto& [recIndex, shapeIndex] : mShapeInstance->mAnimatedShapes)
|
2018-03-18 11:32:45 +00:00
|
|
|
{
|
2020-06-26 10:22:00 +00:00
|
|
|
auto nodePathFound = mRecIndexToNodePath.find(recIndex);
|
2018-03-18 11:32:45 +00:00
|
|
|
if (nodePathFound == mRecIndexToNodePath.end())
|
|
|
|
{
|
|
|
|
NifOsg::FindGroupByRecIndex visitor(recIndex);
|
|
|
|
mPtr.getRefData().getBaseNode()->accept(visitor);
|
|
|
|
if (!visitor.mFound)
|
|
|
|
{
|
|
|
|
Log(Debug::Warning) << "Warning: animateCollisionShapes can't find node " << recIndex << " for " << mPtr.getCellRef().getRefId();
|
|
|
|
|
|
|
|
// Remove nonexistent nodes from animated shapes map and early out
|
|
|
|
mShapeInstance->mAnimatedShapes.erase(recIndex);
|
2020-10-14 13:55:15 +00:00
|
|
|
return false;
|
2018-03-18 11:32:45 +00:00
|
|
|
}
|
|
|
|
osg::NodePath nodePath = visitor.mFoundPath;
|
|
|
|
nodePath.erase(nodePath.begin());
|
2020-06-26 10:22:00 +00:00
|
|
|
nodePathFound = mRecIndexToNodePath.emplace(recIndex, nodePath).first;
|
2018-03-18 11:32:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
osg::NodePath& nodePath = nodePathFound->second;
|
|
|
|
osg::Matrixf matrix = osg::computeLocalToWorld(nodePath);
|
|
|
|
matrix.orthoNormalize(matrix);
|
|
|
|
|
|
|
|
btTransform transform;
|
2019-02-28 20:03:42 +00:00
|
|
|
transform.setOrigin(Misc::Convert::toBullet(matrix.getTrans()) * compound->getLocalScaling());
|
2018-03-18 11:32:45 +00:00
|
|
|
for (int i=0; i<3; ++i)
|
|
|
|
for (int j=0; j<3; ++j)
|
|
|
|
transform.getBasis()[i][j] = matrix(j,i); // NB column/row major difference
|
|
|
|
|
|
|
|
// Note: we can not apply scaling here for now since we treat scaled shapes
|
|
|
|
// as new shapes (btScaledBvhTriangleMeshShape) with 1.0 scale for now
|
|
|
|
if (!(transform == compound->getChildTransform(shapeIndex)))
|
|
|
|
compound->updateChildTransform(shapeIndex, transform);
|
|
|
|
}
|
2020-10-14 13:55:15 +00:00
|
|
|
return true;
|
2018-03-18 11:32:45 +00:00
|
|
|
}
|
|
|
|
}
|