aboutsummaryrefslogtreecommitdiff
path: root/PhysX_3.4/Source/SimulationController/src/ScShapeInteraction.cpp
diff options
context:
space:
mode:
authorgit perforce import user <a@b>2016-10-25 12:29:14 -0600
committerSheikh Dawood Abdul Ajees <Sheikh Dawood Abdul Ajees>2016-10-25 18:56:37 -0500
commit3dfe2108cfab31ba3ee5527e217d0d8e99a51162 (patch)
treefa6485c169e50d7415a651bf838f5bcd0fd3bfbd /PhysX_3.4/Source/SimulationController/src/ScShapeInteraction.cpp
downloadphysx-3.4-3dfe2108cfab31ba3ee5527e217d0d8e99a51162.tar.xz
physx-3.4-3dfe2108cfab31ba3ee5527e217d0d8e99a51162.zip
Initial commit:
PhysX 3.4.0 Update @ 21294896 APEX 1.4.0 Update @ 21275617 [CL 21300167]
Diffstat (limited to 'PhysX_3.4/Source/SimulationController/src/ScShapeInteraction.cpp')
-rw-r--r--PhysX_3.4/Source/SimulationController/src/ScShapeInteraction.cpp1159
1 files changed, 1159 insertions, 0 deletions
diff --git a/PhysX_3.4/Source/SimulationController/src/ScShapeInteraction.cpp b/PhysX_3.4/Source/SimulationController/src/ScShapeInteraction.cpp
new file mode 100644
index 00000000..2235eb9d
--- /dev/null
+++ b/PhysX_3.4/Source/SimulationController/src/ScShapeInteraction.cpp
@@ -0,0 +1,1159 @@
+// This code contains NVIDIA Confidential Information and is disclosed to you
+// under a form of NVIDIA software license agreement provided separately to you.
+//
+// Notice
+// NVIDIA Corporation and its licensors retain all intellectual property and
+// proprietary rights in and to this software and related documentation and
+// any modifications thereto. Any use, reproduction, disclosure, or
+// distribution of this software and related documentation without an express
+// license agreement from NVIDIA Corporation is strictly prohibited.
+//
+// ALL NVIDIA DESIGN SPECIFICATIONS, CODE ARE PROVIDED "AS IS.". NVIDIA MAKES
+// NO WARRANTIES, EXPRESSED, IMPLIED, STATUTORY, OR OTHERWISE WITH RESPECT TO
+// THE MATERIALS, AND EXPRESSLY DISCLAIMS ALL IMPLIED WARRANTIES OF NONINFRINGEMENT,
+// MERCHANTABILITY, AND FITNESS FOR A PARTICULAR PURPOSE.
+//
+// Information and code furnished is believed to be accurate and reliable.
+// However, NVIDIA Corporation assumes no responsibility for the consequences of use of such
+// information or for any infringement of patents or other rights of third parties that may
+// result from its use. No license is granted by implication or otherwise under any patent
+// or patent rights of NVIDIA Corporation. Details are subject to change without notice.
+// This code supersedes and replaces all information previously supplied.
+// NVIDIA Corporation products are not authorized for use as critical
+// components in life support devices or systems without express written approval of
+// NVIDIA Corporation.
+//
+// Copyright (c) 2008-2016 NVIDIA Corporation. All rights reserved.
+// Copyright (c) 2004-2008 AGEIA Technologies, Inc. All rights reserved.
+// Copyright (c) 2001-2004 NovodeX AG. All rights reserved.
+
+#include "ScShapeInteraction.h"
+#include "ScPhysics.h"
+#include "PxsContext.h"
+#include "PxsMaterialCombiner.h"
+#include "GuTriangleMesh.h"
+#include "ScStaticSim.h"
+#include "PxvManager.h"
+#include "PxsSimpleIslandManager.h"
+#include "PxvNphaseImplementationContext.h"
+
+using namespace physx;
+
+
+Sc::ShapeInteraction::ShapeInteraction(ShapeSim& s1, ShapeSim& s2, ActorPair* aPair, PxPairFlags pairFlags, PxsContactManager* contactManager) :
+ RbElementInteraction (s1, s2, InteractionType::eOVERLAP, InteractionFlag::eRB_ELEMENT|InteractionFlag::eFILTERABLE),
+ mContactReportStamp (PX_INVALID_U32),
+ mFlags (0),
+ mActorPair (aPair),
+ mReportPairIndex (INVALID_REPORT_PAIR_ID),
+ mManager (NULL),
+ mEdgeIndex (IG_INVALID_EDGE),
+ mReportStreamIndex (0)
+{
+ // The PxPairFlags get stored in the SipFlag, make sure any changes get noticed
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eSOLVE_CONTACT == (1<<0));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eMODIFY_CONTACTS == (1<<1));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eNOTIFY_TOUCH_FOUND == (1<<2));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eNOTIFY_TOUCH_PERSISTS == (1<<3));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eNOTIFY_TOUCH_LOST == (1<<4));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eNOTIFY_TOUCH_CCD == (1<<5));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eNOTIFY_THRESHOLD_FORCE_FOUND == (1<<6));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eNOTIFY_THRESHOLD_FORCE_PERSISTS == (1<<7));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eNOTIFY_THRESHOLD_FORCE_LOST == (1<<8));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eNOTIFY_CONTACT_POINTS == (1<<9));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eDETECT_DISCRETE_CONTACT == (1<<10));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eDETECT_CCD_CONTACT == (1<<11));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::ePRE_SOLVER_VELOCITY == (1<<12));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::ePOST_SOLVER_VELOCITY == (1<<13));
+ PX_COMPILE_TIME_ASSERT(PxPairFlag::eCONTACT_EVENT_POSE == (1<<14));
+ PX_COMPILE_TIME_ASSERT((PAIR_FLAGS_MASK & PxPairFlag::eSOLVE_CONTACT) == PxPairFlag::eSOLVE_CONTACT);
+ PX_COMPILE_TIME_ASSERT((PxPairFlag::eSOLVE_CONTACT | PAIR_FLAGS_MASK) == PAIR_FLAGS_MASK);
+ PX_COMPILE_TIME_ASSERT((PAIR_FLAGS_MASK & PxPairFlag::eCONTACT_EVENT_POSE) == PxPairFlag::eCONTACT_EVENT_POSE);
+ PX_COMPILE_TIME_ASSERT((PxPairFlag::eCONTACT_EVENT_POSE | PAIR_FLAGS_MASK) == PAIR_FLAGS_MASK);
+
+ setPairFlags(pairFlags);
+
+ //Add a fresh edge to the island manager.
+ Scene& scene = getScene();
+ Sc::BodySim* bs0 = getShape0().getBodySim();
+ Sc::BodySim* bs1 = getShape1().getBodySim();
+
+ PX_ASSERT(bs0); // The shapes have to be sorted such that the first shape belongs to a dynamic
+
+ updateFlags(scene, bs0, bs1, pairFlags);
+
+
+ if (contactManager == NULL)
+ {
+ IG::NodeIndex indexA, indexB;
+ //if(bs0) // the first shape always belongs to a dynamic body (we assert for this above)
+ {
+ indexA = bs0->getNodeIndex();
+ bs0->registerCountedInteraction();
+ }
+ if (bs1)
+ {
+ indexB = bs1->getNodeIndex();
+ bs1->registerCountedInteraction();
+ }
+
+ IG::SimpleIslandManager* simpleIslandManager = scene.getSimpleIslandManager();
+
+
+ mEdgeIndex = simpleIslandManager->addContactManager(NULL, indexA, indexB, this);
+
+ bool active = registerInActors(contactManager);
+ scene.registerInteraction(this, active); // this will call onActivate() on the interaction
+
+ }
+ else
+ {
+ onActivate(contactManager);
+ }
+
+ PX_ASSERT((&getShape0()) && (&getShape1()));
+
+ if(aPair)
+ aPair->incRefCount();
+}
+
+
+Sc::ShapeInteraction::~ShapeInteraction()
+{
+ Sc::BodySim* body0 = getShape0().getBodySim();
+ Sc::BodySim* body1 = getShape1().getBodySim();
+ PX_ASSERT(body0); // the first shape always belongs to a dynamic body
+
+ body0->unregisterCountedInteraction();
+ if (body1)
+ body1->unregisterCountedInteraction();
+
+ if (mManager)
+ {
+ destroyManager();
+ }
+
+ if (mEdgeIndex != IG_INVALID_EDGE)
+ {
+ Scene& scene = getScene();
+
+ scene.getSimpleIslandManager()->removeConnection(mEdgeIndex);
+ mEdgeIndex = IG_INVALID_EDGE;
+
+ scene.unregisterInteraction(this);
+ scene.getNPhaseCore()->unregisterInteraction(this);
+ }
+
+ // This will remove the interaction from the actors list, which will prevent
+ // update calls to this actor because of Body::wakeUp below.
+ unregisterFromActors();
+
+ if (mReportPairIndex != INVALID_REPORT_PAIR_ID)
+ {
+ removeFromReportPairList();
+ }
+}
+
+void Sc::ShapeInteraction::clearIslandGenData()
+{
+ if (mEdgeIndex != IG_INVALID_EDGE)
+ {
+ Scene& scene = getScene();
+ scene.getSimpleIslandManager()->removeConnection(mEdgeIndex);
+ mEdgeIndex = IG_INVALID_EDGE;
+ }
+}
+
+
+void Sc::ShapeInteraction::visualize(Cm::RenderOutput& out, PxsContactManagerOutputIterator& outputs)
+{
+ if (mManager) // sleeping pairs have no contact points -> do not visualize
+ {
+ Scene& scene = getScene();
+ PxReal scale = scene.getVisualizationScale();
+
+ size_t ptrActor0 = reinterpret_cast<size_t>(&getShape0().getRbSim());
+ size_t ptrActor1 = reinterpret_cast<size_t>(&getShape1().getRbSim());
+ const PxReal flipNormal = (ptrActor0 < ptrActor1) ? 1.0f : -1.0f;
+
+ PxU32 offset;
+ PxU32 nextOffset = 0;
+ do
+ {
+ const void* contactPatches;
+ const void* contactPoints;
+ PxU32 contactDataSize;
+ PxU32 contactPointCount;
+ PxU32 contactPatchCount;
+ const PxReal* impulses;
+
+ offset = nextOffset;
+ nextOffset = getContactPointData(contactPatches, contactPoints, contactDataSize, contactPointCount, contactPatchCount, impulses, offset, outputs);
+
+ const PxReal param_contactForce = scene.getVisualizationParameter(PxVisualizationParameter::eCONTACT_FORCE);
+ const PxReal param_contactNormal = scene.getVisualizationParameter(PxVisualizationParameter::eCONTACT_NORMAL);
+ const PxReal param_contactError = scene.getVisualizationParameter(PxVisualizationParameter::eCONTACT_ERROR);
+ const PxReal param_contactPoint = scene.getVisualizationParameter(PxVisualizationParameter::eCONTACT_POINT);
+
+ const PxU32* faceIndices = reinterpret_cast<const PxU32*>(impulses + contactPointCount);
+ PxContactStreamIterator iter(reinterpret_cast<const PxU8*>(contactPatches), reinterpret_cast<const PxU8*>(contactPoints), faceIndices, contactPatchCount, contactPointCount);
+
+ PxU32 i = 0;
+ while(iter.hasNextPatch())
+ {
+ iter.nextPatch();
+ while(iter.hasNextContact())
+ {
+ iter.nextContact();
+
+ PxReal length = 0;
+ PxU32 color = 0;
+
+ if ((param_contactForce != 0.0f) && impulses)
+ {
+ length = scale * param_contactForce * impulses[i];
+ color = 0xff0000;
+ }
+ else if (param_contactNormal != 0.0f)
+ {
+ length = scale * param_contactNormal;
+ color = 0x0000ff;
+ }
+ else if (param_contactError != 0.0f)
+ {
+ length = PxAbs(scale * param_contactError * iter.getSeparation());
+ color = 0xffff00;
+ }
+
+ if (length != 0)
+ out << Cm::RenderOutput::LINES << color << iter.getContactPoint() << iter.getContactPoint() + iter.getContactNormal() * length * flipNormal;
+
+ if (param_contactPoint != 0)
+ {
+ PxReal s = scale * 0.1f;
+ PxVec3 point = iter.getContactPoint();
+
+ if(0) //temp debug to see identical contacts
+ point.x += scale * 0.01f * (contactPointCount - i + 1);
+
+ out << Cm::RenderOutput::LINES << PxU32(PxDebugColor::eARGB_RED);
+ out << point + PxVec3(-s,0,0) << point + PxVec3(s,0,0);
+ out << point + PxVec3(0,-s,0) << point + PxVec3(0,s,0);
+ out << point + PxVec3(0,0,-s) << point + PxVec3(0,0,s);
+
+ }
+ }
+ }
+ }
+ while(nextOffset != offset);
+ }
+}
+
+
+PX_FORCE_INLINE void Sc::ShapeInteraction::processReportPairOnActivate()
+{
+ PX_ASSERT(isReportPair());
+ PX_ASSERT(mReportPairIndex == INVALID_REPORT_PAIR_ID);
+
+ if (readFlag(WAS_IN_PERSISTENT_EVENT_LIST))
+ {
+ getScene().getNPhaseCore()->addToPersistentContactEventPairs(this);
+ mFlags &= ~WAS_IN_PERSISTENT_EVENT_LIST;
+ }
+}
+
+
+PX_FORCE_INLINE void Sc::ShapeInteraction::processReportPairOnDeactivate()
+{
+ PX_ASSERT(isReportPair());
+ PX_ASSERT(mReportPairIndex != INVALID_REPORT_PAIR_ID);
+ PX_COMPILE_TIME_ASSERT(IS_IN_PERSISTENT_EVENT_LIST == (WAS_IN_PERSISTENT_EVENT_LIST >> 1));
+ PX_ASSERT(!(readFlag(WAS_IN_PERSISTENT_EVENT_LIST)));
+
+ PxU32 wasInPersList = (mFlags & IS_IN_PERSISTENT_EVENT_LIST) << 1;
+ mFlags |= wasInPersList;
+
+ removeFromReportPairList();
+}
+
+
+void Sc::ShapeInteraction::setContactReportPostSolverVelocity(ContactStreamManager& cs)
+{
+ Scene& scene = getScene();
+ NPhaseCore* npcore = scene.getNPhaseCore();
+ PxU8* stream = npcore->getContactReportPairData(cs.bufferIndex);
+
+ ActorPairReport& apr = getActorPairReport();
+ cs.setContactReportPostSolverVelocity(stream, apr.getActorA(), apr.getActorB());
+}
+
+void Sc::ShapeInteraction::resetManagerCachedState() const
+{
+ if (mManager)
+ {
+ Sc::Scene& scene = getScene();
+ PxvNphaseImplementationContext* nphaseImplementationContext = scene.getLowLevelContext()->getNphaseImplementationContext();
+ PX_ASSERT(nphaseImplementationContext);
+
+ mManager->resetFrictionCachedState();
+ nphaseImplementationContext->refreshContactManager(mManager);
+ }
+}
+
+/*
+ This method can be called from various stages in the pipeline, some of which operate before the actor has advanced its pose and some after it has advanced its pose.
+ Discrete touch found events operate before the pose has been updated. This is because we are using the found event to active the bodies before solve so that we can just
+ solve the activated bodies.
+ Lost touch events occur after the pose has been updated.
+*/
+void Sc::ShapeInteraction::processUserNotification(PxU32 contactEvent, PxU16 infoFlags, bool touchLost, const PxU32 ccdPass, const bool useCurrentTransform, PxsContactManagerOutputIterator& outputs)
+{
+ PX_ASSERT(hasTouch());
+
+ contactEvent = (!ccdPass) ? contactEvent : (contactEvent | PxPairFlag::eNOTIFY_TOUCH_CCD);
+
+ if(mManager)
+ Ps::prefetchLine(mManager);
+
+ Scene& scene = getScene();
+ NPhaseCore* npcore = scene.getNPhaseCore();
+
+ PxU32 pairFlags = getPairFlags();
+ PX_ASSERT(pairFlags & contactEvent);
+ const PxU32 extraDataFlags = pairFlags & CONTACT_REPORT_EXTRA_DATA;
+
+ // make sure shape A and shape B are the same way round as the actors (in compounds they may be swapped)
+ // TODO: make "unswapped" a SIP flag and set it in updateState()
+ if (mActorPair == NULL)
+ {
+ return;
+ }
+ ActorPairReport& aPairReport = getActorPairReport();
+ const bool unswapped = &aPairReport.getActorA() == &getShape0().getRbSim();
+ const Sc::ShapeSim& shapeA = unswapped ? getShape0() : getShape1();
+ const Sc::ShapeSim& shapeB = unswapped ? getShape1() : getShape0();
+
+ if(!aPairReport.isInContactReportActorPairSet())
+ {
+ aPairReport.setInContactReportActorPairSet();
+ npcore->addToContactReportActorPairSet(&aPairReport);
+ aPairReport.incRefCount();
+ }
+
+ // Prepare user notification
+ PxU32 timeStamp = scene.getTimeStamp();
+ PxU32 shapePairTimeStamp = scene.getReportShapePairTimeStamp();
+
+ PxU8* stream = NULL;
+ ContactShapePair* pairStream = NULL;
+ ContactStreamManager& cs = aPairReport.createContactStreamManager(*npcore);
+ if(aPairReport.streamResetStamp(timeStamp))
+ {
+ PX_ASSERT(mContactReportStamp != shapePairTimeStamp); // actor pair and shape pair timestamps must both be out of sync in this case
+
+ PxU16 maxCount;
+ if (cs.maxPairCount != 0)
+ maxCount = cs.maxPairCount; // use value from previous report
+ else
+ {
+ // TODO: Use some kind of heuristic
+ maxCount = 2;
+ cs.maxPairCount = maxCount;
+ }
+
+ PxU32 maxExtraDataSize;
+ if (!extraDataFlags || touchLost)
+ {
+ maxExtraDataSize = 0;
+ cs.setMaxExtraDataSize(maxExtraDataSize);
+ }
+ else
+ {
+ PxU32 currentMaxExtraDataSize = cs.getMaxExtraDataSize();
+ maxExtraDataSize = ContactStreamManager::computeContactReportExtraDataSize(extraDataFlags, true);
+ PX_ASSERT(maxExtraDataSize > 0);
+ if (maxExtraDataSize <= currentMaxExtraDataSize)
+ maxExtraDataSize = currentMaxExtraDataSize; // use value from previous report
+ else
+ cs.setMaxExtraDataSize(maxExtraDataSize);
+ }
+
+ stream = npcore->reserveContactReportPairData(maxCount, maxExtraDataSize, cs.bufferIndex);
+
+ if (!maxExtraDataSize) // this is the usual case, so set it first for branch prediction
+ cs.reset();
+ else if (stream)
+ {
+ cs.reset();
+ PX_ASSERT(extraDataFlags);
+ PX_ASSERT(!touchLost);
+
+ cs.fillInContactReportExtraData(stream, extraDataFlags, aPairReport.getActorA(), aPairReport.getActorB(), ccdPass, useCurrentTransform, 0, sizeof(ContactStreamHeader));
+ if ((extraDataFlags & PxPairFlag::ePOST_SOLVER_VELOCITY) && (pairFlags & PxPairFlag::eDETECT_CCD_CONTACT))
+ scene.setPostSolverVelocityNeeded();
+ }
+ }
+ else
+ {
+ const PxU32 currentPairCount = cs.currentPairCount;
+ if(currentPairCount != 0)
+ {
+ PxU8* tmpStreamPtr = npcore->getContactReportPairData(cs.bufferIndex);
+ if (!extraDataFlags)
+ stream = tmpStreamPtr; // this is the usual case, so set it first for branch prediction
+ else
+ {
+ if (!touchLost)
+ {
+ // - the first few shape pair events might not request extra data
+ // - the events so far were due to touch lost
+ // - multiple reports due to CCD multiple passes
+ // Hence, the extra data has to be created/extended now.
+ //
+ PxU16 oldExtraDataSize = cs.extraDataSize;
+ PxI32 lastContactPass;
+ if (oldExtraDataSize)
+ {
+ ContactStreamHeader* strHeader = reinterpret_cast<ContactStreamHeader*>(tmpStreamPtr);
+ lastContactPass = strHeader->contactPass;
+ }
+ else
+ lastContactPass = -1;
+
+ if (PxI32(ccdPass) > lastContactPass) // do not send extra data mulitple times for the same contact pass
+ {
+ PxU16 extraDataSize = PxU16(oldExtraDataSize + ContactStreamManager::computeContactReportExtraDataSize(extraDataFlags, (oldExtraDataSize == 0)));
+ PxU8* strPtr;
+ if (extraDataSize <= cs.getMaxExtraDataSize())
+ strPtr = tmpStreamPtr;
+ else
+ strPtr = npcore->resizeContactReportPairData(currentPairCount < cs.maxPairCount ? cs.maxPairCount : PxU32(cs.maxPairCount+1), extraDataSize, cs);
+ // the check for max pair count is there to avoid another potential allocation further below
+
+ if (strPtr)
+ {
+ stream = strPtr;
+ PxU32 sizeOffset;
+ if (oldExtraDataSize)
+ sizeOffset = oldExtraDataSize;
+ else
+ sizeOffset = sizeof(ContactStreamHeader);
+ cs.fillInContactReportExtraData(strPtr, extraDataFlags, aPairReport.getActorA(), aPairReport.getActorB(), ccdPass, useCurrentTransform, currentPairCount, sizeOffset);
+ if ((extraDataFlags & PxPairFlag::ePOST_SOLVER_VELOCITY) && (pairFlags & PxPairFlag::eDETECT_CCD_CONTACT))
+ scene.setPostSolverVelocityNeeded();
+ }
+ else
+ {
+ stream = tmpStreamPtr;
+ cs.raiseFlags(ContactStreamManagerFlag::eINCOMPLETE_STREAM);
+ }
+ }
+ else
+ stream = tmpStreamPtr;
+ }
+ else
+ stream = tmpStreamPtr;
+ }
+ }
+ }
+
+ if(stream)
+ pairStream = cs.getShapePairs(stream);
+ else
+ {
+ cs.raiseFlags(ContactStreamManagerFlag::eINVALID_STREAM);
+ return;
+ }
+
+ ContactShapePair* cp;
+ if (mContactReportStamp != shapePairTimeStamp)
+ {
+ // this shape pair is not in the contact notification stream yet
+
+ if (cs.currentPairCount < cs.maxPairCount)
+ cp = pairStream + cs.currentPairCount;
+ else
+ {
+ PxU32 newSize = PxU32(cs.currentPairCount + (cs.currentPairCount >> 1) + 1);
+ stream = npcore->resizeContactReportPairData(newSize, cs.getMaxExtraDataSize(), cs);
+ if(stream)
+ {
+ pairStream = cs.getShapePairs(stream);
+ cp = pairStream + cs.currentPairCount;
+ }
+ else
+ {
+ cs.raiseFlags(ContactStreamManagerFlag::eINCOMPLETE_STREAM);
+ return;
+ }
+ }
+
+ //!!! why is alignment important here? Looks almost like some refactor nonsense
+ PX_ASSERT(0==(reinterpret_cast<uintptr_t>(stream) & 0x0f)); // check 16Byte alignment
+
+ mReportStreamIndex = cs.currentPairCount;
+ cp->shapes[0] = shapeA.getPxShape();
+ cp->shapes[1] = shapeB.getPxShape();
+ cp->contactPatches = NULL;
+ cp->contactPoints = NULL;
+ cp->contactForces = NULL;
+ cp->contactCount = 0;
+ cp->patchCount = 0;
+ cp->constraintStreamSize = 0;
+ cp->requiredBufferSize = 0;
+ cp->flags = infoFlags;
+ PX_ASSERT(contactEvent <= 0xffff);
+ cp->events = PxU16(contactEvent);
+ cp->shapeID[0] = shapeA.getID();
+ cp->shapeID[1] = shapeB.getID();
+
+ cs.currentPairCount++;
+
+ mContactReportStamp = shapePairTimeStamp;
+ }
+ else
+ {
+ // this shape pair is in the contact notification stream already but there is a second event (can happen with force threshold reports, for example).
+
+ PX_ASSERT(mReportStreamIndex < cs.currentPairCount);
+ cp = &pairStream[mReportStreamIndex];
+ cp->events |= contactEvent;
+ if (touchLost && cp->events & PxPairFlag::eNOTIFY_TOUCH_PERSISTS)
+ cp->events &= PxU16(~PxPairFlag::eNOTIFY_TOUCH_PERSISTS);
+ cp->flags |= infoFlags;
+ }
+
+ if ((getPairFlags() & PxPairFlag::eNOTIFY_CONTACT_POINTS) && mManager && (!cp->contactPatches) && !(contactEvent & PxU32(PxPairFlag::eNOTIFY_TOUCH_LOST | PxPairFlag::eNOTIFY_THRESHOLD_FORCE_LOST)))
+ {
+
+ const PxcNpWorkUnit& workUnit = mManager->getWorkUnit();
+ PxsContactManagerOutput* output = NULL;
+ if(workUnit.mNpIndex & PxsContactManagerBase::NEW_CONTACT_MANAGER_MASK)
+ output = &getScene().getLowLevelContext()->getNphaseImplementationContext()->getNewContactManagerOutput(workUnit.mNpIndex);
+ else
+ output = &outputs.getContactManager(workUnit.mNpIndex);
+
+ const PxsCCDContactHeader* ccdContactData = reinterpret_cast<const PxsCCDContactHeader*>(workUnit.ccdContacts);
+
+ bool isCCDPass = (ccdPass != 0);
+ if ((output->nbPatches && !isCCDPass) || (ccdContactData && (!ccdContactData->isFromPreviousPass) && isCCDPass))
+ {
+ const PxU8* contactPatchData;
+ const PxU8* contactPointData;
+ PxU32 cDataSize;
+ PxU32 alignedContactDataSize;
+ const PxReal* impulses;
+
+ PxU32 nbPoints = output->nbContacts;
+ PxU32 contactPatchCount = output->nbPatches;
+
+ if (!isCCDPass)
+ {
+ PX_ASSERT(0==(reinterpret_cast<const uintptr_t>(output->contactPatches) & 0x0f)); // check 16Byte alignment
+ contactPatchData = output->contactPatches;
+ contactPointData = output->contactPoints;
+ cDataSize = sizeof(PxContactPatch)*output->nbPatches + sizeof(PxContact)*output->nbContacts;
+ alignedContactDataSize = (cDataSize + 0xf) & 0xfffffff0;
+ impulses = output->contactForces;
+ }
+ else
+ {
+ PX_ASSERT(0==(reinterpret_cast<const uintptr_t>(ccdContactData) & 0x0f)); // check 16Byte alignment
+ contactPatchData = reinterpret_cast<const PxU8*>(ccdContactData) + sizeof(PxsCCDContactHeader);
+ contactPointData = contactPatchData + sizeof(PxContactPatch);
+ cDataSize = ccdContactData->contactStreamSize - sizeof(PxsCCDContactHeader);
+ PxU32 tmpAlignedSize = (ccdContactData->contactStreamSize + 0xf) & 0xfffffff0;
+ alignedContactDataSize = tmpAlignedSize - sizeof(PxsCCDContactHeader);
+ impulses = reinterpret_cast<const PxReal*>(contactPatchData + alignedContactDataSize);
+ nbPoints = 1;
+ contactPatchCount = 1;
+ }
+
+ infoFlags = cp->flags;
+ infoFlags |= unswapped ? 0 : PxContactPairFlag::eINTERNAL_CONTACTS_ARE_FLIPPED;
+
+ //PX_ASSERT(0==(reinterpret_cast<const uintptr_t>(impulses) & 0x0f));
+
+ PxU32 impulseSize = impulses ? (nbPoints * sizeof(PxReal)) : 0;
+ if (impulseSize)
+ infoFlags |= PxContactPairFlag::eINTERNAL_HAS_IMPULSES;
+ cp->contactPatches = contactPatchData;
+ cp->contactPoints = contactPointData;
+ cp->contactCount = Ps::to8(nbPoints);
+ cp->patchCount = Ps::to8(contactPatchCount);
+ cp->constraintStreamSize = Ps::to16(cDataSize);
+ cp->requiredBufferSize = alignedContactDataSize + impulseSize;
+ cp->contactForces = impulses;
+
+ cp->flags = infoFlags;
+ }
+ }
+}
+
+PxU32 Sc::ShapeInteraction::getContactPointData(const void*& contactPatches, const void*& contactPoints, PxU32& contactDataSize, PxU32& contactPointCount, PxU32& numPatches, const PxReal*& impulses, PxU32 startOffset,
+ PxsContactManagerOutputIterator& outputs)
+{
+ // Process LL generated contacts
+ if (mManager != NULL)
+ {
+ PxcNpWorkUnit& workUnit = mManager->getWorkUnit();
+
+ PxsContactManagerOutput* output = NULL;
+
+ if(workUnit.mNpIndex & PxsContactManagerBase::NEW_CONTACT_MANAGER_MASK)
+ {
+ output = &getScene().getLowLevelContext()->getNphaseImplementationContext()->getNewContactManagerOutput(workUnit.mNpIndex);
+ }
+ else
+ {
+ output = &outputs.getContactManager(workUnit.mNpIndex);
+ }
+
+ /*const void* dcdContactPatches;
+ const void* dcdContactPoints;
+ PxU32 dcdContactPatchCount;
+ const PxReal* dcdImpulses;
+ const PxsCCDContactHeader* ccdContactStream;
+ PxU32 dcdContactCount = mManager->getContactPointData(dcdContactPatches, dcdContactPoints, dcdContactPatchCount, dcdImpulses, ccdContactStream);
+
+ PX_ASSERT(((dcdContactCount == 0) && (!ccdContactStream)) || ((dcdContactCount > 0) && hasTouch()) || (ccdContactStream && hasCCDTouch()));*/
+
+ const PxsCCDContactHeader* ccdContactStream = reinterpret_cast<const PxsCCDContactHeader*>(workUnit.ccdContacts);
+
+ PxU32 idx = 0;
+ if (output->nbContacts)
+ {
+ if (startOffset == 0)
+ {
+ contactPatches = output->contactPatches;
+ contactPoints = output->contactPoints;
+ contactDataSize = sizeof(PxContactPatch) * output->nbPatches + sizeof(PxContact) * output->nbContacts;
+ contactPointCount = output->nbContacts;
+ numPatches = output->nbPatches;
+ impulses = output->contactForces;
+
+ if (!ccdContactStream)
+ return startOffset;
+ else
+ return (startOffset + 1);
+ }
+
+ idx++;
+ }
+
+ while(ccdContactStream)
+ {
+ if (startOffset == idx)
+ {
+ const PxU8* stream = reinterpret_cast<const PxU8*>(ccdContactStream);
+ PxU16 streamSize = ccdContactStream->contactStreamSize;
+ contactPatches = stream + sizeof(PxsCCDContactHeader);
+ contactPoints = stream + sizeof(PxsCCDContactHeader) + sizeof(PxContactPatch);
+ contactDataSize = streamSize - sizeof(PxsCCDContactHeader);
+ contactPointCount = 1;
+ numPatches = 1;
+ impulses = reinterpret_cast<const PxReal*>(stream + ((streamSize + 0xf) & 0xfffffff0));
+
+ if (!ccdContactStream->nextStream)
+ return startOffset;
+ else
+ return (startOffset + 1);
+ }
+
+ idx++;
+ ccdContactStream = ccdContactStream->nextStream;
+ }
+ }
+
+ contactPatches = NULL;
+ contactPoints = NULL;
+ contactDataSize = 0;
+ contactPointCount = 0;
+ numPatches = 0;
+ impulses = NULL;
+ return startOffset;
+}
+
+
+// Note that LL will not send end touch events for managers that are destroyed while having contact
+void Sc::ShapeInteraction::managerNewTouch(const PxU32 ccdPass, bool adjustCounters, PxsContactManagerOutputIterator& outputs, bool useAdaptiveForce)
+{
+ if (readFlag(HAS_TOUCH))
+ return; // Do not count the touch twice (for instance when recreating a manager with touch)
+ // We have contact this frame
+ setHasTouch();
+
+ if (adjustCounters)
+ adjustCountersOnNewTouch(useAdaptiveForce);
+
+ if(!isReportPair())
+ return;
+ else
+ {
+ PX_ASSERT(hasTouch());
+ PX_ASSERT(!readFlag(IS_IN_PERSISTENT_EVENT_LIST));
+ PX_ASSERT(!readFlag(IS_IN_FORCE_THRESHOLD_EVENT_LIST));
+
+ PxU32 pairFlags = getPairFlags();
+ if (pairFlags & PxPairFlag::eNOTIFY_TOUCH_FOUND)
+ {
+ PxU16 infoFlag = 0;
+ if (mActorPair->getTouchCount() == 1) // this code assumes that the actor pair touch count does get incremented beforehand
+ {
+ infoFlag = PxContactPairFlag::eACTOR_PAIR_HAS_FIRST_TOUCH;
+ }
+
+ processUserNotification(PxPairFlag::eNOTIFY_TOUCH_FOUND, infoFlag, false, ccdPass, true, outputs);
+ }
+
+ if (pairFlags & PxPairFlag::eNOTIFY_TOUCH_PERSISTS)
+ {
+ getScene().getNPhaseCore()->addToPersistentContactEventPairsDelayed(this); // to make sure that from now on, the pairs are tested for persistent contact events
+ }
+ else if (pairFlags & ShapeInteraction::CONTACT_FORCE_THRESHOLD_PAIRS)
+ {
+ // new touch -> need to start checking for force threshold events
+ // Note: this code assumes that it runs before the pairs get tested for force threshold exceeded
+ getScene().getNPhaseCore()->addToForceThresholdContactEventPairs(this);
+ }
+ }
+}
+
+
+bool Sc::ShapeInteraction::managerLostTouch(const PxU32 ccdPass, bool adjustCounters, PxsContactManagerOutputIterator& outputs, bool useAdaptiveForce)
+{
+ if (!readFlag(HAS_TOUCH))
+ return false;
+
+ // We do not have LL contacts this frame and also we lost LL contact this frame
+
+ if (!isReportPair())
+ {
+ setHasNoTouch();
+ }
+ else
+ {
+ PX_ASSERT(hasTouch());
+
+ sendLostTouchReport(false, ccdPass, outputs);
+
+ if (readFlag(IS_IN_CONTACT_EVENT_LIST))
+ {
+ // don't need to worry about persistent/force-threshold contact events until next new touch
+
+ if (readFlag(IS_IN_FORCE_THRESHOLD_EVENT_LIST))
+ {
+ getScene().getNPhaseCore()->removeFromForceThresholdContactEventPairs(this);
+ }
+ else
+ {
+ PX_ASSERT(readFlag(IS_IN_PERSISTENT_EVENT_LIST));
+ getScene().getNPhaseCore()->removeFromPersistentContactEventPairs(this);
+ }
+
+ clearFlag(FORCE_THRESHOLD_EXCEEDED_FLAGS);
+ }
+
+ setHasNoTouch();
+ }
+
+ BodySim* body0 = getShape0().getBodySim();
+ BodySim* body1 = getShape1().getBodySim();
+ PX_ASSERT(body0); // the first shape always belongs to a dynamic body
+
+ if (adjustCounters)
+ adjustCountersOnLostTouch(body0, body1, useAdaptiveForce);
+
+ if (!body1)
+ {
+ body0->internalWakeUp();
+ return false;
+ }
+ return true;
+}
+
+
+PX_FORCE_INLINE void Sc::ShapeInteraction::updateFlags(const Sc::Scene& scene, const Sc::BodySim* bs0, const Sc::BodySim* bs1, const PxU32 pairFlags)
+{
+ PX_ASSERT(bs0); // the first shape always belongs to a dynamic body
+
+ // Check if collision response is disabled
+ bool enabled = (!bs0->isKinematic()) || (bs1 && !bs1->isKinematic()); // If the pair has no dynamic body then disable response
+ enabled = enabled && (pairFlags & PxPairFlag::eSOLVE_CONTACT);
+ setFlag(CONTACTS_RESPONSE_DISABLED, !enabled);
+
+ // Check if contact points needed
+ setFlag(CONTACTS_COLLECT_POINTS, ( (pairFlags & PxPairFlag::eNOTIFY_CONTACT_POINTS) ||
+ (pairFlags & PxPairFlag::eMODIFY_CONTACTS) ||
+ scene.getVisualizationParameter(PxVisualizationParameter::eCONTACT_POINT) ||
+ scene.getVisualizationParameter(PxVisualizationParameter::eCONTACT_NORMAL) ||
+ scene.getVisualizationParameter(PxVisualizationParameter::eCONTACT_ERROR) ||
+ scene.getVisualizationParameter(PxVisualizationParameter::eCONTACT_FORCE)) );
+}
+
+
+void Sc::ShapeInteraction::updateState(const PxU8 externalDirtyFlags)
+{
+ const PxU32 oldContactState = getManagerContactState();
+
+ const PxU8 dirtyFlags = PxU8(getDirtyFlags() | externalDirtyFlags);
+ const PxU32 pairFlags = getPairFlags();
+ Scene& scene = getScene();
+
+ if (dirtyFlags & (InteractionDirtyFlag::eFILTER_STATE | InteractionDirtyFlag::eVISUALIZATION))
+ {
+ Sc::BodySim* bs0 = getShape0().getBodySim();
+ Sc::BodySim* bs1 = getShape1().getBodySim();
+
+ Ps::IntBool wasDisabled = readFlag(CONTACTS_RESPONSE_DISABLED);
+
+ updateFlags(scene, bs0, bs1, pairFlags);
+
+ Ps::IntBool isDisabled = readFlag(CONTACTS_RESPONSE_DISABLED);
+
+ if (!wasDisabled && isDisabled)
+ {
+ scene.getSimpleIslandManager()->setEdgeDisconnected(mEdgeIndex);
+ }
+ else if (wasDisabled && !isDisabled)
+ {
+ if (readFlag(ShapeInteraction::HAS_TOUCH))
+ {
+ scene.getSimpleIslandManager()->setEdgeConnected(mEdgeIndex);
+ }
+ }
+ }
+
+ PxU32 newContactState = getManagerContactState();
+ bool recreateManager = (oldContactState != newContactState);
+
+ // No use in updating manager properties if the manager is going to be re-created or does not exist yet
+ if ((!recreateManager) && (mManager != 0))
+ {
+ ShapeSim& shapeSim0 = getShape0();
+ ShapeSim& shapeSim1 = getShape1();
+
+ // Update dominance
+ if (dirtyFlags & InteractionDirtyFlag::eDOMINANCE)
+ {
+ Sc::BodySim* bs0 = shapeSim0.getBodySim();
+ Sc::BodySim* bs1 = shapeSim1.getBodySim();
+ PX_ASSERT(bs0); // the first shape always belongs to a dynamic body
+
+ // Static actors are in dominance group zero and must remain there
+ const PxDominanceGroup dom0 = bs0->getActorCore().getDominanceGroup();
+ const PxDominanceGroup dom1 = bs1 ? bs1->getActorCore().getDominanceGroup() : PxDominanceGroup(0);
+
+ const PxDominanceGroupPair cdom = getScene().getDominanceGroupPair(dom0, dom1);
+ mManager->setDominance0(cdom.dominance0);
+ mManager->setDominance1(cdom.dominance1);
+ }
+
+ // Update skin width
+ if (dirtyFlags & InteractionDirtyFlag::eREST_OFFSET)
+ {
+ mManager->setRestDistance(shapeSim0.getRestOffset() + shapeSim1.getRestOffset());
+ }
+
+ //we may want to only write these if they have changed, the set code is a bit painful for the integration flags because of bit unpacking + packing.
+ mManager->setCCD((getPairFlags() & PxPairFlag::eDETECT_CCD_CONTACT) != 0);
+ }
+ else if (readInteractionFlag(InteractionFlag::eIS_ACTIVE)) // only re-create the manager if the pair is active
+ {
+ PX_ASSERT(mManager); // if the pair is active, there has to be a manager
+ PX_ASSERT(activeManagerAllowed());
+
+ // A) This is a newly created pair
+ //
+ // B) The contact notification or processing state has changed.
+ // All existing managers need to be deleted and recreated with the correct flag set
+ // These flags can only be set at creation in LL
+ destroyManager();
+ createManager(NULL);
+ }
+}
+
+
+bool Sc::ShapeInteraction::onActivate(void* contactManager)
+{
+ if (isReportPair())// && !(infoFlag & ActorSim::AS_PART_OF_ISLAND_GEN_PASS_1))
+ {
+ // for pairs that go through a second island pass, there is the possibility that they get put to sleep again after the second pass.
+ // So we do not want to check for re-insertion into the persistent report pair list yet.
+ processReportPairOnActivate();
+ }
+
+ if (updateManager(contactManager))
+ {
+ raiseInteractionFlag(InteractionFlag::eIS_ACTIVE);
+ return true;
+ }
+ else
+ return false;
+}
+
+
+bool Sc::ShapeInteraction::onDeactivate(PxU32 /*infoFlag*/)
+{
+ PX_ASSERT(getShape0().getActor().isDynamicRigid() || getShape1().getActor().isDynamicRigid());
+
+ const BodySim* bodySim0 = getShape0().getBodySim();
+ const BodySim* bodySim1 = getShape1().getBodySim();
+ PX_ASSERT(bodySim0); // the first shape always belongs to a dynamic body
+
+ PX_ASSERT( (!bodySim0 && bodySim1 && !bodySim1->isActive()) ||
+ (!bodySim1 && bodySim0 && !bodySim0->isActive()) ||
+ ((bodySim0 && bodySim1 && (!bodySim0->isActive() || !bodySim1->isActive()))) );
+
+ if ((!bodySim0->isActive()) && (!bodySim1 || !bodySim1->isActive()))
+ {
+ if (mReportPairIndex != INVALID_REPORT_PAIR_ID)
+ {
+
+ processReportPairOnDeactivate();
+ }
+
+ PX_ASSERT((mManager->getTouchStatus() > 0) == (hasTouch() > 0));
+
+ if (mManager != 0)
+ {
+ if ((!readFlag(TOUCH_KNOWN)) && mManager->touchStatusKnown() && (!mManager->getTouchStatus()))
+ {
+ // for pairs that are inserted asleep, we do not know the touch state. If they run through narrowphase and a touch is found,
+ // then a managerNewTouch() call will inform this object about the found touch. However, if narrowphase detects that there
+ // is no touch, this object will not be informed about it. The low level manager will always know though. Now, before destroying
+ // the pair manager, we need to record "does not have touch" state if available.
+ raiseFlag(HAS_NO_TOUCH);
+ }
+
+ destroyManager();
+ if (mEdgeIndex != IG_INVALID_EDGE)
+ getScene().getSimpleIslandManager()->clearEdgeRigidCM(mEdgeIndex);
+ }
+ Scene& scene = getScene();
+ scene.getSimpleIslandManager()->deactivateEdge(mEdgeIndex);
+
+ //
+ // We distinguish two scenarios here:
+ //
+ // A) island generation deactivates objects:
+ // -> the deactivated body was active
+ // -> narrowphase ran on this pair
+ // -> the touch status is known
+ // -> touch: the objects of the pair are in the same island
+ // -> no touch: the objects of the pair are in different islands
+ //
+ // As a consequence, the edge state is not changed. The assumption is that anything that could break the touch status
+ // from here on will have to mark the edges connected (for example if the object gets moved).
+ //
+ // B) user deactivates objects:
+ // -> the touch status might not be known (for example, the pose gets integrated after the solver which might cause a change
+ // in touch status. If the object gets put to sleep after that, we have to be conservative and mark the edge connected.
+ // other example: an active object gets moved by the user and then deactivated).
+ //
+
+ clearInteractionFlag(InteractionFlag::eIS_ACTIVE);
+ return true;
+ }
+ else
+ {
+ return false;
+ }
+}
+
+
+void Sc::ShapeInteraction::createManager(void* contactManager)
+{
+ //PX_PROFILE_ZONE("ShapeInteraction.createManager", 0);
+
+ Sc::Scene& scene = getScene();
+
+ const PxU32 pairFlags = getPairFlags();
+
+ const int disableCCDContact = !(pairFlags & PxPairFlag::eDETECT_CCD_CONTACT);
+
+ PxsContactManager* manager = scene.getLowLevelContext()->createContactManager(reinterpret_cast<PxsContactManager*>(contactManager), !disableCCDContact);
+ PxcNpWorkUnit& mNpUnit = manager->getWorkUnit();
+
+
+
+ // Check if contact generation callback has been ordered on the pair
+ int contactChangeable = 0;
+ if (pairFlags & PxPairFlag::eMODIFY_CONTACTS)
+ contactChangeable = 1;
+
+ ShapeSim& shapeSim0 = getShape0();
+ ShapeSim& shapeSim1 = getShape1();
+
+ PxActorType::Enum type0 = shapeSim0.getActor().getActorType(), type1 = shapeSim1.getActor().getActorType();
+
+
+ const int disableResponse = readFlag(CONTACTS_RESPONSE_DISABLED) ? 1 : 0;
+ const int disableDiscreteContact = !(pairFlags & PxPairFlag::eDETECT_DISCRETE_CONTACT);
+ const int reportContactInfo = readFlag(CONTACTS_COLLECT_POINTS);
+ const int hasForceThreshold = !disableResponse && (pairFlags & CONTACT_FORCE_THRESHOLD_PAIRS);
+ int touching;
+ if (readFlag(TOUCH_KNOWN))
+ touching = readFlag(HAS_TOUCH) ? 1 : -1;
+ else
+ touching = 0;
+
+
+
+ // Static actors are in dominance group zero and must remain there
+
+ Sc::BodySim* bs0 = shapeSim0.getBodySim();
+ Sc::BodySim* bs1 = shapeSim1.getBodySim();
+
+ PX_ASSERT(bs0); // The shapes have to be sorted such that the first shape belongs to a dynamic
+
+ const PxDominanceGroup dom0 = bs0->getActorCore().getDominanceGroup();
+ const PxDominanceGroup dom1 = bs1 ? bs1->getActorCore().getDominanceGroup() : PxDominanceGroup(0);
+
+ const bool kinematicActor = bs1 ? bs1->isKinematic() : false;
+
+ const PxDominanceGroupPair cdom = scene.getDominanceGroupPair(dom0, dom1);
+
+ PxI32 hasArticulations = (type0 == PxActorType::eARTICULATION_LINK) | (type1 == PxActorType::eARTICULATION_LINK)<<1;
+ PxI32 hasDynamics = (type0 != PxActorType::eRIGID_STATIC) | (type1 != PxActorType::eRIGID_STATIC)<<1;
+
+ const PxsShapeCore* shapeCore0 = &shapeSim0.getCore().getCore();
+ const PxsShapeCore* shapeCore1 = &shapeSim1.getCore().getCore();
+
+
+
+ //Initialize the manager....
+
+ manager->mRigidBody0 = &bs0->getLowLevelBody();
+ manager->mRigidBody1 = bs1 ? &bs1->getLowLevelBody() : NULL;
+ manager->mShapeInteraction = this;
+ mNpUnit.shapeCore0 = shapeCore0;
+ mNpUnit.shapeCore1 = shapeCore1;
+
+ PX_ASSERT(shapeCore0->transform.isValid() && shapeCore1->transform.isValid());
+
+ mNpUnit.rigidCore0 = &shapeSim0.getPxsRigidCore();
+ mNpUnit.rigidCore1 = &shapeSim1.getPxsRigidCore();
+
+ mNpUnit.restDistance = shapeSim0.getRestOffset() + shapeSim1.getRestOffset();
+ mNpUnit.dominance0 = cdom.dominance0;
+ mNpUnit.dominance1 = cdom.dominance1;
+ mNpUnit.geomType0 = PxU8(shapeCore0->geometry.getType());
+ mNpUnit.geomType1 = PxU8(shapeCore1->geometry.getType());
+ mNpUnit.mTransformCache0 = shapeSim0.getTransformCacheID();
+ mNpUnit.mTransformCache1 = shapeSim1.getTransformCacheID();
+
+
+ PxU16 wuflags = 0;
+
+ if(hasArticulations & 1)
+ wuflags |= PxcNpWorkUnitFlag::eARTICULATION_BODY0;
+
+ if(hasArticulations & 2)
+ wuflags |= PxcNpWorkUnitFlag::eARTICULATION_BODY1;
+
+ if(hasDynamics & 1)
+ wuflags |= PxcNpWorkUnitFlag::eDYNAMIC_BODY0;
+
+ if(hasDynamics & 2)
+ wuflags |= PxcNpWorkUnitFlag::eDYNAMIC_BODY1;
+
+ if(!disableResponse && !contactChangeable)
+ wuflags |= PxcNpWorkUnitFlag::eOUTPUT_CONSTRAINTS;
+
+ if(!disableDiscreteContact)
+ wuflags |= PxcNpWorkUnitFlag::eDETECT_DISCRETE_CONTACT;
+ if(kinematicActor)
+ wuflags |= PxcNpWorkUnitFlag::eHAS_KINEMATIC_ACTOR;
+
+ if(disableResponse)
+ wuflags |= PxcNpWorkUnitFlag::eDISABLE_RESPONSE;
+ if(!disableCCDContact)
+ wuflags |= PxcNpWorkUnitFlag::eDETECT_CCD_CONTACTS;
+
+ // this is just the user req: contact reports can also be generated by body thresholding
+ if(reportContactInfo || contactChangeable)
+ wuflags |= PxcNpWorkUnitFlag::eOUTPUT_CONTACTS;
+
+ if(hasForceThreshold)
+ wuflags |= PxcNpWorkUnitFlag::eFORCE_THRESHOLD;
+
+ if(contactChangeable)
+ wuflags |= PxcNpWorkUnitFlag::eMODIFIABLE_CONTACT;
+
+ mNpUnit.flags = wuflags;
+
+ manager->mFlags = PxU32(contactChangeable ? PxsContactManager::PXS_CM_CHANGEABLE : 0) | PxU32(disableCCDContact ? 0 : PxsContactManager::PXS_CM_CCD_LINEAR);
+
+ //manager->mUserData = this;
+
+ mNpUnit.mNpIndex = 0xFFffFFff;
+
+ mManager = manager;
+
+
+ PxU8 statusFlags = 0;
+
+ if (touching > 0)
+ {
+ statusFlags |= PxcNpWorkUnitStatusFlag::eHAS_TOUCH;
+ }
+ else if (touching < 0)
+ {
+ statusFlags |= PxcNpWorkUnitStatusFlag::eHAS_NO_TOUCH;
+ }
+
+ mNpUnit.statusFlags = statusFlags;
+
+ //KS - do not register the CMs here if contactManager isn't null. This implies this is a newly-found pair so we'll do that addition outside in parallel
+
+
+ if (contactManager == NULL)
+ {
+ scene.getSimpleIslandManager()->setEdgeRigidCM(mEdgeIndex, mManager);
+ PxvNphaseImplementationContext* nphaseImplementationContext = scene.getLowLevelContext()->getNphaseImplementationContext();
+ PX_ASSERT(nphaseImplementationContext);
+ nphaseImplementationContext->registerContactManager(mManager, touching, 0);
+ }
+}
+
+
+void Sc::ShapeInteraction::onShapeChangeWhileSleeping(bool shapeOfDynamicChanged)
+{
+ //
+ // if an operation that can break touch status occurs, all deactivated pairs need to set the sleep island edge
+ // to connected to make sure that potentially joined islands get detected once parts of the island wake up.
+ // Active interactions can be ignored because the edges of those will be marked connected on deactivation.
+ //
+ if (!mManager)
+ {
+ Scene& scene = getScene();
+
+
+ BodySim* body0 = getShape0().getBodySim();
+ PX_ASSERT(body0); // the first shape always belongs to a dynamic body
+
+ if (shapeOfDynamicChanged && !readFlag(TOUCH_KNOWN))
+ {
+ // conservative approach: if a pair was added asleep, and a body/shape gets moved, we want to check next frame
+ // whether the other body should get woken up. The motivation behind this is to get a similar behavior as in
+ // the case where the objects fell asleep rather than have been added asleep (in that case the object will be
+ // woken up with one frame delay).
+
+ BodySim* body1 = getShape1().getBodySim();
+
+ if (body1 && !readFlag(ShapeInteraction::CONTACTS_RESPONSE_DISABLED)) // the first shape always belongs to a dynamic body, hence no need to test body0
+ scene.addToLostTouchList(body0, body1); // note: this will cause duplicate entries if the pair loses AABB overlap the next frame
+ }
+ else if ((!shapeOfDynamicChanged) && !body0->isActive())
+ {
+ // if the shape of a static rigid body moves or is scaled, then this can break touch status and a second island gen
+ // pass will be needed. However, statics do not have island nodes that can be marked accordinlgy and instead we mark
+ // the dynamics of all pairs if they are asleep (if they are awake, they will get marked when the user puts them to sleep).
+
+ /*PX_ASSERT(!getShape1().getBodySim());
+ body0->notifySecondIslandGenPassNeeded();*/
+ }
+ }
+}
+