You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1584 lines
46 KiB
1584 lines
46 KiB
/*==LICENSE==* |
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools |
|
Copyright (C) 2011 Cyan Worlds, Inc. |
|
|
|
This program is free software: you can redistribute it and/or modify |
|
it under the terms of the GNU General Public License as published by |
|
the Free Software Foundation, either version 3 of the License, or |
|
(at your option) any later version. |
|
|
|
This program is distributed in the hope that it will be useful, |
|
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
GNU General Public License for more details. |
|
|
|
You should have received a copy of the GNU General Public License |
|
along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
|
|
Additional permissions under GNU GPL version 3 section 7 |
|
|
|
If you modify this Program, or any covered work, by linking or |
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, |
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent |
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK |
|
(or a modified version of those libraries), |
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, |
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG |
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the |
|
licensors of this Program grant you additional |
|
permission to convey the resulting work. Corresponding Source for a |
|
non-source form of such a combination shall include the source code for |
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered |
|
work. |
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com |
|
or by snail mail at: |
|
Cyan Worlds, Inc. |
|
14617 N Newport Hwy |
|
Mead, WA 99021 |
|
|
|
*==LICENSE==*/ |
|
#include "plPXPhysical.h" |
|
|
|
#include "NxPhysics.h" |
|
#include "NxCooking.h" |
|
|
|
#include "hsResMgr.h" |
|
#include "hsStream.h" |
|
#include "hsTimer.h" |
|
#include "plProfile.h" |
|
#include "hsQuat.h" |
|
#include "hsSTLStream.h" |
|
|
|
#include "plSimulationMgr.h" |
|
#include "../plPhysical/plPhysicalSDLModifier.h" |
|
#include "../plPhysical/plPhysicalSndGroup.h" |
|
#include "../plPhysical/plPhysicalProxy.h" |
|
#include "../pnSceneObject/plSimulationInterface.h" |
|
#include "../pnSceneObject/plCoordinateInterface.h" |
|
|
|
#include "../pnKeyedObject/plKey.h" |
|
#include "../pnMessage/plCorrectionMsg.h" |
|
#include "../pnMessage/plNodeRefMsg.h" |
|
#include "../pnMessage/plObjRefMsg.h" |
|
#include "../pnMessage/plSDLModifierMsg.h" |
|
#include "../plMessage/plSimStateMsg.h" |
|
#include "../plMessage/plSimInfluenceMsg.h" |
|
#include "../plMessage/plLinearVelocityMsg.h" |
|
#include "../plMessage/plAngularVelocityMsg.h" |
|
#include "../plDrawable/plDrawableGenerator.h" |
|
#include "../plNetClient/plNetClientMgr.h" |
|
#include "../plNetTransport/plNetTransportMember.h" |
|
#include "../plStatusLog/plStatusLog.h" |
|
#include "plPXConvert.h" |
|
#include "plPXPhysicalControllerCore.h" |
|
#include "plPXSubWorld.h" |
|
|
|
#include "../plModifier/plDetectorLog.h" |
|
|
|
|
|
#if 0 |
|
#define SpamMsg(x) x |
|
#else |
|
#define SpamMsg(x) |
|
#endif |
|
|
|
#define LogActivate(func) if (fActor->isSleeping()) SimLog("%s activated by %s", GetKeyName(), func); |
|
|
|
plProfile_Extern(MaySendLocation); |
|
plProfile_Extern(LocationsSent); |
|
plProfile_Extern(PhysicsUpdates); |
|
|
|
static void ClearMatrix(hsMatrix44 &m) |
|
{ |
|
m.fMap[0][0] = 0.0f; m.fMap[0][1] = 0.0f; m.fMap[0][2] = 0.0f; m.fMap[0][3] = 0.0f; |
|
m.fMap[1][0] = 0.0f; m.fMap[1][1] = 0.0f; m.fMap[1][2] = 0.0f; m.fMap[1][3] = 0.0f; |
|
m.fMap[2][0] = 0.0f; m.fMap[2][1] = 0.0f; m.fMap[2][2] = 0.0f; m.fMap[2][3] = 0.0f; |
|
m.fMap[3][0] = 0.0f; m.fMap[3][1] = 0.0f; m.fMap[3][2] = 0.0f; m.fMap[3][3] = 0.0f; |
|
m.NotIdentity(); |
|
} |
|
|
|
int plPXPhysical::fNumberAnimatedPhysicals = 0; |
|
int plPXPhysical::fNumberAnimatedActivators = 0; |
|
|
|
///////////////////////////////////////////////////////////////// |
|
// |
|
// plPXPhysical IMPLEMENTATION |
|
// |
|
///////////////////////////////////////////////////////////////// |
|
|
|
plPXPhysical::plPXPhysical() |
|
: fSDLMod(nil) |
|
, fActor(nil) |
|
, fLOSDBs(plSimDefs::kLOSDBNone) |
|
, fGroup(plSimDefs::kGroupMax) |
|
, fReportsOn(0) |
|
, fLastSyncTime(0.0f) |
|
, fProxyGen(nil) |
|
, fSndGroup(nil) |
|
, fWorldHull(nil) |
|
, fSaveTriangles(nil) |
|
, fHullNumberPlanes(0) |
|
, fWeWereHit(false) |
|
, fHitForce(0,0,0) |
|
, fHitPos(0,0,0) |
|
, fInsideConvexHull(false) |
|
{ |
|
} |
|
|
|
plPXPhysical::~plPXPhysical() |
|
{ |
|
SpamMsg(plSimulationMgr::Log("Destroying physical %s", GetKeyName())); |
|
|
|
if (fActor) |
|
{ |
|
// Grab any mesh we may have (they need to be released manually) |
|
NxConvexMesh* convexMesh = nil; |
|
NxTriangleMesh* triMesh = nil; |
|
NxShape* shape = fActor->getShapes()[0]; |
|
if (NxConvexShape* convexShape = shape->isConvexMesh()) |
|
convexMesh = &convexShape->getConvexMesh(); |
|
else if (NxTriangleMeshShape* trimeshShape = shape->isTriangleMesh()) |
|
triMesh = &trimeshShape->getTriangleMesh(); |
|
|
|
if (!fActor->isDynamic()) |
|
plPXPhysicalControllerCore::RebuildCache(); |
|
|
|
if (fActor->isDynamic() && fActor->readBodyFlag(NX_BF_KINEMATIC)) |
|
{ |
|
if (fGroup == plSimDefs::kGroupDynamic) |
|
fNumberAnimatedPhysicals--; |
|
else |
|
fNumberAnimatedActivators--; |
|
} |
|
|
|
// Release the actor |
|
NxScene* scene = plSimulationMgr::GetInstance()->GetScene(fWorldKey); |
|
scene->releaseActor(*fActor); |
|
fActor = nil; |
|
|
|
// Now that the actor is freed, release the mesh |
|
if (convexMesh) |
|
plSimulationMgr::GetInstance()->GetSDK()->releaseConvexMesh(*convexMesh); |
|
if (triMesh) |
|
plSimulationMgr::GetInstance()->GetSDK()->releaseTriangleMesh(*triMesh); |
|
|
|
// Release the scene, so it can be cleaned up if no one else is using it |
|
plSimulationMgr::GetInstance()->ReleaseScene(fWorldKey); |
|
} |
|
|
|
if (fWorldHull) |
|
delete [] fWorldHull; |
|
if (fSaveTriangles) |
|
delete [] fSaveTriangles; |
|
|
|
delete fProxyGen; |
|
|
|
// remove sdl modifier |
|
plSceneObject* sceneObj = plSceneObject::ConvertNoRef(fObjectKey->ObjectIsLoaded()); |
|
if (sceneObj && fSDLMod) |
|
{ |
|
sceneObj->RemoveModifier(fSDLMod); |
|
} |
|
delete fSDLMod; |
|
} |
|
|
|
static void MakeBoxFromHull(NxConvexMesh* convexMesh, NxBoxShapeDesc& box) |
|
{ |
|
NxConvexMeshDesc desc; |
|
convexMesh->saveToDesc(desc); |
|
|
|
hsScalar minX, minY, minZ, maxX, maxY, maxZ; |
|
minX = minY = minZ = FLT_MAX; |
|
maxX = maxY = maxZ = -FLT_MAX; |
|
|
|
for (int i = 0; i < desc.numVertices; i++) |
|
{ |
|
float* point = (float*)(((char*)desc.points) + desc.pointStrideBytes*i); |
|
float x = point[0]; |
|
float y = point[1]; |
|
float z = point[2]; |
|
|
|
minX = hsMinimum(minX, x); |
|
minY = hsMinimum(minY, y); |
|
minZ = hsMinimum(minZ, z); |
|
maxX = hsMaximum(maxX, x); |
|
maxY = hsMaximum(maxY, y); |
|
maxZ = hsMaximum(maxZ, z); |
|
} |
|
|
|
float xWidth = maxX - minX; |
|
float yWidth = maxY - minY; |
|
float zWidth = maxZ - minZ; |
|
box.dimensions.x = xWidth / 2; |
|
box.dimensions.y = yWidth / 2; |
|
box.dimensions.z = zWidth / 2; |
|
|
|
//hsMatrix44 mat; |
|
//box.localPose.getRowMajor44(&mat.fMap[0][0]); |
|
hsPoint3 trans(minX + (xWidth / 2), minY + (yWidth / 2), minZ + (zWidth / 2)); |
|
//mat.SetTranslate(&trans); |
|
//box.localPose.setRowMajor44(&mat.fMap[0][0]); |
|
|
|
hsMatrix44 boxL2W; |
|
boxL2W.Reset(); |
|
boxL2W.SetTranslate(&trans); |
|
plPXConvert::Matrix(boxL2W, box.localPose); |
|
|
|
} |
|
|
|
void plPXPhysical::IMakeHull(NxConvexMesh* convexMesh, hsMatrix44 l2w) |
|
{ |
|
NxConvexMeshDesc desc; |
|
convexMesh->saveToDesc(desc); |
|
|
|
// make sure there are some triangles to work with |
|
if (desc.numTriangles == 0) |
|
return; |
|
|
|
// get rid of any we may have already had |
|
if (fSaveTriangles) |
|
delete [] fSaveTriangles; |
|
|
|
fHullNumberPlanes = desc.numTriangles; |
|
fSaveTriangles = TRACKED_NEW hsPoint3[fHullNumberPlanes*3]; |
|
|
|
for (int i = 0; i < desc.numTriangles; i++) |
|
{ |
|
UInt32* triangle = (UInt32*)(((char*)desc.triangles) + desc.triangleStrideBytes*i); |
|
float* vertex1 = (float*)(((char*)desc.points) + desc.pointStrideBytes*triangle[0]); |
|
float* vertex2 = (float*)(((char*)desc.points) + desc.pointStrideBytes*triangle[1]); |
|
float* vertex3 = (float*)(((char*)desc.points) + desc.pointStrideBytes*triangle[2]); |
|
hsPoint3 pt1(vertex1[0],vertex1[1],vertex1[2]); |
|
hsPoint3 pt2(vertex2[0],vertex2[1],vertex2[2]); |
|
hsPoint3 pt3(vertex3[0],vertex3[1],vertex3[2]); |
|
|
|
fSaveTriangles[(i*3)+0] = pt1; |
|
fSaveTriangles[(i*3)+1] = pt2; |
|
fSaveTriangles[(i*3)+2] = pt3; |
|
} |
|
} |
|
|
|
void plPXPhysical::ISetHullToWorldWTriangles() |
|
{ |
|
// if we have a detector hull and the world hasn't been updated |
|
if (fWorldHull == nil) |
|
{ |
|
fWorldHull = TRACKED_NEW hsPlane3[fHullNumberPlanes]; |
|
// use the local2world from the physics engine so that it matches the transform of the positions from the triggerees |
|
hsMatrix44 l2w; |
|
plPXConvert::Matrix(fActor->getGlobalPose(), l2w); |
|
int i; |
|
for( i = 0; i < fHullNumberPlanes; i++ ) |
|
{ |
|
hsPoint3 pt1 = fSaveTriangles[i*3]; |
|
hsPoint3 pt2 = fSaveTriangles[(i*3)+1]; |
|
hsPoint3 pt3 = fSaveTriangles[(i*3)+2]; |
|
|
|
// local to world translation |
|
pt1 = l2w * pt1; |
|
pt2 = l2w * pt2; |
|
pt3 = l2w * pt3; |
|
|
|
hsPlane3 plane(&pt1, &pt2, &pt3); |
|
fWorldHull[i] = plane; |
|
} |
|
} |
|
} |
|
|
|
|
|
hsBool plPXPhysical::IsObjectInsideHull(const hsPoint3& pos) |
|
{ |
|
if (fSaveTriangles) |
|
{ |
|
ISetHullToWorldWTriangles(); |
|
int i; |
|
for( i = 0; i < fHullNumberPlanes; i++ ) |
|
{ |
|
if (!ITestPlane(pos, fWorldHull[i])) |
|
return false; |
|
} |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
hsBool plPXPhysical::Should_I_Trigger(hsBool enter, hsPoint3& pos) |
|
{ |
|
// see if we are inside the detector hull, if so, then don't trigger |
|
bool trigger = false; |
|
bool inside = IsObjectInsideHull(pos); |
|
if ( !inside) |
|
{ |
|
trigger = true; |
|
fInsideConvexHull = enter; |
|
} |
|
else |
|
{ |
|
// catch those rare cases on slow machines that miss the collision before avatar penetrated the face |
|
if (enter && !fInsideConvexHull) |
|
{ |
|
#ifdef PHYSX_SAVE_TRIGGERS_WORKAROUND |
|
trigger = true; |
|
fInsideConvexHull = enter; |
|
DetectorLogSpecial("**>Saved a missing enter collision: %s",GetObjectKey()->GetName()); |
|
#else |
|
DetectorLogSpecial("**>Could have saved a missing enter collision: %s",GetObjectKey()->GetName()); |
|
#endif PHYSX_SAVE_TRIGGERS_WORKAROUND |
|
} |
|
} |
|
|
|
return trigger; |
|
} |
|
|
|
|
|
hsBool plPXPhysical::Init() |
|
{ |
|
hsBool startAsleep = false; |
|
|
|
fGroup = fRecipe.group; |
|
fReportsOn = fRecipe.reportsOn; |
|
fObjectKey = fRecipe.objectKey; |
|
fSceneNode = fRecipe.sceneNode; |
|
|
|
NxActorDesc actorDesc; |
|
NxSphereShapeDesc sphereDesc; |
|
NxConvexShapeDesc convexShapeDesc; |
|
NxTriangleMeshShapeDesc trimeshShapeDesc; |
|
NxBoxShapeDesc boxDesc; |
|
|
|
plPXConvert::Matrix(fRecipe.l2s, actorDesc.globalPose); |
|
|
|
switch (fRecipe.bounds) |
|
{ |
|
case plSimDefs::kSphereBounds: |
|
{ |
|
hsMatrix44 sphereL2W; |
|
sphereL2W.Reset(); |
|
sphereL2W.SetTranslate(&fRecipe.offset); |
|
|
|
sphereDesc.radius = fRecipe.radius; |
|
plPXConvert::Matrix(sphereL2W, sphereDesc.localPose); |
|
sphereDesc.group = fRecipe.group; |
|
actorDesc.shapes.pushBack(&sphereDesc); |
|
} |
|
break; |
|
case plSimDefs::kHullBounds: |
|
// FIXME PHYSX - Remove when hull detection is fixed |
|
// If this is read time (ie, meshStream is nil), turn the convex hull |
|
// into a box. That way the data won't have to change when convex hulls |
|
// actually work right. |
|
if (fRecipe.group == plSimDefs::kGroupDetector && fRecipe.meshStream == nil) |
|
{ |
|
#ifdef USE_BOXES_FOR_DETECTOR_HULLS |
|
MakeBoxFromHull(fRecipe.convexMesh, boxDesc); |
|
plSimulationMgr::GetInstance()->GetSDK()->releaseConvexMesh(*fRecipe.convexMesh); |
|
boxDesc.group = fRecipe.group; |
|
actorDesc.shapes.push_back(&boxDesc); |
|
#else |
|
#ifdef USE_PHYSX_CONVEXHULL_WORKAROUND |
|
// make a hull of planes for testing IsInside |
|
IMakeHull(fRecipe.convexMesh,fRecipe.l2s); |
|
#endif // USE_PHYSX_CONVEXHULL_WORKAROUND |
|
convexShapeDesc.meshData = fRecipe.convexMesh; |
|
convexShapeDesc.userData = fRecipe.meshStream; |
|
convexShapeDesc.group = fRecipe.group; |
|
actorDesc.shapes.pushBack(&convexShapeDesc); |
|
#endif // USE_BOXES_FOR_DETECTOR_HULLS |
|
} |
|
else |
|
{ |
|
convexShapeDesc.meshData = fRecipe.convexMesh; |
|
convexShapeDesc.userData = fRecipe.meshStream; |
|
convexShapeDesc.group = fRecipe.group; |
|
actorDesc.shapes.pushBack(&convexShapeDesc); |
|
} |
|
break; |
|
case plSimDefs::kBoxBounds: |
|
{ |
|
boxDesc.dimensions = plPXConvert::Point(fRecipe.bDimensions); |
|
|
|
hsMatrix44 boxL2W; |
|
boxL2W.Reset(); |
|
boxL2W.SetTranslate(&fRecipe.bOffset); |
|
plPXConvert::Matrix(boxL2W, boxDesc.localPose); |
|
|
|
boxDesc.group = fRecipe.group; |
|
actorDesc.shapes.push_back(&boxDesc); |
|
} |
|
break; |
|
case plSimDefs::kExplicitBounds: |
|
case plSimDefs::kProxyBounds: |
|
if (fRecipe.group == plSimDefs::kGroupDetector) |
|
{ |
|
SimLog("Someone using an Exact on a detector region: %s", GetKeyName()); |
|
} |
|
trimeshShapeDesc.meshData = fRecipe.triMesh; |
|
trimeshShapeDesc.userData = fRecipe.meshStream; |
|
trimeshShapeDesc.group = fRecipe.group; |
|
actorDesc.shapes.pushBack(&trimeshShapeDesc); |
|
break; |
|
default: |
|
hsAssert(false, "Unknown geometry type during read."); |
|
return false; |
|
break; |
|
} |
|
|
|
// Now fill out the body, or dynamic part of the physical |
|
NxBodyDesc bodyDesc; |
|
if (fRecipe.mass != 0) |
|
{ |
|
bodyDesc.mass = fRecipe.mass; |
|
actorDesc.body = &bodyDesc; |
|
|
|
if (GetProperty(plSimulationInterface::kPinned)) |
|
{ |
|
bodyDesc.flags |= NX_BF_FROZEN; |
|
startAsleep = true; // put it to sleep if they are going to be frozen |
|
} |
|
|
|
if (fRecipe.group != plSimDefs::kGroupDynamic || GetProperty(plSimulationInterface::kPhysAnim)) |
|
{ |
|
SetProperty(plSimulationInterface::kPassive, true); |
|
|
|
// Even though the code for animated physicals and animated activators are the same |
|
// keep these code snippets separated for fine tuning. Thanks. |
|
if (fRecipe.group == plSimDefs::kGroupDynamic) |
|
{ |
|
// handle the animated physicals.... make kinematic for now. |
|
fNumberAnimatedPhysicals++; |
|
bodyDesc.flags |= NX_BF_KINEMATIC; |
|
startAsleep = true; |
|
} |
|
else |
|
{ |
|
// handle the animated activators.... |
|
fNumberAnimatedActivators++; |
|
bodyDesc.flags |= NX_BF_KINEMATIC; |
|
startAsleep = true; |
|
} |
|
|
|
} |
|
} |
|
else |
|
{ |
|
if ( GetProperty(plSimulationInterface::kPhysAnim) ) |
|
SimLog("An animated physical that has no mass: %s", GetKeyName()); |
|
} |
|
|
|
actorDesc.userData = this; |
|
actorDesc.name = GetKeyName(); |
|
|
|
// Put the dynamics into actor group 1. The actor groups are only used for |
|
// deciding who we get contact reports for. |
|
if (fRecipe.group == plSimDefs::kGroupDynamic) |
|
actorDesc.group = 1; |
|
|
|
NxScene* scene = plSimulationMgr::GetInstance()->GetScene(fWorldKey); |
|
try |
|
{ |
|
fActor = scene->createActor(actorDesc); |
|
} catch (...) |
|
{ |
|
hsAssert(false, "Actor creation crashed"); |
|
return false; |
|
} |
|
hsAssert(fActor, "Actor creation failed"); |
|
if (!fActor) |
|
return false; |
|
|
|
NxShape* shape = fActor->getShapes()[0]; |
|
shape->setMaterial(plSimulationMgr::GetInstance()->GetMaterialIdx(scene, fRecipe.friction, fRecipe.restitution)); |
|
|
|
// Turn on the trigger flags for any detectors. |
|
// |
|
// Normally, we'd set these flags on the shape before it's created. However, |
|
// in the case where the detector is going to be animated, it'll have a rigid |
|
// body too, and that will cause problems at creation. According to Ageia, |
|
// a detector shape doesn't actually count as a shape, so the SDK will have |
|
// problems trying to calculate an intertial tensor. By letting it be |
|
// created as a normal dynamic first, then setting the flags, we work around |
|
// that problem. |
|
if (fRecipe.group == plSimDefs::kGroupDetector) |
|
{ |
|
shape->setFlag(NX_TRIGGER_ON_ENTER, true); |
|
shape->setFlag(NX_TRIGGER_ON_LEAVE, true); |
|
} |
|
|
|
if (GetProperty(plSimulationInterface::kStartInactive) || startAsleep) |
|
{ |
|
if (!fActor->isSleeping()) |
|
{ |
|
if (plSimulationMgr::fExtraProfile) |
|
SimLog("Deactivating %s in SetPositionAndRotationSim", GetKeyName()); |
|
fActor->putToSleep(); |
|
} |
|
} |
|
|
|
if (GetProperty(plSimulationInterface::kDisable)) |
|
IEnable(false); |
|
if (GetProperty(plSimulationInterface::kSuppressed_DEAD)) |
|
IEnable(false); |
|
|
|
plNodeRefMsg* refMsg = TRACKED_NEW plNodeRefMsg(fSceneNode, plRefMsg::kOnCreate, -1, plNodeRefMsg::kPhysical); |
|
hsgResMgr::ResMgr()->AddViaNotify(GetKey(), refMsg, plRefFlags::kActiveRef); |
|
|
|
// only dynamic physicals without noSync need SDLs |
|
if ( fRecipe.group == plSimDefs::kGroupDynamic && !fProps.IsBitSet(plSimulationInterface::kNoSynchronize) ) |
|
{ |
|
// add SDL modifier |
|
plSceneObject* sceneObj = plSceneObject::ConvertNoRef(fObjectKey->ObjectIsLoaded()); |
|
hsAssert(sceneObj, "nil sceneObject, failed to create and attach SDL modifier"); |
|
|
|
delete fSDLMod; |
|
fSDLMod = TRACKED_NEW plPhysicalSDLModifier; |
|
sceneObj->AddModifier(fSDLMod); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
///////////////////////////////////////////////////////////////// |
|
// |
|
// MESSAGE HANDLING |
|
// |
|
///////////////////////////////////////////////////////////////// |
|
|
|
// MSGRECEIVE |
|
hsBool plPXPhysical::MsgReceive( plMessage* msg ) |
|
{ |
|
if(plGenRefMsg *refM = plGenRefMsg::ConvertNoRef(msg)) |
|
{ |
|
return HandleRefMsg(refM); |
|
} |
|
else if(plSimulationMsg *simM = plSimulationMsg::ConvertNoRef(msg)) |
|
{ |
|
plLinearVelocityMsg* velMsg = plLinearVelocityMsg::ConvertNoRef(msg); |
|
if(velMsg) |
|
{ |
|
SetLinearVelocitySim(velMsg->Velocity()); |
|
return true; |
|
} |
|
plAngularVelocityMsg* angvelMsg = plAngularVelocityMsg::ConvertNoRef(msg); |
|
if(angvelMsg) |
|
{ |
|
SetAngularVelocitySim(angvelMsg->AngularVelocity()); |
|
return true; |
|
} |
|
|
|
|
|
return false; |
|
} |
|
// couldn't find a local handler: pass to the base |
|
else |
|
return plPhysical::MsgReceive(msg); |
|
} |
|
|
|
// IHANDLEREFMSG |
|
// there's two things we hold references to: subworlds |
|
// and the simulation manager. |
|
// right now, we're only worrying about the subworlds |
|
hsBool plPXPhysical::HandleRefMsg(plGenRefMsg* refMsg) |
|
{ |
|
UInt8 refCtxt = refMsg->GetContext(); |
|
|
|
switch (refMsg->fType) { |
|
case kPhysRefWorld: { |
|
switch (refCtxt) { |
|
case plRefMsg::kOnCreate: |
|
case plRefMsg::kOnRequest: |
|
// PotS files specify a plHKSubWorld as the subworld key. For everything else, |
|
// the subworlds are a plSceneObject key. For sanity purposes, we will allow |
|
// references to the plPXSubWorld here. HOWEVER, we will need to grab the target |
|
// and replace our reference... |
|
if (plPXSubWorld* subWorldIface = plPXSubWorld::ConvertNoRef(refMsg->GetRef())) { |
|
hsAssert(subWorldIface->GetOwnerKey(), "subworld owner is NULL?! Uh oh..."); |
|
plGenRefMsg* replaceRefMsg = new plGenRefMsg(GetKey(), plRefMsg::kOnReplace, 0, kPhysRefWorld); |
|
hsgResMgr::ResMgr()->AddViaNotify(subWorldIface->GetOwnerKey(), replaceRefMsg, plRefFlags::kActiveRef); |
|
} |
|
// fall-thru is intentional :) |
|
case plRefMsg::kOnReplace: |
|
// loading into a subworld |
|
if (plSceneObject* subSO = plSceneObject::ConvertNoRef(refMsg->GetRef())) { |
|
fWorldKey = subSO->GetKey(); |
|
|
|
// Cyan produced files will never have plPXSubWorld as this is a H'uru-ism. |
|
// Let us make a default one such that we can play with default subworlds at runtime. |
|
// HAAAAAAAAAX!!! |
|
plPXSubWorld* subWorldIface = plPXSubWorld::ConvertNoRef(subSO->GetGenericInterface(plPXSubWorld::Index())); |
|
if (!subWorldIface) { |
|
subWorldIface = new plPXSubWorld(); |
|
|
|
// This can be simplified if/when incorporating zrax' String Theory library |
|
std::string strKeyName = std::string(subSO->GetKeyName()) + "_DefSubWorld"; |
|
const char *cKeyName = strKeyName.c_str(); |
|
|
|
hsgResMgr::ResMgr()->NewKey(cKeyName, |
|
subWorldIface, GetKey()->GetUoid().GetLocation()); |
|
|
|
plObjRefMsg* subIfaceRef = new plObjRefMsg(subSO->GetKey(), plRefMsg::kOnCreate, 0, plObjRefMsg::kInterface); |
|
// this will send the reference immediately |
|
hsgResMgr::ResMgr()->SendRef(subWorldIface, subIfaceRef, plRefFlags::kActiveRef); |
|
} |
|
|
|
// Now, we can initialize the physical... |
|
Init(); |
|
IGetTransformGlobal(fCachedLocal2World); |
|
return true; |
|
} |
|
} |
|
break; |
|
case plRefMsg::kOnDestroy: { |
|
// our world was deleted out from under us: move to the main world |
|
// NOTE: this was not implemented even before the PXSubWorld rewrite. |
|
// not going to bother! |
|
// hsAssert(0, "Lost world"); |
|
} |
|
break; |
|
} |
|
break; |
|
|
|
case kPhysRefSndGroup: { |
|
switch (refCtxt) |
|
{ |
|
case plRefMsg::kOnCreate: |
|
case plRefMsg::kOnRequest: |
|
fSndGroup = plPhysicalSndGroup::ConvertNoRef( refMsg->GetRef() ); |
|
break; |
|
|
|
case plRefMsg::kOnDestroy: |
|
fSndGroup = nil; |
|
break; |
|
} |
|
} |
|
break; |
|
|
|
default: |
|
hsAssert(0, "Unknown ref type, who sent us this?"); |
|
break; |
|
} |
|
|
|
return true; |
|
} |
|
|
|
void plPXPhysical::IEnable(hsBool enable) |
|
{ |
|
fProps.SetBit(plSimulationInterface::kDisable, !enable); |
|
if (!enable) |
|
{ |
|
fActor->raiseActorFlag(NX_AF_DISABLE_COLLISION); |
|
if (fActor->isDynamic()) |
|
fActor->raiseBodyFlag(NX_BF_FROZEN); |
|
else |
|
plPXPhysicalControllerCore::RebuildCache(); |
|
} |
|
else |
|
{ |
|
fActor->clearActorFlag(NX_AF_DISABLE_COLLISION); |
|
|
|
// PHYSX FIXME - after re-enabling a possible detector, we need to check to see if any avatar is already in the PhysX turdy hull detector region |
|
plSimulationMgr::GetInstance()->UpdateAvatarInDetector(fWorldKey, this); |
|
|
|
if (fActor->isDynamic()) |
|
fActor->clearBodyFlag(NX_BF_FROZEN); |
|
else |
|
plPXPhysicalControllerCore::RebuildCache(); |
|
} |
|
} |
|
|
|
plPhysical& plPXPhysical::SetProperty(int prop, hsBool status) |
|
{ |
|
if (GetProperty(prop) == status) |
|
{ |
|
const char* propName = "(unknown)"; |
|
switch (prop) |
|
{ |
|
case plSimulationInterface::kDisable: propName = "kDisable"; break; |
|
case plSimulationInterface::kPinned: propName = "kPinned"; break; |
|
case plSimulationInterface::kPassive: propName = "kPassive"; break; |
|
case plSimulationInterface::kPhysAnim: propName = "kPhysAnim"; break; |
|
case plSimulationInterface::kStartInactive: propName = "kStartInactive"; break; |
|
case plSimulationInterface::kNoSynchronize: propName = "kNoSynchronize"; break; |
|
} |
|
|
|
const char* name = "(unknown)"; |
|
if (GetKey()) |
|
name = GetKeyName(); |
|
if (plSimulationMgr::fExtraProfile) |
|
plSimulationMgr::Log("Warning: Redundant physical property set (property %s, value %s) on %s", propName, status ? "true" : "false", name); |
|
} |
|
|
|
switch (prop) |
|
{ |
|
case plSimulationInterface::kDisable: |
|
IEnable(!status); |
|
break; |
|
|
|
case plSimulationInterface::kPinned: |
|
if (fActor->isDynamic()) |
|
{ |
|
// if the body is already unpinned and you unpin it again, |
|
// you'll wipe out its velocity. hence the check. |
|
hsBool current = fActor->readBodyFlag(NX_BF_FROZEN); |
|
if (status != current) |
|
{ |
|
if (status) |
|
fActor->raiseBodyFlag(NX_BF_FROZEN); |
|
else |
|
{ |
|
fActor->clearBodyFlag(NX_BF_FROZEN); |
|
LogActivate("SetProperty"); |
|
fActor->wakeUp(); |
|
} |
|
} |
|
} |
|
break; |
|
} |
|
|
|
fProps.SetBit(prop, status); |
|
|
|
return *this; |
|
} |
|
|
|
plProfile_Extern(SetTransforms); |
|
|
|
#define kMaxNegativeZPos -2000.f |
|
|
|
bool CompareMatrices(const hsMatrix44 &matA, const hsMatrix44 &matB, float tolerance) |
|
{ |
|
return |
|
(fabs(matA.fMap[0][0] - matB.fMap[0][0]) < tolerance) && |
|
(fabs(matA.fMap[0][1] - matB.fMap[0][1]) < tolerance) && |
|
(fabs(matA.fMap[0][2] - matB.fMap[0][2]) < tolerance) && |
|
(fabs(matA.fMap[0][3] - matB.fMap[0][3]) < tolerance) && |
|
|
|
(fabs(matA.fMap[1][0] - matB.fMap[1][0]) < tolerance) && |
|
(fabs(matA.fMap[1][1] - matB.fMap[1][1]) < tolerance) && |
|
(fabs(matA.fMap[1][2] - matB.fMap[1][2]) < tolerance) && |
|
(fabs(matA.fMap[1][3] - matB.fMap[1][3]) < tolerance) && |
|
|
|
(fabs(matA.fMap[2][0] - matB.fMap[2][0]) < tolerance) && |
|
(fabs(matA.fMap[2][1] - matB.fMap[2][1]) < tolerance) && |
|
(fabs(matA.fMap[2][2] - matB.fMap[2][2]) < tolerance) && |
|
(fabs(matA.fMap[2][3] - matB.fMap[2][3]) < tolerance) && |
|
|
|
(fabs(matA.fMap[3][0] - matB.fMap[3][0]) < tolerance) && |
|
(fabs(matA.fMap[3][1] - matB.fMap[3][1]) < tolerance) && |
|
(fabs(matA.fMap[3][2] - matB.fMap[3][2]) < tolerance) && |
|
(fabs(matA.fMap[3][3] - matB.fMap[3][3]) < tolerance); |
|
} |
|
|
|
// Called after the simulation has run....sends new positions to the various scene objects |
|
// *** want to do this in response to an update message.... |
|
void plPXPhysical::SendNewLocation(hsBool synchTransform, hsBool isSynchUpdate) |
|
{ |
|
// we only send if: |
|
// - the body is active or forceUpdate is on |
|
// - the mass is non-zero |
|
// - the physical is not passive |
|
hsBool bodyActive = !fActor->isSleeping(); |
|
hsBool dynamic = fActor->isDynamic(); |
|
|
|
if ((bodyActive || isSynchUpdate) && dynamic)// && fInitialTransform) |
|
{ |
|
plProfile_Inc(MaySendLocation); |
|
|
|
if (!GetProperty(plSimulationInterface::kPassive)) |
|
{ |
|
hsMatrix44 curl2w = fCachedLocal2World; |
|
// we're going to cache the transform before sending so we can recognize if it comes back |
|
IGetTransformGlobal(fCachedLocal2World); |
|
|
|
if (!CompareMatrices(curl2w, fCachedLocal2World, .0001f)) |
|
{ |
|
plProfile_Inc(LocationsSent); |
|
plProfile_BeginLap(PhysicsUpdates, GetKeyName()); |
|
|
|
// quick peek at the translation...last time it was corrupted because we applied a non-unit quaternion |
|
// hsAssert(real_finite(fCachedLocal2World.fMap[0][3]) && |
|
// real_finite(fCachedLocal2World.fMap[1][3]) && |
|
// real_finite(fCachedLocal2World.fMap[2][3]), "Bad transform outgoing"); |
|
|
|
if (fCachedLocal2World.GetTranslate().fZ < kMaxNegativeZPos) |
|
{ |
|
SimLog("Physical %s fell to %.1f (%.1f is the max). Suppressing.", GetKeyName(), fCachedLocal2World.GetTranslate().fZ, kMaxNegativeZPos); |
|
// Since this has probably been falling for a while, and thus not getting any syncs, |
|
// make sure to save it's current pos so we'll know to reset it later |
|
DirtySynchState(kSDLPhysical, plSynchedObject::kBCastToClients); |
|
IEnable(false); |
|
} |
|
|
|
hsMatrix44 w2l; |
|
fCachedLocal2World.GetInverse(&w2l); |
|
plCorrectionMsg *pCorrMsg = TRACKED_NEW plCorrectionMsg(GetObjectKey(), fCachedLocal2World, w2l, synchTransform); |
|
pCorrMsg->Send(); |
|
if (fProxyGen) |
|
fProxyGen->SetTransform(fCachedLocal2World, w2l); |
|
plProfile_EndLap(PhysicsUpdates, GetKeyName()); |
|
} |
|
} |
|
} |
|
} |
|
|
|
void plPXPhysical::ApplyHitForce() |
|
{ |
|
if (fActor && fWeWereHit) |
|
{ |
|
fActor->addForceAtPos(plPXConvert::Vector(fHitForce), plPXConvert::Point(fHitPos), NX_FORCE); |
|
fWeWereHit = false; |
|
} |
|
} |
|
|
|
|
|
void plPXPhysical::ISetTransformGlobal(const hsMatrix44& l2w) |
|
{ |
|
hsAssert(fActor->isDynamic(), "Shouldn't move a static actor"); |
|
fActor->wakeUp(); |
|
|
|
NxMat34 mat; |
|
|
|
if (fWorldKey) |
|
{ |
|
plSceneObject* so = plSceneObject::ConvertNoRef(fWorldKey->ObjectIsLoaded()); |
|
hsAssert(so, "Scene object not loaded while accessing subworld."); |
|
// physical to subworld (simulation space) |
|
hsMatrix44 p2s = so->GetCoordinateInterface()->GetWorldToLocal() * l2w; |
|
plPXConvert::Matrix(p2s, mat); |
|
if (fProxyGen) |
|
{ |
|
hsMatrix44 w2l; |
|
p2s.GetInverse(&w2l); |
|
fProxyGen->SetTransform(p2s, w2l); |
|
} |
|
} |
|
// No need to localize |
|
else |
|
{ |
|
plPXConvert::Matrix(l2w, mat); |
|
if (fProxyGen) |
|
{ |
|
hsMatrix44 w2l; |
|
l2w.GetInverse(&w2l); |
|
fProxyGen->SetTransform(l2w, w2l); |
|
} |
|
} |
|
|
|
// This used to check for the kPhysAnim flag, however animated detectors |
|
// are also kinematic but not kPhysAnim, therefore, this would break on PhysX |
|
// SDKs (yes, I'm looking at you, 2.6.4) that actually obey the ***GlobalPose |
|
// rules set forth in the SDK documentation. |
|
if (fActor->readBodyFlag(NX_BF_KINEMATIC)) |
|
fActor->moveGlobalPose(mat); |
|
else |
|
fActor->setGlobalPose(mat); |
|
} |
|
|
|
// the physical may have several parents between it and the subworld object, |
|
// but the *havok* transform is only one level away from the subworld. |
|
// to avoid any confusion about this difference, we avoid referring to the |
|
// subworld as "parent" and use, for example, "l2s" (local-to-sub) instead |
|
// of the canonical plasma "l2p" (local-to-parent) |
|
void plPXPhysical::IGetTransformGlobal(hsMatrix44& l2w) const |
|
{ |
|
plPXConvert::Matrix(fActor->getGlobalPose(), l2w); |
|
|
|
if (fWorldKey) |
|
{ |
|
plSceneObject* so = plSceneObject::ConvertNoRef(fWorldKey->ObjectIsLoaded()); |
|
hsAssert(so, "Scene object not loaded while accessing subworld."); |
|
if (so->GetCoordinateInterface()) |
|
{ |
|
const hsMatrix44& s2w = so->GetCoordinateInterface()->GetLocalToWorld(); |
|
l2w = s2w * l2w; |
|
} |
|
} |
|
} |
|
|
|
void plPXPhysical::IGetPositionSim(hsPoint3& pos) const |
|
{ |
|
pos = plPXConvert::Point(fActor->getGlobalPosition()); |
|
} |
|
|
|
void plPXPhysical::IGetRotationSim(hsQuat& rot) const |
|
{ |
|
rot = plPXConvert::Quat(fActor->getGlobalOrientationQuat()); |
|
} |
|
void plPXPhysical::ISetPositionSim(const hsPoint3& pos) |
|
{ |
|
if (GetProperty(plSimulationInterface::kPhysAnim)) |
|
fActor->moveGlobalPosition(plPXConvert::Point(pos)); |
|
else |
|
fActor->setGlobalPosition(plPXConvert::Point(pos)); |
|
} |
|
|
|
void plPXPhysical::ISetRotationSim(const hsQuat& rot) |
|
{ |
|
if (GetProperty(plSimulationInterface::kPhysAnim)) |
|
fActor->moveGlobalOrientation(plPXConvert::Quat(rot)); |
|
else |
|
fActor->setGlobalOrientation(plPXConvert::Quat(rot)); |
|
} |
|
|
|
// This form is assumed by convention to be global. |
|
void plPXPhysical::SetTransform(const hsMatrix44& l2w, const hsMatrix44& w2l, hsBool force) |
|
{ |
|
// hsAssert(real_finite(l2w.fMap[0][3]) && real_finite(l2w.fMap[1][3]) && real_finite(l2w.fMap[2][3]), "Bad transform incoming"); |
|
|
|
|
|
// make sure the physical is dynamic. |
|
// also make sure there is some difference between the matrices... |
|
// ... but not when a subworld... because the subworld maybe animating and if the object is still then it is actually moving within the subworld |
|
if (force || (fActor->isDynamic() && (fWorldKey || !CompareMatrices(l2w, fCachedLocal2World, .0001f))) ) |
|
{ |
|
ISetTransformGlobal(l2w); |
|
plProfile_Inc(SetTransforms); |
|
} |
|
else |
|
{ |
|
if ( !fActor->isDynamic() && plSimulationMgr::fExtraProfile) |
|
SimLog("Setting transform on non-dynamic: %s.", GetKeyName()); |
|
} |
|
} |
|
|
|
// GETTRANSFORM |
|
void plPXPhysical::GetTransform(hsMatrix44& l2w, hsMatrix44& w2l) |
|
{ |
|
IGetTransformGlobal(l2w); |
|
l2w.GetInverse(&w2l); |
|
} |
|
|
|
hsBool plPXPhysical::GetLinearVelocitySim(hsVector3& vel) const |
|
{ |
|
hsBool result = false; |
|
|
|
if (fActor->isDynamic()) |
|
{ |
|
vel = plPXConvert::Vector(fActor->getLinearVelocity()); |
|
result = true; |
|
} |
|
else |
|
vel.Set(0, 0, 0); |
|
|
|
return result; |
|
} |
|
|
|
void plPXPhysical::SetLinearVelocitySim(const hsVector3& vel) |
|
{ |
|
if (fActor->isDynamic()) |
|
fActor->setLinearVelocity(plPXConvert::Vector(vel)); |
|
} |
|
|
|
void plPXPhysical::ClearLinearVelocity() |
|
{ |
|
SetLinearVelocitySim(hsVector3(0, 0, 0)); |
|
} |
|
|
|
hsBool plPXPhysical::GetAngularVelocitySim(hsVector3& vel) const |
|
{ |
|
hsBool result = false; |
|
if (fActor->isDynamic()) |
|
{ |
|
vel = plPXConvert::Vector(fActor->getAngularVelocity()); |
|
result = true; |
|
} |
|
else |
|
vel.Set(0, 0, 0); |
|
|
|
return result; |
|
} |
|
|
|
void plPXPhysical::SetAngularVelocitySim(const hsVector3& vel) |
|
{ |
|
if (fActor->isDynamic()) |
|
fActor->setAngularVelocity(plPXConvert::Vector(vel)); |
|
} |
|
|
|
/////////////////////////////////////////////////////////////// |
|
// |
|
// NETWORK SYNCHRONIZATION |
|
// |
|
/////////////////////////////////////////////////////////////// |
|
|
|
plKey plPXPhysical::GetSceneNode() const |
|
{ |
|
return fSceneNode; |
|
} |
|
|
|
void plPXPhysical::SetSceneNode(plKey newNode) |
|
{ |
|
plKey oldNode = GetSceneNode(); |
|
if (oldNode == newNode) |
|
return; |
|
|
|
// If we don't do this, we get leaked keys and a crash on exit with certain clones |
|
// Note this has nothing do to with the world that the physical is in |
|
if (newNode) { |
|
plNodeRefMsg* refMsg = new plNodeRefMsg(newNode, plNodeRefMsg::kOnRequest, -1, plNodeRefMsg::kPhysical); |
|
hsgResMgr::ResMgr()->SendRef(GetKey(), refMsg, plRefFlags::kActiveRef); |
|
} |
|
if (oldNode) |
|
oldNode->Release(GetKey()); |
|
} |
|
|
|
///////////////////////////////////////////////////////////////////// |
|
// |
|
// READING AND WRITING |
|
// |
|
///////////////////////////////////////////////////////////////////// |
|
|
|
#include "plPXStream.h" |
|
|
|
void plPXPhysical::Read(hsStream* stream, hsResMgr* mgr) |
|
{ |
|
plPhysical::Read(stream, mgr); |
|
ClearMatrix(fCachedLocal2World); |
|
|
|
fRecipe.mass = stream->ReadSwapScalar(); |
|
fRecipe.friction = stream->ReadSwapScalar(); |
|
fRecipe.restitution = stream->ReadSwapScalar(); |
|
fRecipe.bounds = (plSimDefs::Bounds)stream->ReadByte(); |
|
fRecipe.group = (plSimDefs::Group)stream->ReadByte(); |
|
fRecipe.reportsOn = stream->ReadSwap32(); |
|
fLOSDBs = stream->ReadSwap16(); |
|
//hack for swim regions currently they are labeled as static av blockers |
|
if(fLOSDBs==plSimDefs::kLOSDBSwimRegion) |
|
{ |
|
fRecipe.group=plSimDefs::kGroupMax; |
|
} |
|
// |
|
fRecipe.objectKey = mgr->ReadKey(stream); |
|
fRecipe.sceneNode = mgr->ReadKey(stream); |
|
fRecipe.worldKey = mgr->ReadKeyNotifyMe(stream, new plGenRefMsg(GetKey(), plRefMsg::kOnCreate, 0, kPhysRefWorld), plRefFlags::kActiveRef); |
|
|
|
mgr->ReadKeyNotifyMe(stream, TRACKED_NEW plGenRefMsg(GetKey(), plRefMsg::kOnCreate, 0, kPhysRefSndGroup), plRefFlags::kActiveRef); |
|
|
|
hsPoint3 pos; |
|
hsQuat rot; |
|
pos.Read(stream); |
|
rot.Read(stream); |
|
rot.MakeMatrix(&fRecipe.l2s); |
|
fRecipe.l2s.SetTranslate(&pos); |
|
|
|
fProps.Read(stream); |
|
|
|
if (fRecipe.bounds == plSimDefs::kSphereBounds) |
|
{ |
|
fRecipe.radius = stream->ReadSwapScalar(); |
|
fRecipe.offset.Read(stream); |
|
} |
|
else if (fRecipe.bounds == plSimDefs::kBoxBounds) |
|
{ |
|
fRecipe.bDimensions.Read(stream); |
|
fRecipe.bOffset.Read(stream); |
|
} |
|
else |
|
{ |
|
if (fRecipe.bounds == plSimDefs::kHullBounds) |
|
fRecipe.convexMesh = IReadHull(stream); |
|
else |
|
fRecipe.triMesh = IReadTriMesh(stream); |
|
} |
|
|
|
// If we do not have a world, specified, we go ahead and init into the main world... |
|
// This will been done in MsgReceive otherwise |
|
if (!fRecipe.worldKey) |
|
Init(); |
|
|
|
hsAssert(!fProxyGen, "Already have proxy gen, double read?"); |
|
|
|
hsColorRGBA physColor; |
|
hsScalar opac = 1.0f; |
|
|
|
if (fGroup == plSimDefs::kGroupAvatar) |
|
{ |
|
// local avatar is light purple and transparent |
|
physColor.Set(.2f, .1f, .2f, 1.f); |
|
opac = 0.4f; |
|
} |
|
else if (fGroup == plSimDefs::kGroupDynamic) |
|
{ |
|
// Dynamics are red |
|
physColor.Set(1.f,0.f,0.f,1.f); |
|
} |
|
else if (fGroup == plSimDefs::kGroupDetector) |
|
{ |
|
if(!fWorldKey) |
|
{ |
|
// Detectors are blue, and transparent |
|
physColor.Set(0.f,0.f,1.f,1.f); |
|
opac = 0.3f; |
|
} |
|
else |
|
{ |
|
// subworld Detectors are green |
|
physColor.Set(0.f,1.f,0.f,1.f); |
|
opac = 0.3f; |
|
} |
|
} |
|
else if (fGroup == plSimDefs::kGroupStatic) |
|
{ |
|
if (GetProperty(plSimulationInterface::kPhysAnim)) |
|
// Statics that are animated are more reddish? |
|
physColor.Set(1.f,0.6f,0.2f,1.f); |
|
else |
|
// Statics are yellow |
|
physColor.Set(1.f,0.8f,0.2f,1.f); |
|
// if in a subworld... slightly transparent |
|
if(fRecipe.worldKey) |
|
opac = 0.6f; |
|
} |
|
else |
|
{ |
|
// don't knows are grey |
|
physColor.Set(0.6f,0.6f,0.6f,1.f); |
|
} |
|
|
|
fProxyGen = TRACKED_NEW plPhysicalProxy(hsColorRGBA().Set(0,0,0,1.f), physColor, opac); |
|
fProxyGen->Init(this); |
|
} |
|
|
|
static bool IIsNxStream(hsStream* s) |
|
{ |
|
char tag[4]; |
|
s->Read(sizeof(tag), tag); |
|
|
|
// PhysX streams begin with the magic string "NXS\x01" |
|
// Our hacked streams begin with the magic string "HSP\x01" |
|
if (memcmp("HSP\x01", tag, sizeof(tag)) == 0) |
|
return false; |
|
|
|
// We're not going to compare to see if it says NXS. We will let the PhysX SDK |
|
// worry about garbage data. Just rewind the stream back so PhysX can deal with it. |
|
s->SetPosition(s->GetPosition() - sizeof(tag)); |
|
return true; |
|
} |
|
|
|
NxConvexMesh* plPXPhysical::IReadHull(hsStream* s) |
|
{ |
|
if (IIsNxStream(s)) |
|
{ |
|
plPXStream pxs(s); |
|
return plSimulationMgr::GetInstance()->GetSDK()->createConvexMesh(pxs); |
|
} |
|
else |
|
{ |
|
std::vector<hsPoint3> verts; |
|
verts.resize(s->ReadSwap32()); |
|
for (size_t i = 0; i < verts.size(); ++i) |
|
verts[i].Read(s); |
|
|
|
// Unfortunately, the only way I know of to accomplish this is to cook to a RAM stream, |
|
// then have PhysX read the cooked data from the RAM stream. Yes, this is very sad. |
|
// I blame PhysX. It needs to die in a fiaaaaaaaaaaah |
|
hsRAMStream ram; |
|
plPXStream pxs(&ram); |
|
|
|
NxConvexMeshDesc desc; |
|
desc.numVertices = verts.size(); |
|
desc.pointStrideBytes = sizeof(hsPoint3); |
|
desc.points = &verts[0]; |
|
desc.flags = NX_CF_COMPUTE_CONVEX | NX_CF_USE_UNCOMPRESSED_NORMALS; |
|
if (!NxCookConvexMesh(desc, pxs)) |
|
{ |
|
SimLog("Failed to cook hull for '%s'", GetKeyName()); |
|
return NULL; |
|
} |
|
|
|
ram.Rewind(); |
|
return plSimulationMgr::GetInstance()->GetSDK()->createConvexMesh(pxs); |
|
} |
|
} |
|
|
|
NxTriangleMesh* plPXPhysical::IReadTriMesh(hsStream* s) |
|
{ |
|
if (IIsNxStream(s)) |
|
{ |
|
plPXStream pxs(s); |
|
return plSimulationMgr::GetInstance()->GetSDK()->createTriangleMesh(pxs); |
|
} |
|
else |
|
{ |
|
std::vector<hsPoint3> verts; |
|
verts.resize(s->ReadSwap32()); |
|
for (size_t i = 0; i < verts.size(); ++i) |
|
verts[i].Read(s); |
|
std::vector<UInt32> indices; |
|
UInt32 numTriangles = s->ReadSwap32(); |
|
indices.resize(numTriangles * 3); |
|
for (size_t i = 0; i < indices.size(); ++i) |
|
indices[i] = s->ReadSwap32(); |
|
|
|
// Unfortunately, the only way I know of to accomplish this is to cook to a RAM stream, |
|
// then have PhysX read the cooked data from the RAM stream. Yes, this is very sad. |
|
// I blame PhysX. It needs to die in a fiaaaaaaaaaaah |
|
hsRAMStream ram; |
|
plPXStream pxs(&ram); |
|
|
|
NxTriangleMeshDesc desc; |
|
desc.numVertices = verts.size(); |
|
desc.pointStrideBytes = sizeof(hsPoint3); |
|
desc.points = &verts[0]; |
|
desc.numTriangles = numTriangles; |
|
desc.triangleStrideBytes = sizeof(UInt32) * 3; |
|
desc.triangles = &indices[0]; |
|
desc.flags = 0; |
|
if (!NxCookTriangleMesh(desc, pxs)) |
|
{ |
|
SimLog("Failed to cook trimesh for '%s'", GetKeyName()); |
|
return NULL; |
|
} |
|
|
|
ram.Rewind(); |
|
return plSimulationMgr::GetInstance()->GetSDK()->createTriangleMesh(pxs); |
|
} |
|
} |
|
|
|
void plPXPhysical::Write(hsStream* stream, hsResMgr* mgr) |
|
{ |
|
plPhysical::Write(stream, mgr); |
|
|
|
hsAssert(fActor, "nil actor"); |
|
hsAssert(fActor->getNbShapes() == 1, "Can only write actors with one shape. Writing first only."); |
|
NxShape* shape = fActor->getShapes()[0]; |
|
|
|
NxMaterialIndex matIdx = shape->getMaterial(); |
|
NxScene* scene = plSimulationMgr::GetInstance()->GetScene(fWorldKey); |
|
NxMaterial* mat = scene->getMaterialFromIndex(matIdx); |
|
float friction = mat->getStaticFriction(); |
|
float restitution = mat->getRestitution(); |
|
|
|
stream->WriteSwapScalar(fActor->getMass()); |
|
stream->WriteSwapScalar(friction); |
|
stream->WriteSwapScalar(restitution); |
|
stream->WriteByte(fRecipe.bounds); |
|
stream->WriteByte(fGroup); |
|
stream->WriteSwap32(fReportsOn); |
|
stream->WriteSwap16(fLOSDBs); |
|
mgr->WriteKey(stream, fObjectKey); |
|
mgr->WriteKey(stream, fSceneNode); |
|
mgr->WriteKey(stream, fWorldKey); |
|
mgr->WriteKey(stream, fSndGroup); |
|
|
|
hsPoint3 pos; |
|
hsQuat rot; |
|
IGetPositionSim(pos); |
|
IGetRotationSim(rot); |
|
pos.Write(stream); |
|
rot.Write(stream); |
|
|
|
fProps.Write(stream); |
|
|
|
if (fRecipe.bounds == plSimDefs::kSphereBounds) |
|
{ |
|
const NxSphereShape* sphereShape = shape->isSphere(); |
|
stream->WriteSwapScalar(sphereShape->getRadius()); |
|
hsPoint3 localPos = plPXConvert::Point(sphereShape->getLocalPosition()); |
|
localPos.Write(stream); |
|
} |
|
else if (fRecipe.bounds == plSimDefs::kBoxBounds) |
|
{ |
|
const NxBoxShape* boxShape = shape->isBox(); |
|
hsPoint3 dim = plPXConvert::Point(boxShape->getDimensions()); |
|
dim.Write(stream); |
|
hsPoint3 localPos = plPXConvert::Point(boxShape->getLocalPosition()); |
|
localPos.Write(stream); |
|
} |
|
else |
|
{ |
|
if (fRecipe.bounds == plSimDefs::kHullBounds) |
|
hsAssert(shape->isConvexMesh(), "Hull shape isn't a convex mesh"); |
|
else |
|
hsAssert(shape->isTriangleMesh(), "Exact shape isn't a trimesh"); |
|
|
|
// We hide the stream we used to create this mesh away in the shape user data. |
|
// Pull it out and write it to disk. |
|
hsVectorStream* vecStream = (hsVectorStream*)shape->userData; |
|
stream->Write(vecStream->GetEOF(), vecStream->GetData()); |
|
delete vecStream; |
|
} |
|
} |
|
|
|
// |
|
// TESTING SDL |
|
// Send phys sendState msg to object's plPhysicalSDLModifier |
|
// |
|
hsBool plPXPhysical::DirtySynchState(const char* SDLStateName, UInt32 synchFlags ) |
|
{ |
|
if (GetObjectKey()) |
|
{ |
|
plSynchedObject* so=plSynchedObject::ConvertNoRef(GetObjectKey()->ObjectIsLoaded()); |
|
if (so) |
|
{ |
|
fLastSyncTime = hsTimer::GetSysSeconds(); |
|
return so->DirtySynchState(SDLStateName, synchFlags); |
|
} |
|
} |
|
|
|
return false; |
|
} |
|
|
|
void plPXPhysical::GetSyncState(hsPoint3& pos, hsQuat& rot, hsVector3& linV, hsVector3& angV) |
|
{ |
|
IGetPositionSim(pos); |
|
IGetRotationSim(rot); |
|
GetLinearVelocitySim(linV); |
|
GetAngularVelocitySim(angV); |
|
} |
|
|
|
void plPXPhysical::SetSyncState(hsPoint3* pos, hsQuat* rot, hsVector3* linV, hsVector3* angV) |
|
{ |
|
bool isLoading = plNetClientApp::GetInstance()->IsLoadingInitialAgeState(); |
|
bool isFirstIn = plNetClientApp::GetInstance()->GetJoinOrder() == 0; |
|
bool initialSync = isLoading && isFirstIn; |
|
// If the physical has fallen out of the sim, and this is initial age state, and we're |
|
// the first person in, reset it to the original position. (ie, prop the default state |
|
// we've got right now) |
|
if (pos && pos->fZ < kMaxNegativeZPos && initialSync) |
|
{ |
|
SimLog("Physical %s loaded out of range state. Forcing initial state to server.", GetKeyName()); |
|
DirtySynchState(kSDLPhysical, plSynchedObject::kBCastToClients); |
|
return; |
|
} |
|
|
|
if (pos) |
|
ISetPositionSim(*pos); |
|
if (rot) |
|
ISetRotationSim(*rot); |
|
|
|
if (linV) |
|
SetLinearVelocitySim(*linV); |
|
if (angV) |
|
SetAngularVelocitySim(*angV); |
|
// If we're loading the age, then we should ensure the objects stay asleep if they're supposed to be asleep. |
|
// NOTE: We should only do this if the objects are not at their initial locations. Otherwise, they might |
|
// sleep inside each other and explode or float randomly in midair |
|
if (isLoading && GetProperty(plSimulationInterface::kStartInactive) && !fActor->readBodyFlag(NX_BF_KINEMATIC)) { |
|
if (!pos && !rot) |
|
fActor->putToSleep(); |
|
} |
|
|
|
SendNewLocation(false, true); |
|
} |
|
|
|
void plPXPhysical::ExcludeRegionHack(hsBool cleared) |
|
{ |
|
NxShape* shape = fActor->getShapes()[0]; |
|
shape->setFlag(NX_TRIGGER_ON_ENTER, !cleared); |
|
shape->setFlag(NX_TRIGGER_ON_LEAVE, !cleared); |
|
fGroup = cleared ? plSimDefs::kGroupExcludeRegion : plSimDefs::kGroupDetector; |
|
shape->setGroup(fGroup); |
|
/*if switching a static need to inform the controller that it needs to rebuild |
|
the collision cache otherwise will still think that the detector is still static or that |
|
the static is still a detector*/ |
|
plPXPhysicalControllerCore::RebuildCache(); |
|
|
|
} |
|
hsBool plPXPhysical::OverlapWithCapsule(NxCapsule& cap) |
|
{ |
|
NxShape* shape = fActor->getShapes()[0]; |
|
return shape->checkOverlapCapsule(cap); |
|
} |
|
|
|
hsBool plPXPhysical::IsDynamic() const |
|
{ |
|
return fGroup == plSimDefs::kGroupDynamic && |
|
!GetProperty(plSimulationInterface::kPhysAnim); |
|
} |
|
|
|
#include "../plSurface/hsGMaterial.h" |
|
#include "../plSurface/plLayerInterface.h" |
|
|
|
// Some helper functions for pulling info out of a PhysX trimesh description |
|
inline hsPoint3& GetTrimeshVert(NxTriangleMeshDesc& desc, int idx) |
|
{ |
|
return *((hsPoint3*)(((char*)desc.points)+desc.pointStrideBytes*idx)); |
|
} |
|
|
|
void GetTrimeshTri(NxTriangleMeshDesc& desc, int idx, UInt16* out) |
|
{ |
|
if (hsCheckBits(desc.flags, NX_MF_16_BIT_INDICES)) |
|
{ |
|
UInt16* descTris = ((UInt16*)(((char*)desc.triangles)+desc.pointStrideBytes*idx)); |
|
out[0] = descTris[0]; |
|
out[1] = descTris[1]; |
|
out[2] = descTris[2]; |
|
} |
|
else |
|
{ |
|
UInt32* descTris = ((UInt32*)(((char*)desc.triangles)+desc.pointStrideBytes*idx)); |
|
out[0] = (UInt16)descTris[0]; |
|
out[1] = (UInt16)descTris[1]; |
|
out[2] = (UInt16)descTris[2]; |
|
} |
|
} |
|
|
|
// Some helper functions for pulling info out of a PhysX trimesh description |
|
inline hsPoint3& GetConvexVert(NxConvexMeshDesc& desc, int idx) |
|
{ |
|
return *((hsPoint3*)(((char*)desc.points)+desc.pointStrideBytes*idx)); |
|
} |
|
|
|
void GetConvexTri(NxConvexMeshDesc& desc, int idx, UInt16* out) |
|
{ |
|
if (hsCheckBits(desc.flags, NX_MF_16_BIT_INDICES)) |
|
{ |
|
UInt16* descTris = ((UInt16*)(((char*)desc.triangles)+desc.pointStrideBytes*idx)); |
|
out[0] = descTris[0]; |
|
out[1] = descTris[1]; |
|
out[2] = descTris[2]; |
|
} |
|
else |
|
{ |
|
UInt32* descTris = ((UInt32*)(((char*)desc.triangles)+desc.pointStrideBytes*idx)); |
|
out[0] = (UInt16)descTris[0]; |
|
out[1] = (UInt16)descTris[1]; |
|
out[2] = (UInt16)descTris[2]; |
|
} |
|
} |
|
|
|
// Make a visible object that can be viewed by users for debugging purposes. |
|
plDrawableSpans* plPXPhysical::CreateProxy(hsGMaterial* mat, hsTArray<UInt32>& idx, plDrawableSpans* addTo) |
|
{ |
|
plDrawableSpans* myDraw = addTo; |
|
hsMatrix44 l2w, unused; |
|
GetTransform(l2w, unused); |
|
|
|
hsBool blended = ((mat->GetLayer(0)->GetBlendFlags() & hsGMatState::kBlendMask)); |
|
|
|
NxShape* shape = fActor->getShapes()[0]; |
|
|
|
NxTriangleMeshShape* trimeshShape = shape->isTriangleMesh(); |
|
if (trimeshShape) |
|
{ |
|
NxTriangleMeshDesc desc; |
|
trimeshShape->getTriangleMesh().saveToDesc(desc); |
|
|
|
hsTArray<hsPoint3> pos; |
|
hsTArray<UInt16> tris; |
|
|
|
const int kMaxTris = 10000; |
|
const int kMaxVerts = 32000; |
|
if ((desc.numVertices < kMaxVerts) && (desc.numTriangles < kMaxTris)) |
|
{ |
|
pos.SetCount(desc.numVertices); |
|
tris.SetCount(desc.numTriangles * 3); |
|
|
|
for (int i = 0; i < desc.numVertices; i++ ) |
|
pos[i] = GetTrimeshVert(desc, i); |
|
|
|
for (int i = 0; i < desc.numTriangles; i++) |
|
GetTrimeshTri(desc, i, &tris[i*3]); |
|
|
|
myDraw = plDrawableGenerator::GenerateDrawable(pos.GetCount(), |
|
pos.AcquireArray(), |
|
nil, // normals - def to avg (smooth) norm |
|
nil, // uvws |
|
0, // uvws per vertex |
|
nil, // colors - def to white |
|
true, // do a quick fake shade |
|
nil, // optional color modulation |
|
tris.GetCount(), |
|
tris.AcquireArray(), |
|
mat, |
|
l2w, |
|
blended, |
|
&idx, |
|
myDraw); |
|
} |
|
else |
|
{ |
|
int curTri = 0; |
|
int trisToDo = desc.numTriangles; |
|
while (trisToDo > 0) |
|
{ |
|
int trisThisRound = trisToDo > kMaxTris ? kMaxTris : trisToDo; |
|
|
|
trisToDo -= trisThisRound; |
|
|
|
pos.SetCount(trisThisRound * 3); |
|
tris.SetCount(trisThisRound * 3); |
|
|
|
for (int i = 0; i < trisThisRound; i++) |
|
{ |
|
GetTrimeshTri(desc, curTri, &tris[i*3]); |
|
pos[i*3 + 0] = GetTrimeshVert(desc, tris[i*3+0]); |
|
pos[i*3 + 1] = GetTrimeshVert(desc, tris[i*3+1]); |
|
pos[i*3 + 2] = GetTrimeshVert(desc, tris[i*3+2]); |
|
|
|
curTri++; |
|
} |
|
myDraw = plDrawableGenerator::GenerateDrawable(pos.GetCount(), |
|
pos.AcquireArray(), |
|
nil, // normals - def to avg (smooth) norm |
|
nil, // uvws |
|
0, // uvws per vertex |
|
nil, // colors - def to white |
|
true, // do a quick fake shade |
|
nil, // optional color modulation |
|
tris.GetCount(), |
|
tris.AcquireArray(), |
|
mat, |
|
l2w, |
|
blended, |
|
&idx, |
|
myDraw); |
|
} |
|
} |
|
} |
|
|
|
NxConvexShape* convexShape = shape->isConvexMesh(); |
|
if (convexShape) |
|
{ |
|
NxConvexMeshDesc desc; |
|
convexShape->getConvexMesh().saveToDesc(desc); |
|
|
|
hsTArray<hsPoint3> pos; |
|
hsTArray<UInt16> tris; |
|
|
|
pos.SetCount(desc.numVertices); |
|
tris.SetCount(desc.numTriangles * 3); |
|
|
|
for (int i = 0; i < desc.numVertices; i++ ) |
|
pos[i] = GetConvexVert(desc, i); |
|
|
|
for (int i = 0; i < desc.numTriangles; i++) |
|
GetConvexTri(desc, i, &tris[i*3]); |
|
|
|
myDraw = plDrawableGenerator::GenerateDrawable(pos.GetCount(), |
|
pos.AcquireArray(), |
|
nil, // normals - def to avg (smooth) norm |
|
nil, // uvws |
|
0, // uvws per vertex |
|
nil, // colors - def to white |
|
true, // do a quick fake shade |
|
nil, // optional color modulation |
|
tris.GetCount(), |
|
tris.AcquireArray(), |
|
mat, |
|
l2w, |
|
blended, |
|
&idx, |
|
myDraw); |
|
} |
|
|
|
NxSphereShape* sphere = shape->isSphere(); |
|
if (sphere) |
|
{ |
|
float radius = sphere->getRadius(); |
|
hsPoint3 offset = plPXConvert::Point(sphere->getLocalPosition()); |
|
myDraw = plDrawableGenerator::GenerateSphericalDrawable(offset, radius, |
|
mat, l2w, blended, |
|
nil, &idx, myDraw); |
|
} |
|
|
|
NxBoxShape* box = shape->isBox(); |
|
if (box) |
|
{ |
|
hsPoint3 dim = plPXConvert::Point(box->getDimensions()); |
|
myDraw = plDrawableGenerator::GenerateBoxDrawable(dim.fX*2.f, dim.fY*2.f, dim.fZ*2.f, |
|
mat,l2w,blended, |
|
nil,&idx,myDraw); |
|
} |
|
return myDraw; |
|
}
|
|
|