2015-05-12 01:02:15 +00:00
|
|
|
#include "actor.hpp"
|
|
|
|
|
2018-05-27 13:05:40 +00:00
|
|
|
/*
|
|
|
|
Start of tes3mp addition
|
|
|
|
|
|
|
|
Include additional headers for multiplayer purposes
|
|
|
|
*/
|
2019-08-19 18:39:33 +00:00
|
|
|
#include <components/openmw-mp/TimedLog.hpp>
|
2018-05-27 13:05:40 +00:00
|
|
|
#include "../mwmp/Main.hpp"
|
|
|
|
#include "../mwmp/Networking.hpp"
|
|
|
|
#include "../mwmp/PlayerList.hpp"
|
|
|
|
/*
|
|
|
|
End of tes3mp addition
|
|
|
|
*/
|
|
|
|
|
2015-05-27 20:32:11 +00:00
|
|
|
#include <BulletCollision/CollisionShapes/btBoxShape.h>
|
2015-05-27 21:09:38 +00:00
|
|
|
#include <BulletCollision/CollisionDispatch/btCollisionWorld.h>
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2015-11-20 20:57:04 +00:00
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
2015-11-16 22:30:10 +00:00
|
|
|
#include <components/resource/bulletshape.hpp>
|
2018-10-22 11:14:25 +00:00
|
|
|
#include <components/debug/debuglog.hpp>
|
2019-02-28 20:03:42 +00:00
|
|
|
#include <components/misc/convert.hpp>
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2015-05-27 20:32:11 +00:00
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
#include "collisiontype.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"
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2020-12-27 22:16:11 +00:00
|
|
|
#include <cmath>
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
namespace MWPhysics
|
|
|
|
{
|
|
|
|
|
|
|
|
|
2021-07-23 15:59:54 +00:00
|
|
|
Actor::Actor(const MWWorld::Ptr& ptr, const Resource::BulletShape* shape, PhysicsTaskScheduler* scheduler, bool canWaterWalk)
|
|
|
|
: mStandingOnPtr(nullptr), mCanWaterWalk(canWaterWalk), mWalkingOnWater(false)
|
2020-11-16 16:37:30 +00:00
|
|
|
, mCollisionObject(nullptr), mMeshTranslation(shape->mCollisionBox.center), mHalfExtents(shape->mCollisionBox.extents)
|
2021-05-01 18:28:17 +00:00
|
|
|
, mVelocity(0,0,0), mStuckFrames(0), mLastStuckPosition{0, 0, 0}
|
2020-10-14 09:32:12 +00:00
|
|
|
, mForce(0.f, 0.f, 0.f), mOnGround(true), mOnSlope(false)
|
2015-05-12 01:02:15 +00:00
|
|
|
, mInternalCollisionMode(true)
|
|
|
|
, mExternalCollisionMode(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)
|
2015-05-12 01:02:15 +00:00
|
|
|
{
|
|
|
|
mPtr = ptr;
|
|
|
|
|
2018-10-22 11:14:25 +00:00
|
|
|
// We can not create actor without collisions - he will fall through the ground.
|
|
|
|
// In this case we should autogenerate collision box based on mesh shape
|
|
|
|
// (NPCs have bodyparts and use a different approach)
|
|
|
|
if (!ptr.getClass().isNpc() && mHalfExtents.length2() == 0.f)
|
|
|
|
{
|
2020-10-14 09:32:12 +00:00
|
|
|
if (shape->mCollisionShape)
|
2018-10-22 11:14:25 +00:00
|
|
|
{
|
|
|
|
btTransform transform;
|
|
|
|
transform.setIdentity();
|
|
|
|
btVector3 min;
|
|
|
|
btVector3 max;
|
|
|
|
|
2020-10-14 09:32:12 +00:00
|
|
|
shape->mCollisionShape->getAabb(transform, min, max);
|
2018-10-22 11:14:25 +00:00
|
|
|
mHalfExtents.x() = (max[0] - min[0])/2.f;
|
|
|
|
mHalfExtents.y() = (max[1] - min[1])/2.f;
|
|
|
|
mHalfExtents.z() = (max[2] - min[2])/2.f;
|
|
|
|
|
|
|
|
mMeshTranslation = osg::Vec3f(0.f, 0.f, mHalfExtents.z());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mHalfExtents.length2() == 0.f)
|
|
|
|
Log(Debug::Error) << "Error: Failed to calculate bounding box for actor \"" << ptr.getCellRef().getRefId() << "\".";
|
|
|
|
}
|
|
|
|
|
2020-12-27 22:16:11 +00:00
|
|
|
mShape.reset(new btBoxShape(Misc::Convert::toBullet(mHalfExtents)));
|
|
|
|
mRotationallyInvariant = (mMeshTranslation.x() == 0.0 && mMeshTranslation.y() == 0.0) && std::fabs(mHalfExtents.x() - mHalfExtents.y()) < 2.2;
|
2017-02-23 21:34:42 +00:00
|
|
|
|
2017-02-10 00:58:27 +00:00
|
|
|
mConvexShape = static_cast<btConvexShape*>(mShape.get());
|
2015-05-12 01:02:15 +00:00
|
|
|
|
2020-12-25 17:20:42 +00:00
|
|
|
mCollisionObject = std::make_unique<btCollisionObject>();
|
2015-05-12 01:02:15 +00:00
|
|
|
mCollisionObject->setCollisionFlags(btCollisionObject::CF_KINEMATIC_OBJECT);
|
|
|
|
mCollisionObject->setActivationState(DISABLE_DEACTIVATION);
|
|
|
|
mCollisionObject->setCollisionShape(mShape.get());
|
2019-02-13 07:30:16 +00:00
|
|
|
mCollisionObject->setUserPointer(this);
|
2015-05-12 01:02:15 +00:00
|
|
|
|
|
|
|
updateScale();
|
2020-12-27 22:16:11 +00:00
|
|
|
|
|
|
|
if(!mRotationallyInvariant)
|
|
|
|
updateRotation();
|
|
|
|
|
2020-12-19 19:25:46 +00:00
|
|
|
updatePosition();
|
2016-12-16 19:22:07 +00:00
|
|
|
addCollisionMask(getCollisionMask());
|
2018-05-27 13:05:40 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
Start of tes3mp addition
|
|
|
|
|
|
|
|
Make it possible to disable collision for players or regular actors from a packet
|
|
|
|
*/
|
|
|
|
mwmp::BaseWorldstate *worldstate = mwmp::Main::get().getNetworking()->getWorldstate();
|
|
|
|
|
|
|
|
if (mwmp::PlayerList::isDedicatedPlayer(ptr))
|
|
|
|
{
|
|
|
|
if (!worldstate->hasPlayerCollision)
|
|
|
|
enableCollisionBody(false);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!worldstate->hasActorCollision)
|
|
|
|
enableCollisionBody(false);
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
End of tes3mp addition
|
|
|
|
*/
|
2021-01-05 05:25:31 +00:00
|
|
|
|
2020-12-18 08:22:02 +00:00
|
|
|
updateCollisionObjectPosition();
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Actor::~Actor()
|
|
|
|
{
|
2020-12-25 17:20:42 +00:00
|
|
|
mTaskScheduler->removeCollisionObject(mCollisionObject.get());
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::enableCollisionMode(bool collision)
|
|
|
|
{
|
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
|
|
|
mInternalCollisionMode.store(collision, std::memory_order_release);
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::enableCollisionBody(bool collision)
|
|
|
|
{
|
|
|
|
if (mExternalCollisionMode != collision)
|
|
|
|
{
|
|
|
|
mExternalCollisionMode = collision;
|
|
|
|
updateCollisionMask();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-16 19:22:07 +00:00
|
|
|
void Actor::addCollisionMask(int collisionMask)
|
|
|
|
{
|
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->addCollisionObject(mCollisionObject.get(), CollisionType_Actor, collisionMask);
|
2016-12-16 19:22:07 +00:00
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
void Actor::updateCollisionMask()
|
|
|
|
{
|
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->setCollisionFilterMask(mCollisionObject.get(), getCollisionMask());
|
2016-12-16 19:22:07 +00:00
|
|
|
}
|
|
|
|
|
2020-10-14 09:32:12 +00:00
|
|
|
int Actor::getCollisionMask() const
|
2016-12-16 19:22:07 +00:00
|
|
|
{
|
2015-05-12 01:02:15 +00:00
|
|
|
int collisionMask = CollisionType_World | CollisionType_HeightMap;
|
|
|
|
if (mExternalCollisionMode)
|
2015-12-18 17:32:42 +00:00
|
|
|
collisionMask |= CollisionType_Actor | CollisionType_Projectile | CollisionType_Door;
|
2015-05-12 01:02:15 +00:00
|
|
|
if (mCanWaterWalk)
|
|
|
|
collisionMask |= CollisionType_Water;
|
2016-12-16 19:22:07 +00:00
|
|
|
return collisionMask;
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
2020-12-19 19:25:46 +00:00
|
|
|
void Actor::updatePosition()
|
2020-11-28 19:45:23 +00:00
|
|
|
{
|
2020-12-19 19:25:46 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
Remove both racy and useless code.
Actor's position can be determined in 3 ways:
1/ as a result of physics simulation
2/ after a script require a relative position change (SetPos, Move)
3/ absolutely set from games mechanics event (teleport) or script
(PositionCell)
In case 1/, RefData::mPosition is updated with the physics simulation result
In case 2/, when RefData::mPosition is updated, physics simulation is informed of the change and update accordingly
In case 3/, when RefData::mPosition is updated, the physics simulation state is reset
In all 3 cases, we don't need to check the RefData::mPosition to get a
correct behaviour.
TSAN reported the following data race:
Read of size 4 at 0x7b50005b75b0 by thread T12 (mutexes: write M656173, write M84859534346343880):
#0 ESM::Position::asVec3() const /build/openmw/openmw/master2/.build/freebsd/TSAN/../../.././components/esm/defs.hpp:55:27 (openmw+0xb809d5)
#1 MWPhysics::Actor::updateWorldPosition() /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwphysics/actor.cpp:131:59 (openmw+0xb809d5)
#2 MWPhysics::Actor::setPosition(osg::Vec3f const&) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwphysics/actor.cpp:177:5 (openmw+0xb809d5)
#3 MWPhysics::PhysicsTaskScheduler::updateActorsPositions() /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwphysics/mtphysics.cpp:524:28 (openmw+0xb91ac0)
#4 MWPhysics::PhysicsTaskScheduler::afterPostStep() /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwphysics/mtphysics.cpp:614:13 (openmw+0xb915e7)
#5 MWPhysics::PhysicsTaskScheduler::worker()::$_5::operator()() const /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwphysics/mtphysics.cpp:498:45 (openmw+0xb915e7)
#6 void Misc::Barrier::wait<MWPhysics::PhysicsTaskScheduler::worker()::$_5>(MWPhysics::PhysicsTaskScheduler::worker()::$_5&&) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../.././components/misc/barrier.hpp:30:21 (openmw+0xb915e7)
#7 MWPhysics::PhysicsTaskScheduler::worker() /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwphysics/mtphysics.cpp:498:31 (openmw+0xb915e7)
#8 MWPhysics::PhysicsTaskScheduler::PhysicsTaskScheduler(float, btCollisionWorld*, MWRender::DebugDrawer*)::$_0::operator()() const /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwphysics/mtphysics.cpp:162:45 (openmw+0xb92630)
#9 decltype(std::__1::forward<MWPhysics::PhysicsTaskScheduler::PhysicsTaskScheduler(float, btCollisionWorld*, MWRender::DebugDrawer*)::$_0>(fp)()) std::__1::__invoke<MWPhysics::PhysicsTaskScheduler::PhysicsTaskScheduler(float, btCollisionWorld*, MWRender::DebugDrawer*)::$_0>(MWPhysics::PhysicsTaskScheduler::PhysicsTaskScheduler(float, btCollisionWorld*, MWRender::DebugDrawer*)::$_0&&) /usr/include/c++/v1/type_traits:3899:1 (openmw+0xb92630)
#10 void std::__1::__thread_execute<std::__1::unique_ptr<std::__1::__thread_struct, std::__1::default_delete<std::__1::__thread_struct> >, MWPhysics::PhysicsTaskScheduler::PhysicsTaskScheduler(float, btCollisionWorld*, MWRender::DebugDrawer*)::$_0>(std::__1::tuple<std::__1::unique_ptr<std::__1::__thread_struct, std::__1::default_delete<std::__1::__thread_struct> >, MWPhysics::PhysicsTaskScheduler::PhysicsTaskScheduler(float, btCollisionWorld*, MWRender::DebugDrawer*)::$_0>&, std::__1::__tuple_indices<>) /usr/include/c++/v1/thread:280:5 (openmw+0xb92630)
#11 void* std::__1::__thread_proxy<std::__1::tuple<std::__1::unique_ptr<std::__1::__thread_struct, std::__1::default_delete<std::__1::__thread_struct> >, MWPhysics::PhysicsTaskScheduler::PhysicsTaskScheduler(float, btCollisionWorld*, MWRender::DebugDrawer*)::$_0> >(void*) /usr/include/c++/v1/thread:291:5 (openmw+0xb92630)
Previous write of size 8 at 0x7b50005b75b0 by main thread:
#0 memcpy /wrkdirs/usr/ports/devel/llvm-devel/work-default/llvm-project-3f6753efe1990a928ed120bd907940a9fb3e2fc3/compiler-rt/lib/tsan/../sanitizer_common/sanitizer_common_interceptors.inc:827:5 (openmw+0x55a057)
#1 MWWorld::RefData::setPosition(ESM::Position const&) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwworld/refdata.cpp:216:19 (openmw+0xa3de1c)
#2 MWWorld::World::moveObject(MWWorld::Ptr const&, MWWorld::CellStore*, float, float, float, bool) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwworld/worldimp.cpp:1130:26 (openmw+0xa57300)
#3 MWWorld::World::moveObject(MWWorld::Ptr const&, float, float, float, bool, bool) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwworld/worldimp.cpp:1253:16 (openmw+0xa580c8)
#4 MWWorld::World::doPhysics(float, unsigned long long, unsigned int, osg::Stats&) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwworld/worldimp.cpp:1530:17 (openmw+0xa5af8f)
#5 MWWorld::World::updatePhysics(float, bool, unsigned long long, unsigned int, osg::Stats&) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/mwworld/worldimp.cpp:1862:13 (openmw+0xa61a7c)
#6 OMW::Engine::frame(float) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/engine.cpp:333:42 (openmw+0xcce9e7)
#7 OMW::Engine::go() /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/engine.cpp:935:14 (openmw+0xcd86ed)
#8 runApplication(int, char**) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/main.cpp:296:17 (openmw+0xcbffac)
#9 wrapApplication(int (*)(int, char**), int, char**, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&) /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../components/debug/debugging.cpp:205:15 (openmw+0x1335442)
#10 main /build/openmw/openmw/master2/.build/freebsd/TSAN/../../../apps/openmw/main.cpp:308:12 (openmw+0xcc008a)
:wqa
2021-05-20 03:53:26 +00:00
|
|
|
const auto worldPosition = mPtr.getRefData().getPosition().asVec3();
|
|
|
|
mPreviousPosition = worldPosition;
|
|
|
|
mPosition = worldPosition;
|
|
|
|
mSimulationPosition = worldPosition;
|
2021-04-15 20:32:42 +00:00
|
|
|
mPositionOffset = osg::Vec3f();
|
2020-12-19 19:25:46 +00:00
|
|
|
mStandingOnPtr = nullptr;
|
2021-05-01 12:13:17 +00:00
|
|
|
mSkipCollisions = true;
|
2021-07-16 07:50:26 +00:00
|
|
|
mSkipSimulation = true;
|
2020-11-28 19:45:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::setSimulationPosition(const osg::Vec3f& position)
|
2020-11-01 16:14:59 +00:00
|
|
|
{
|
2021-07-16 07:50:26 +00:00
|
|
|
if (!std::exchange(mSkipSimulation, false))
|
|
|
|
mSimulationPosition = position;
|
2016-02-13 01:56:41 +00:00
|
|
|
}
|
|
|
|
|
2020-11-28 19:45:23 +00:00
|
|
|
osg::Vec3f Actor::getSimulationPosition() const
|
2016-02-13 01:56:41 +00:00
|
|
|
{
|
2020-11-28 19:45:23 +00:00
|
|
|
return mSimulationPosition;
|
2016-02-13 01:56:41 +00:00
|
|
|
}
|
|
|
|
|
2020-12-27 22:16:11 +00:00
|
|
|
osg::Vec3f Actor::getScaledMeshTranslation() const
|
2016-02-13 01:56:41 +00:00
|
|
|
{
|
2020-12-27 22:16:11 +00:00
|
|
|
return mRotation * osg::componentMultiply(mMeshTranslation, mScale);
|
|
|
|
}
|
|
|
|
|
2020-12-19 19:25:46 +00:00
|
|
|
void Actor::updateCollisionObjectPosition()
|
2016-02-13 01:56:41 +00:00
|
|
|
{
|
2020-12-19 19:25:46 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
2020-11-01 16:14:59 +00:00
|
|
|
mShape->setLocalScaling(Misc::Convert::toBullet(mScale));
|
2021-07-17 06:54:20 +00:00
|
|
|
osg::Vec3f newPosition = getScaledMeshTranslation() + mPosition;
|
|
|
|
|
|
|
|
auto& trans = mCollisionObject->getWorldTransform();
|
|
|
|
trans.setOrigin(Misc::Convert::toBullet(newPosition));
|
|
|
|
trans.setRotation(Misc::Convert::toBullet(mRotation));
|
|
|
|
mCollisionObject->setWorldTransform(trans);
|
|
|
|
|
2020-12-18 07:18:07 +00:00
|
|
|
mWorldPositionChanged = false;
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
2016-02-13 01:56:41 +00:00
|
|
|
osg::Vec3f Actor::getCollisionObjectPosition() const
|
2015-11-03 17:15:47 +00:00
|
|
|
{
|
2020-11-01 16:14:59 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
2021-07-17 06:54:20 +00:00
|
|
|
return getScaledMeshTranslation() + mPosition;
|
2015-11-03 17:15:47 +00:00
|
|
|
}
|
|
|
|
|
2020-12-18 07:18:07 +00:00
|
|
|
bool Actor::setPosition(const osg::Vec3f& position)
|
2016-02-13 01:56:41 +00:00
|
|
|
{
|
2020-11-28 19:45:23 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
2021-03-28 19:19:14 +00:00
|
|
|
applyOffsetChange();
|
2021-05-01 12:13:17 +00:00
|
|
|
bool hasChanged = mPosition != position || mWorldPositionChanged;
|
2021-03-28 19:19:14 +00:00
|
|
|
mPreviousPosition = mPosition;
|
|
|
|
mPosition = position;
|
2020-12-18 07:18:07 +00:00
|
|
|
return hasChanged;
|
2020-11-01 16:14:59 +00:00
|
|
|
}
|
2016-02-13 01:56:41 +00:00
|
|
|
|
2021-05-01 12:13:17 +00:00
|
|
|
void Actor::adjustPosition(const osg::Vec3f& offset, bool ignoreCollisions)
|
2020-11-01 16:14:59 +00:00
|
|
|
{
|
2020-11-28 19:45:23 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
2020-12-18 07:18:07 +00:00
|
|
|
mPositionOffset += offset;
|
2021-05-01 12:13:17 +00:00
|
|
|
mSkipCollisions = mSkipCollisions || ignoreCollisions;
|
2016-02-13 01:56:41 +00:00
|
|
|
}
|
|
|
|
|
2021-03-13 08:52:05 +00:00
|
|
|
void Actor::applyOffsetChange()
|
|
|
|
{
|
|
|
|
if (mPositionOffset.length() == 0)
|
|
|
|
return;
|
|
|
|
mPosition += mPositionOffset;
|
|
|
|
mPreviousPosition += mPositionOffset;
|
2021-03-28 19:19:14 +00:00
|
|
|
mSimulationPosition += mPositionOffset;
|
2021-03-13 08:52:05 +00:00
|
|
|
mPositionOffset = osg::Vec3f();
|
|
|
|
mWorldPositionChanged = true;
|
|
|
|
}
|
|
|
|
|
2016-02-13 01:56:41 +00:00
|
|
|
osg::Vec3f Actor::getPosition() const
|
|
|
|
{
|
|
|
|
return mPosition;
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::Vec3f Actor::getPreviousPosition() const
|
|
|
|
{
|
|
|
|
return mPreviousPosition;
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
void Actor::updateRotation ()
|
|
|
|
{
|
2020-11-01 16:14:59 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
2015-05-12 14:49:21 +00:00
|
|
|
mRotation = mPtr.getRefData().getBaseNode()->getAttitude();
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
2017-02-23 21:34:42 +00:00
|
|
|
bool Actor::isRotationallyInvariant() const
|
|
|
|
{
|
|
|
|
return mRotationallyInvariant;
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
void Actor::updateScale()
|
|
|
|
{
|
2020-11-01 16:14:59 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
2015-05-12 01:02:15 +00:00
|
|
|
float scale = mPtr.getCellRef().getScale();
|
|
|
|
osg::Vec3f scaleVec(scale,scale,scale);
|
|
|
|
|
2015-11-01 20:45:58 +00:00
|
|
|
mPtr.getClass().adjustScale(mPtr, scaleVec, false);
|
2015-05-12 01:02:15 +00:00
|
|
|
mScale = scaleVec;
|
|
|
|
|
2015-11-01 20:45:58 +00:00
|
|
|
scaleVec = osg::Vec3f(scale,scale,scale);
|
|
|
|
mPtr.getClass().adjustScale(mPtr, scaleVec, true);
|
|
|
|
mRenderingScale = scaleVec;
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
osg::Vec3f Actor::getHalfExtents() const
|
|
|
|
{
|
2020-11-01 16:14:59 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
2015-05-12 01:02:15 +00:00
|
|
|
return osg::componentMultiply(mHalfExtents, mScale);
|
|
|
|
}
|
|
|
|
|
2019-03-03 11:45:36 +00:00
|
|
|
osg::Vec3f Actor::getOriginalHalfExtents() const
|
|
|
|
{
|
|
|
|
return mHalfExtents;
|
|
|
|
}
|
|
|
|
|
2015-11-01 20:45:58 +00:00
|
|
|
osg::Vec3f Actor::getRenderingHalfExtents() const
|
|
|
|
{
|
2020-11-01 16:14:59 +00:00
|
|
|
std::scoped_lock lock(mPositionMutex);
|
2015-11-01 20:45:58 +00:00
|
|
|
return osg::componentMultiply(mHalfExtents, mRenderingScale);
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
void Actor::setInertialForce(const osg::Vec3f &force)
|
|
|
|
{
|
|
|
|
mForce = force;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::setOnGround(bool grounded)
|
|
|
|
{
|
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
|
|
|
mOnGround.store(grounded, std::memory_order_release);
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
2017-02-06 03:46:44 +00:00
|
|
|
void Actor::setOnSlope(bool slope)
|
|
|
|
{
|
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
|
|
|
mOnSlope.store(slope, std::memory_order_release);
|
2017-02-06 03:46:44 +00:00
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
bool Actor::isWalkingOnWater() 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
|
|
|
return mWalkingOnWater.load(std::memory_order_acquire);
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::setWalkingOnWater(bool walkingOnWater)
|
|
|
|
{
|
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
|
|
|
mWalkingOnWater.store(walkingOnWater, std::memory_order_release);
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::setCanWaterWalk(bool waterWalk)
|
|
|
|
{
|
|
|
|
if (waterWalk != mCanWaterWalk)
|
|
|
|
{
|
|
|
|
mCanWaterWalk = waterWalk;
|
|
|
|
updateCollisionMask();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-07 17:30:52 +00:00
|
|
|
MWWorld::Ptr Actor::getStandingOnPtr() const
|
|
|
|
{
|
|
|
|
std::scoped_lock lock(mPositionMutex);
|
|
|
|
return mStandingOnPtr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::setStandingOnPtr(const MWWorld::Ptr& ptr)
|
|
|
|
{
|
|
|
|
std::scoped_lock lock(mPositionMutex);
|
|
|
|
mStandingOnPtr = ptr;
|
|
|
|
}
|
|
|
|
|
2021-05-01 12:13:17 +00:00
|
|
|
bool Actor::skipCollisions()
|
|
|
|
{
|
|
|
|
return std::exchange(mSkipCollisions, false);
|
|
|
|
}
|
|
|
|
|
2021-05-01 18:28:17 +00:00
|
|
|
void Actor::setVelocity(osg::Vec3f velocity)
|
|
|
|
{
|
|
|
|
mVelocity = velocity;
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::Vec3f Actor::velocity()
|
|
|
|
{
|
|
|
|
return std::exchange(mVelocity, osg::Vec3f());
|
|
|
|
}
|
|
|
|
|
2015-05-12 01:02:15 +00:00
|
|
|
}
|