#include "joint.h"
#include "body.h"
#include "contact.h"
#include "core.h"
#include "island.h"
#include "shape.h"
#include "solver.h"
#include "solver_set.h"
#include "world.h"
#include "box2d/box2d.h"
#include <stddef.h>
B2_ARRAY_SOURCE( b2Joint, b2Joint );
B2_ARRAY_SOURCE( b2JointSim, b2JointSim );
b2DistanceJointDef b2DefaultDistanceJointDef( void )
{
b2DistanceJointDef def = { 0 };
def.length = 1.0f;
def.maxLength = b2_huge;
def.internalValue = B2_SECRET_COOKIE;
return def;
}
b2MotorJointDef b2DefaultMotorJointDef( void )
{
b2MotorJointDef def = { 0 };
def.maxForce = 1.0f;
def.maxTorque = 1.0f;
def.correctionFactor = 0.3f;
def.internalValue = B2_SECRET_COOKIE;
return def;
}
b2MouseJointDef b2DefaultMouseJointDef( void )
{
b2MouseJointDef def = { 0 };
def.hertz = 4.0f;
def.dampingRatio = 1.0f;
def.maxForce = 1.0f;
def.internalValue = B2_SECRET_COOKIE;
return def;
}
b2PrismaticJointDef b2DefaultPrismaticJointDef( void )
{
b2PrismaticJointDef def = { 0 };
def.localAxisA = ( b2Vec2 ){ 1.0f, 0.0f };
def.internalValue = B2_SECRET_COOKIE;
return def;
}
b2RevoluteJointDef b2DefaultRevoluteJointDef( void )
{
b2RevoluteJointDef def = { 0 };
def.drawSize = 0.25f;
def.internalValue = B2_SECRET_COOKIE;
return def;
}
b2WeldJointDef b2DefaultWeldJointDef( void )
{
b2WeldJointDef def = { 0 };
def.internalValue = B2_SECRET_COOKIE;
return def;
}
b2WheelJointDef b2DefaultWheelJointDef( void )
{
b2WheelJointDef def = { 0 };
def.localAxisA.y = 1.0f;
def.enableSpring = true;
def.hertz = 1.0f;
def.dampingRatio = 0.7f;
def.internalValue = B2_SECRET_COOKIE;
return def;
}
static b2Joint* b2GetJointFullId( b2World* world, b2JointId jointId )
{
int id = jointId.index1 - 1;
b2Joint* joint = b2JointArray_Get( &world->joints, id );
B2_ASSERT( joint->jointId == id && joint->revision == jointId.revision );
return joint;
}
b2JointSim* b2GetJointSim( b2World* world, b2Joint* joint )
{
if ( joint->setIndex == b2_awakeSet )
{
B2_ASSERT( 0 <= joint->colorIndex && joint->colorIndex < b2_graphColorCount );
b2GraphColor* color = world->constraintGraph.colors + joint->colorIndex;
return b2JointSimArray_Get( &color->jointSims, joint->localIndex );
}
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, joint->setIndex );
return b2JointSimArray_Get( &set->jointSims, joint->localIndex );
}
b2JointSim* b2GetJointSimCheckType( b2JointId jointId, b2JointType type )
{
B2_MAYBE_UNUSED( type );
b2World* world = b2GetWorld( jointId.world0 );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return NULL;
}
b2Joint* joint = b2GetJointFullId( world, jointId );
B2_ASSERT( joint->type == type );
b2JointSim* jointSim = b2GetJointSim( world, joint );
B2_ASSERT( jointSim->type == type );
return jointSim;
}
typedef struct b2JointPair
{
b2Joint* joint;
b2JointSim* jointSim;
} b2JointPair;
static b2JointPair b2CreateJoint( b2World* world, b2Body* bodyA, b2Body* bodyB, void* userData, float drawSize, b2JointType type,
bool collideConnected )
{
int bodyIdA = bodyA->id;
int bodyIdB = bodyB->id;
int maxSetIndex = b2MaxInt( bodyA->setIndex, bodyB->setIndex );
int jointId = b2AllocId( &world->jointIdPool );
if ( jointId == world->joints.count )
{
b2JointArray_Push( &world->joints, ( b2Joint ){ 0 } );
}
b2Joint* joint = b2JointArray_Get( &world->joints, jointId );
joint->jointId = jointId;
joint->userData = userData;
joint->revision += 1;
joint->setIndex = B2_NULL_INDEX;
joint->colorIndex = B2_NULL_INDEX;
joint->localIndex = B2_NULL_INDEX;
joint->islandId = B2_NULL_INDEX;
joint->islandPrev = B2_NULL_INDEX;
joint->islandNext = B2_NULL_INDEX;
joint->drawSize = drawSize;
joint->type = type;
joint->collideConnected = collideConnected;
joint->isMarked = false;
joint->edges[0].bodyId = bodyIdA;
joint->edges[0].prevKey = B2_NULL_INDEX;
joint->edges[0].nextKey = bodyA->headJointKey;
int keyA = ( jointId << 1 ) | 0;
if ( bodyA->headJointKey != B2_NULL_INDEX )
{
b2Joint* jointA = b2JointArray_Get( &world->joints, bodyA->headJointKey >> 1 );
b2JointEdge* edgeA = jointA->edges + ( bodyA->headJointKey & 1 );
edgeA->prevKey = keyA;
}
bodyA->headJointKey = keyA;
bodyA->jointCount += 1;
joint->edges[1].bodyId = bodyIdB;
joint->edges[1].prevKey = B2_NULL_INDEX;
joint->edges[1].nextKey = bodyB->headJointKey;
int keyB = ( jointId << 1 ) | 1;
if ( bodyB->headJointKey != B2_NULL_INDEX )
{
b2Joint* jointB = b2JointArray_Get( &world->joints, bodyB->headJointKey >> 1 );
b2JointEdge* edgeB = jointB->edges + ( bodyB->headJointKey & 1 );
edgeB->prevKey = keyB;
}
bodyB->headJointKey = keyB;
bodyB->jointCount += 1;
b2JointSim* jointSim;
if ( bodyA->setIndex == b2_disabledSet || bodyB->setIndex == b2_disabledSet )
{
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, b2_disabledSet );
joint->setIndex = b2_disabledSet;
joint->localIndex = set->jointSims.count;
jointSim = b2JointSimArray_Add( &set->jointSims );
jointSim->jointId = jointId;
jointSim->bodyIdA = bodyIdA;
jointSim->bodyIdB = bodyIdB;
}
else if ( bodyA->setIndex == b2_staticSet && bodyB->setIndex == b2_staticSet )
{
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, b2_staticSet );
joint->setIndex = b2_staticSet;
joint->localIndex = set->jointSims.count;
jointSim = b2JointSimArray_Add( &set->jointSims );
jointSim->jointId = jointId;
jointSim->bodyIdA = bodyIdA;
jointSim->bodyIdB = bodyIdB;
}
else if ( bodyA->setIndex == b2_awakeSet || bodyB->setIndex == b2_awakeSet )
{
if ( maxSetIndex >= b2_firstSleepingSet )
{
b2WakeSolverSet( world, maxSetIndex );
}
joint->setIndex = b2_awakeSet;
jointSim = b2CreateJointInGraph( world, joint );
jointSim->jointId = jointId;
jointSim->bodyIdA = bodyIdA;
jointSim->bodyIdB = bodyIdB;
}
else
{
B2_ASSERT( bodyA->setIndex >= b2_firstSleepingSet || bodyB->setIndex >= b2_firstSleepingSet );
B2_ASSERT( bodyA->setIndex != b2_staticSet || bodyB->setIndex != b2_staticSet );
int setIndex = maxSetIndex;
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, setIndex );
joint->setIndex = setIndex;
joint->localIndex = set->jointSims.count;
jointSim = b2JointSimArray_Add( &set->jointSims );
jointSim->jointId = jointId;
jointSim->bodyIdA = bodyIdA;
jointSim->bodyIdB = bodyIdB;
if ( bodyA->setIndex != bodyB->setIndex && bodyA->setIndex >= b2_firstSleepingSet &&
bodyB->setIndex >= b2_firstSleepingSet )
{
b2MergeSolverSets( world, bodyA->setIndex, bodyB->setIndex );
B2_ASSERT( bodyA->setIndex == bodyB->setIndex );
setIndex = bodyA->setIndex;
b2SolverSet* mergedSet = b2SolverSetArray_Get( &world->solverSets, setIndex );
jointSim = b2JointSimArray_Get( &mergedSet->jointSims, joint->localIndex );
}
B2_ASSERT( joint->setIndex == setIndex );
}
B2_ASSERT( jointSim->jointId == jointId );
B2_ASSERT( jointSim->bodyIdA == bodyIdA );
B2_ASSERT( jointSim->bodyIdB == bodyIdB );
if ( joint->setIndex > b2_disabledSet )
{
bool mergeIslands = true;
b2LinkJoint( world, joint, mergeIslands );
}
b2ValidateSolverSets( world );
return ( b2JointPair ){ joint, jointSim };
}
static void b2DestroyContactsBetweenBodies( b2World* world, b2Body* bodyA, b2Body* bodyB )
{
int contactKey;
int otherBodyId;
if ( bodyA->contactCount < bodyB->contactCount )
{
contactKey = bodyA->headContactKey;
otherBodyId = bodyB->id;
}
else
{
contactKey = bodyB->headContactKey;
otherBodyId = bodyA->id;
}
bool wakeBodies = false;
while ( contactKey != B2_NULL_INDEX )
{
int contactId = contactKey >> 1;
int edgeIndex = contactKey & 1;
b2Contact* contact = b2ContactArray_Get( &world->contacts, contactId );
contactKey = contact->edges[edgeIndex].nextKey;
int otherEdgeIndex = edgeIndex ^ 1;
if ( contact->edges[otherEdgeIndex].bodyId == otherBodyId )
{
b2DestroyContact( world, contact, wakeBodies );
}
}
b2ValidateSolverSets( world );
}
b2JointId b2CreateDistanceJoint( b2WorldId worldId, const b2DistanceJointDef* def )
{
b2CheckDef( def );
b2World* world = b2GetWorldFromId( worldId );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return ( b2JointId ){ 0 };
}
B2_ASSERT( b2Body_IsValid( def->bodyIdA ) );
B2_ASSERT( b2Body_IsValid( def->bodyIdB ) );
B2_ASSERT( b2IsValid( def->length ) && def->length > 0.0f );
b2Body* bodyA = b2GetBodyFullId( world, def->bodyIdA );
b2Body* bodyB = b2GetBodyFullId( world, def->bodyIdB );
b2JointPair pair = b2CreateJoint( world, bodyA, bodyB, def->userData, 1.0f, b2_distanceJoint, def->collideConnected );
b2JointSim* joint = pair.jointSim;
joint->type = b2_distanceJoint;
joint->localOriginAnchorA = def->localAnchorA;
joint->localOriginAnchorB = def->localAnchorB;
b2DistanceJoint empty = { 0 };
joint->distanceJoint = empty;
joint->distanceJoint.length = b2MaxFloat( def->length, b2_linearSlop );
joint->distanceJoint.hertz = def->hertz;
joint->distanceJoint.dampingRatio = def->dampingRatio;
joint->distanceJoint.minLength = b2MaxFloat( def->minLength, b2_linearSlop );
joint->distanceJoint.maxLength = b2MaxFloat( def->minLength, def->maxLength );
joint->distanceJoint.maxMotorForce = def->maxMotorForce;
joint->distanceJoint.motorSpeed = def->motorSpeed;
joint->distanceJoint.enableSpring = def->enableSpring;
joint->distanceJoint.enableLimit = def->enableLimit;
joint->distanceJoint.enableMotor = def->enableMotor;
joint->distanceJoint.impulse = 0.0f;
joint->distanceJoint.lowerImpulse = 0.0f;
joint->distanceJoint.upperImpulse = 0.0f;
joint->distanceJoint.motorImpulse = 0.0f;
if ( def->collideConnected == false )
{
b2DestroyContactsBetweenBodies( world, bodyA, bodyB );
}
b2JointId jointId = { joint->jointId + 1, world->worldId, pair.joint->revision };
return jointId;
}
b2JointId b2CreateMotorJoint( b2WorldId worldId, const b2MotorJointDef* def )
{
b2CheckDef( def );
b2World* world = b2GetWorldFromId( worldId );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return ( b2JointId ){ 0 };
}
b2Body* bodyA = b2GetBodyFullId( world, def->bodyIdA );
b2Body* bodyB = b2GetBodyFullId( world, def->bodyIdB );
b2JointPair pair = b2CreateJoint( world, bodyA, bodyB, def->userData, 1.0f, b2_motorJoint, def->collideConnected );
b2JointSim* joint = pair.jointSim;
joint->type = b2_motorJoint;
joint->localOriginAnchorA = ( b2Vec2 ){ 0.0f, 0.0f };
joint->localOriginAnchorB = ( b2Vec2 ){ 0.0f, 0.0f };
joint->motorJoint = ( b2MotorJoint ){ 0 };
joint->motorJoint.linearOffset = def->linearOffset;
joint->motorJoint.angularOffset = def->angularOffset;
joint->motorJoint.maxForce = def->maxForce;
joint->motorJoint.maxTorque = def->maxTorque;
joint->motorJoint.correctionFactor = b2ClampFloat( def->correctionFactor, 0.0f, 1.0f );
if ( def->collideConnected == false )
{
b2DestroyContactsBetweenBodies( world, bodyA, bodyB );
}
b2JointId jointId = { joint->jointId + 1, world->worldId, pair.joint->revision };
return jointId;
}
b2JointId b2CreateMouseJoint( b2WorldId worldId, const b2MouseJointDef* def )
{
b2CheckDef( def );
b2World* world = b2GetWorldFromId( worldId );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return ( b2JointId ){ 0 };
}
b2Body* bodyA = b2GetBodyFullId( world, def->bodyIdA );
b2Body* bodyB = b2GetBodyFullId( world, def->bodyIdB );
b2Transform transformA = b2GetBodyTransformQuick( world, bodyA );
b2Transform transformB = b2GetBodyTransformQuick( world, bodyB );
b2JointPair pair = b2CreateJoint( world, bodyA, bodyB, def->userData, 1.0f, b2_mouseJoint, def->collideConnected );
b2JointSim* joint = pair.jointSim;
joint->type = b2_mouseJoint;
joint->localOriginAnchorA = b2InvTransformPoint( transformA, def->target );
joint->localOriginAnchorB = b2InvTransformPoint( transformB, def->target );
b2MouseJoint empty = { 0 };
joint->mouseJoint = empty;
joint->mouseJoint.targetA = def->target;
joint->mouseJoint.hertz = def->hertz;
joint->mouseJoint.dampingRatio = def->dampingRatio;
joint->mouseJoint.maxForce = def->maxForce;
b2JointId jointId = { joint->jointId + 1, world->worldId, pair.joint->revision };
return jointId;
}
b2JointId b2CreateRevoluteJoint( b2WorldId worldId, const b2RevoluteJointDef* def )
{
b2CheckDef( def );
b2World* world = b2GetWorldFromId( worldId );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return ( b2JointId ){ 0 };
}
b2Body* bodyA = b2GetBodyFullId( world, def->bodyIdA );
b2Body* bodyB = b2GetBodyFullId( world, def->bodyIdB );
b2JointPair pair =
b2CreateJoint( world, bodyA, bodyB, def->userData, def->drawSize, b2_revoluteJoint, def->collideConnected );
b2JointSim* joint = pair.jointSim;
joint->type = b2_revoluteJoint;
joint->localOriginAnchorA = def->localAnchorA;
joint->localOriginAnchorB = def->localAnchorB;
b2RevoluteJoint empty = { 0 };
joint->revoluteJoint = empty;
joint->revoluteJoint.referenceAngle = b2ClampFloat( def->referenceAngle, -b2_pi, b2_pi );
joint->revoluteJoint.linearImpulse = b2Vec2_zero;
joint->revoluteJoint.axialMass = 0.0f;
joint->revoluteJoint.springImpulse = 0.0f;
joint->revoluteJoint.motorImpulse = 0.0f;
joint->revoluteJoint.lowerImpulse = 0.0f;
joint->revoluteJoint.upperImpulse = 0.0f;
joint->revoluteJoint.hertz = def->hertz;
joint->revoluteJoint.dampingRatio = def->dampingRatio;
joint->revoluteJoint.lowerAngle = b2MinFloat( def->lowerAngle, def->upperAngle );
joint->revoluteJoint.upperAngle = b2MaxFloat( def->lowerAngle, def->upperAngle );
joint->revoluteJoint.lowerAngle = b2ClampFloat( joint->revoluteJoint.lowerAngle, -b2_pi, b2_pi );
joint->revoluteJoint.upperAngle = b2ClampFloat( joint->revoluteJoint.upperAngle, -b2_pi, b2_pi );
joint->revoluteJoint.maxMotorTorque = def->maxMotorTorque;
joint->revoluteJoint.motorSpeed = def->motorSpeed;
joint->revoluteJoint.enableSpring = def->enableSpring;
joint->revoluteJoint.enableLimit = def->enableLimit;
joint->revoluteJoint.enableMotor = def->enableMotor;
if ( def->collideConnected == false )
{
b2DestroyContactsBetweenBodies( world, bodyA, bodyB );
}
b2JointId jointId = { joint->jointId + 1, world->worldId, pair.joint->revision };
return jointId;
}
b2JointId b2CreatePrismaticJoint( b2WorldId worldId, const b2PrismaticJointDef* def )
{
b2CheckDef( def );
b2World* world = b2GetWorldFromId( worldId );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return ( b2JointId ){ 0 };
}
b2Body* bodyA = b2GetBodyFullId( world, def->bodyIdA );
b2Body* bodyB = b2GetBodyFullId( world, def->bodyIdB );
b2JointPair pair = b2CreateJoint( world, bodyA, bodyB, def->userData, 1.0f, b2_prismaticJoint, def->collideConnected );
b2JointSim* joint = pair.jointSim;
joint->type = b2_prismaticJoint;
joint->localOriginAnchorA = def->localAnchorA;
joint->localOriginAnchorB = def->localAnchorB;
b2PrismaticJoint empty = { 0 };
joint->prismaticJoint = empty;
joint->prismaticJoint.localAxisA = b2Normalize( def->localAxisA );
joint->prismaticJoint.referenceAngle = def->referenceAngle;
joint->prismaticJoint.impulse = b2Vec2_zero;
joint->prismaticJoint.axialMass = 0.0f;
joint->prismaticJoint.springImpulse = 0.0f;
joint->prismaticJoint.motorImpulse = 0.0f;
joint->prismaticJoint.lowerImpulse = 0.0f;
joint->prismaticJoint.upperImpulse = 0.0f;
joint->prismaticJoint.hertz = def->hertz;
joint->prismaticJoint.dampingRatio = def->dampingRatio;
joint->prismaticJoint.lowerTranslation = def->lowerTranslation;
joint->prismaticJoint.upperTranslation = def->upperTranslation;
joint->prismaticJoint.maxMotorForce = def->maxMotorForce;
joint->prismaticJoint.motorSpeed = def->motorSpeed;
joint->prismaticJoint.enableSpring = def->enableSpring;
joint->prismaticJoint.enableLimit = def->enableLimit;
joint->prismaticJoint.enableMotor = def->enableMotor;
if ( def->collideConnected == false )
{
b2DestroyContactsBetweenBodies( world, bodyA, bodyB );
}
b2JointId jointId = { joint->jointId + 1, world->worldId, pair.joint->revision };
return jointId;
}
b2JointId b2CreateWeldJoint( b2WorldId worldId, const b2WeldJointDef* def )
{
b2CheckDef( def );
b2World* world = b2GetWorldFromId( worldId );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return ( b2JointId ){ 0 };
}
b2Body* bodyA = b2GetBodyFullId( world, def->bodyIdA );
b2Body* bodyB = b2GetBodyFullId( world, def->bodyIdB );
b2JointPair pair = b2CreateJoint( world, bodyA, bodyB, def->userData, 1.0f, b2_weldJoint, def->collideConnected );
b2JointSim* joint = pair.jointSim;
joint->type = b2_weldJoint;
joint->localOriginAnchorA = def->localAnchorA;
joint->localOriginAnchorB = def->localAnchorB;
b2WeldJoint empty = { 0 };
joint->weldJoint = empty;
joint->weldJoint.referenceAngle = def->referenceAngle;
joint->weldJoint.linearHertz = def->linearHertz;
joint->weldJoint.linearDampingRatio = def->linearDampingRatio;
joint->weldJoint.angularHertz = def->angularHertz;
joint->weldJoint.angularDampingRatio = def->angularDampingRatio;
joint->weldJoint.linearImpulse = b2Vec2_zero;
joint->weldJoint.angularImpulse = 0.0f;
if ( def->collideConnected == false )
{
b2DestroyContactsBetweenBodies( world, bodyA, bodyB );
}
b2JointId jointId = { joint->jointId + 1, world->worldId, pair.joint->revision };
return jointId;
}
b2JointId b2CreateWheelJoint( b2WorldId worldId, const b2WheelJointDef* def )
{
b2CheckDef( def );
b2World* world = b2GetWorldFromId( worldId );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return ( b2JointId ){ 0 };
}
b2Body* bodyA = b2GetBodyFullId( world, def->bodyIdA );
b2Body* bodyB = b2GetBodyFullId( world, def->bodyIdB );
b2JointPair pair = b2CreateJoint( world, bodyA, bodyB, def->userData, 1.0f, b2_wheelJoint, def->collideConnected );
b2JointSim* joint = pair.jointSim;
joint->type = b2_wheelJoint;
joint->localOriginAnchorA = def->localAnchorA;
joint->localOriginAnchorB = def->localAnchorB;
joint->wheelJoint = ( b2WheelJoint ){ 0 };
joint->wheelJoint.localAxisA = b2Normalize( def->localAxisA );
joint->wheelJoint.perpMass = 0.0f;
joint->wheelJoint.axialMass = 0.0f;
joint->wheelJoint.motorImpulse = 0.0f;
joint->wheelJoint.lowerImpulse = 0.0f;
joint->wheelJoint.upperImpulse = 0.0f;
joint->wheelJoint.lowerTranslation = def->lowerTranslation;
joint->wheelJoint.upperTranslation = def->upperTranslation;
joint->wheelJoint.maxMotorTorque = def->maxMotorTorque;
joint->wheelJoint.motorSpeed = def->motorSpeed;
joint->wheelJoint.hertz = def->hertz;
joint->wheelJoint.dampingRatio = def->dampingRatio;
joint->wheelJoint.enableSpring = def->enableSpring;
joint->wheelJoint.enableLimit = def->enableLimit;
joint->wheelJoint.enableMotor = def->enableMotor;
if ( def->collideConnected == false )
{
b2DestroyContactsBetweenBodies( world, bodyA, bodyB );
}
b2JointId jointId = { joint->jointId + 1, world->worldId, pair.joint->revision };
return jointId;
}
void b2DestroyJointInternal( b2World* world, b2Joint* joint, bool wakeBodies )
{
int jointId = joint->jointId;
b2JointEdge* edgeA = joint->edges + 0;
b2JointEdge* edgeB = joint->edges + 1;
int idA = edgeA->bodyId;
int idB = edgeB->bodyId;
b2Body* bodyA = b2BodyArray_Get( &world->bodies, idA );
b2Body* bodyB = b2BodyArray_Get( &world->bodies, idB );
if ( edgeA->prevKey != B2_NULL_INDEX )
{
b2Joint* prevJoint = b2JointArray_Get( &world->joints, edgeA->prevKey >> 1 );
b2JointEdge* prevEdge = prevJoint->edges + ( edgeA->prevKey & 1 );
prevEdge->nextKey = edgeA->nextKey;
}
if ( edgeA->nextKey != B2_NULL_INDEX )
{
b2Joint* nextJoint = b2JointArray_Get( &world->joints, edgeA->nextKey >> 1 );
b2JointEdge* nextEdge = nextJoint->edges + ( edgeA->nextKey & 1 );
nextEdge->prevKey = edgeA->prevKey;
}
int edgeKeyA = ( jointId << 1 ) | 0;
if ( bodyA->headJointKey == edgeKeyA )
{
bodyA->headJointKey = edgeA->nextKey;
}
bodyA->jointCount -= 1;
if ( edgeB->prevKey != B2_NULL_INDEX )
{
b2Joint* prevJoint = b2JointArray_Get( &world->joints, edgeB->prevKey >> 1 );
b2JointEdge* prevEdge = prevJoint->edges + ( edgeB->prevKey & 1 );
prevEdge->nextKey = edgeB->nextKey;
}
if ( edgeB->nextKey != B2_NULL_INDEX )
{
b2Joint* nextJoint = b2JointArray_Get( &world->joints, edgeB->nextKey >> 1 );
b2JointEdge* nextEdge = nextJoint->edges + ( edgeB->nextKey & 1 );
nextEdge->prevKey = edgeB->prevKey;
}
int edgeKeyB = ( jointId << 1 ) | 1;
if ( bodyB->headJointKey == edgeKeyB )
{
bodyB->headJointKey = edgeB->nextKey;
}
bodyB->jointCount -= 1;
if ( joint->islandId != B2_NULL_INDEX )
{
B2_ASSERT( joint->setIndex > b2_disabledSet );
b2UnlinkJoint( world, joint );
}
else
{
B2_ASSERT( joint->setIndex <= b2_disabledSet );
}
int setIndex = joint->setIndex;
int localIndex = joint->localIndex;
if ( setIndex == b2_awakeSet )
{
b2RemoveJointFromGraph( world, joint->edges[0].bodyId, joint->edges[1].bodyId, joint->colorIndex, localIndex );
}
else
{
b2SolverSet* set = b2SolverSetArray_Get( &world->solverSets, setIndex );
int movedIndex = b2JointSimArray_RemoveSwap( &set->jointSims, localIndex );
if ( movedIndex != B2_NULL_INDEX )
{
b2JointSim* movedJointSim = set->jointSims.data + localIndex;
int movedId = movedJointSim->jointId;
b2Joint* movedJoint = b2JointArray_Get( &world->joints, movedId );
B2_ASSERT( movedJoint->localIndex == movedIndex );
movedJoint->localIndex = localIndex;
}
}
joint->setIndex = B2_NULL_INDEX;
joint->localIndex = B2_NULL_INDEX;
joint->colorIndex = B2_NULL_INDEX;
joint->jointId = B2_NULL_INDEX;
b2FreeId( &world->jointIdPool, jointId );
if ( wakeBodies )
{
b2WakeBody( world, bodyA );
b2WakeBody( world, bodyB );
}
b2ValidateSolverSets( world );
}
void b2DestroyJoint( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
B2_ASSERT( world->locked == false );
if ( world->locked )
{
return;
}
b2Joint* joint = b2GetJointFullId( world, jointId );
b2DestroyJointInternal( world, joint, true );
}
b2JointType b2Joint_GetType( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
return joint->type;
}
b2BodyId b2Joint_GetBodyA( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
return b2MakeBodyId( world, joint->edges[0].bodyId );
}
b2BodyId b2Joint_GetBodyB( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
return b2MakeBodyId( world, joint->edges[1].bodyId );
}
b2WorldId b2Joint_GetWorld( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
return ( b2WorldId ){ jointId.world0 + 1, world->revision };
}
b2Vec2 b2Joint_GetLocalAnchorA( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
b2JointSim* jointSim = b2GetJointSim( world, joint );
return jointSim->localOriginAnchorA;
}
b2Vec2 b2Joint_GetLocalAnchorB( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
b2JointSim* jointSim = b2GetJointSim( world, joint );
return jointSim->localOriginAnchorB;
}
void b2Joint_SetCollideConnected( b2JointId jointId, bool shouldCollide )
{
b2World* world = b2GetWorldLocked( jointId.world0 );
if ( world == NULL )
{
return;
}
b2Joint* joint = b2GetJointFullId( world, jointId );
if ( joint->collideConnected == shouldCollide )
{
return;
}
joint->collideConnected = shouldCollide;
b2Body* bodyA = b2BodyArray_Get( &world->bodies, joint->edges[0].bodyId );
b2Body* bodyB = b2BodyArray_Get( &world->bodies, joint->edges[1].bodyId );
if ( shouldCollide )
{
int shapeCountA = bodyA->shapeCount;
int shapeCountB = bodyB->shapeCount;
int shapeId = shapeCountA < shapeCountB ? bodyA->headShapeId : bodyB->headShapeId;
while ( shapeId != B2_NULL_INDEX )
{
b2Shape* shape = b2ShapeArray_Get( &world->shapes, shapeId );
if ( shape->proxyKey != B2_NULL_INDEX )
{
b2BufferMove( &world->broadPhase, shape->proxyKey );
}
shapeId = shape->nextShapeId;
}
}
else
{
b2DestroyContactsBetweenBodies( world, bodyA, bodyB );
}
}
bool b2Joint_GetCollideConnected( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
return joint->collideConnected;
}
void b2Joint_SetUserData( b2JointId jointId, void* userData )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
joint->userData = userData;
}
void* b2Joint_GetUserData( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
return joint->userData;
}
void b2Joint_WakeBodies( b2JointId jointId )
{
b2World* world = b2GetWorldLocked( jointId.world0 );
if ( world == NULL )
{
return;
}
b2Joint* joint = b2GetJointFullId( world, jointId );
b2Body* bodyA = b2BodyArray_Get( &world->bodies, joint->edges[0].bodyId );
b2Body* bodyB = b2BodyArray_Get( &world->bodies, joint->edges[1].bodyId );
b2WakeBody( world, bodyA );
b2WakeBody( world, bodyB );
}
extern b2Vec2 b2GetDistanceJointForce( b2World* world, b2JointSim* base );
extern b2Vec2 b2GetMotorJointForce( b2World* world, b2JointSim* base );
extern b2Vec2 b2GetMouseJointForce( b2World* world, b2JointSim* base );
extern b2Vec2 b2GetPrismaticJointForce( b2World* world, b2JointSim* base );
extern b2Vec2 b2GetRevoluteJointForce( b2World* world, b2JointSim* base );
extern b2Vec2 b2GetWeldJointForce( b2World* world, b2JointSim* base );
extern b2Vec2 b2GetWheelJointForce( b2World* world, b2JointSim* base );
b2Vec2 b2Joint_GetConstraintForce( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
b2JointSim* base = b2GetJointSim( world, joint );
switch ( joint->type )
{
case b2_distanceJoint:
return b2GetDistanceJointForce( world, base );
case b2_motorJoint:
return b2GetMotorJointForce( world, base );
case b2_mouseJoint:
return b2GetMouseJointForce( world, base );
case b2_prismaticJoint:
return b2GetPrismaticJointForce( world, base );
case b2_revoluteJoint:
return b2GetRevoluteJointForce( world, base );
case b2_weldJoint:
return b2GetWeldJointForce( world, base );
case b2_wheelJoint:
return b2GetWheelJointForce( world, base );
default:
B2_ASSERT( false );
return b2Vec2_zero;
}
}
extern float b2GetMotorJointTorque( b2World* world, b2JointSim* base );
extern float b2GetMouseJointTorque( b2World* world, b2JointSim* base );
extern float b2GetPrismaticJointTorque( b2World* world, b2JointSim* base );
extern float b2GetRevoluteJointTorque( b2World* world, b2JointSim* base );
extern float b2GetWeldJointTorque( b2World* world, b2JointSim* base );
extern float b2GetWheelJointTorque( b2World* world, b2JointSim* base );
float b2Joint_GetConstraintTorque( b2JointId jointId )
{
b2World* world = b2GetWorld( jointId.world0 );
b2Joint* joint = b2GetJointFullId( world, jointId );
b2JointSim* base = b2GetJointSim( world, joint );
switch ( joint->type )
{
case b2_distanceJoint:
return 0.0f;
case b2_motorJoint:
return b2GetMotorJointTorque( world, base );
case b2_mouseJoint:
return b2GetMouseJointTorque( world, base );
case b2_prismaticJoint:
return b2GetPrismaticJointTorque( world, base );
case b2_revoluteJoint:
return b2GetRevoluteJointTorque( world, base );
case b2_weldJoint:
return b2GetWeldJointTorque( world, base );
case b2_wheelJoint:
return b2GetWheelJointTorque( world, base );
default:
B2_ASSERT( false );
return 0.0f;
}
}
extern void b2PrepareDistanceJoint( b2JointSim* base, b2StepContext* context );
extern void b2PrepareMotorJoint( b2JointSim* base, b2StepContext* context );
extern void b2PrepareMouseJoint( b2JointSim* base, b2StepContext* context );
extern void b2PreparePrismaticJoint( b2JointSim* base, b2StepContext* context );
extern void b2PrepareRevoluteJoint( b2JointSim* base, b2StepContext* context );
extern void b2PrepareWeldJoint( b2JointSim* base, b2StepContext* context );
extern void b2PrepareWheelJoint( b2JointSim* base, b2StepContext* context );
void b2PrepareJoint( b2JointSim* joint, b2StepContext* context )
{
switch ( joint->type )
{
case b2_distanceJoint:
b2PrepareDistanceJoint( joint, context );
break;
case b2_motorJoint:
b2PrepareMotorJoint( joint, context );
break;
case b2_mouseJoint:
b2PrepareMouseJoint( joint, context );
break;
case b2_prismaticJoint:
b2PreparePrismaticJoint( joint, context );
break;
case b2_revoluteJoint:
b2PrepareRevoluteJoint( joint, context );
break;
case b2_weldJoint:
b2PrepareWeldJoint( joint, context );
break;
case b2_wheelJoint:
b2PrepareWheelJoint( joint, context );
break;
default:
B2_ASSERT( false );
}
}
extern void b2WarmStartDistanceJoint( b2JointSim* base, b2StepContext* context );
extern void b2WarmStartMotorJoint( b2JointSim* base, b2StepContext* context );
extern void b2WarmStartMouseJoint( b2JointSim* base, b2StepContext* context );
extern void b2WarmStartPrismaticJoint( b2JointSim* base, b2StepContext* context );
extern void b2WarmStartRevoluteJoint( b2JointSim* base, b2StepContext* context );
extern void b2WarmStartWeldJoint( b2JointSim* base, b2StepContext* context );
extern void b2WarmStartWheelJoint( b2JointSim* base, b2StepContext* context );
void b2WarmStartJoint( b2JointSim* joint, b2StepContext* context )
{
switch ( joint->type )
{
case b2_distanceJoint:
b2WarmStartDistanceJoint( joint, context );
break;
case b2_motorJoint:
b2WarmStartMotorJoint( joint, context );
break;
case b2_mouseJoint:
b2WarmStartMouseJoint( joint, context );
break;
case b2_prismaticJoint:
b2WarmStartPrismaticJoint( joint, context );
break;
case b2_revoluteJoint:
b2WarmStartRevoluteJoint( joint, context );
break;
case b2_weldJoint:
b2WarmStartWeldJoint( joint, context );
break;
case b2_wheelJoint:
b2WarmStartWheelJoint( joint, context );
break;
default:
B2_ASSERT( false );
}
}
extern void b2SolveDistanceJoint( b2JointSim* base, b2StepContext* context, bool useBias );
extern void b2SolveMotorJoint( b2JointSim* base, b2StepContext* context, bool useBias );
extern void b2SolveMouseJoint( b2JointSim* base, b2StepContext* context );
extern void b2SolvePrismaticJoint( b2JointSim* base, b2StepContext* context, bool useBias );
extern void b2SolveRevoluteJoint( b2JointSim* base, b2StepContext* context, bool useBias );
extern void b2SolveWeldJoint( b2JointSim* base, b2StepContext* context, bool useBias );
extern void b2SolveWheelJoint( b2JointSim* base, b2StepContext* context, bool useBias );
void b2SolveJoint( b2JointSim* joint, b2StepContext* context, bool useBias )
{
switch ( joint->type )
{
case b2_distanceJoint:
b2SolveDistanceJoint( joint, context, useBias );
break;
case b2_motorJoint:
b2SolveMotorJoint( joint, context, useBias );
break;
case b2_mouseJoint:
b2SolveMouseJoint( joint, context );
break;
case b2_prismaticJoint:
b2SolvePrismaticJoint( joint, context, useBias );
break;
case b2_revoluteJoint:
b2SolveRevoluteJoint( joint, context, useBias );
break;
case b2_weldJoint:
b2SolveWeldJoint( joint, context, useBias );
break;
case b2_wheelJoint:
b2SolveWheelJoint( joint, context, useBias );
break;
default:
B2_ASSERT( false );
}
}
void b2PrepareOverflowJoints( b2StepContext* context )
{
b2TracyCZoneNC( prepare_joints, "PrepJoints", b2_colorOldLace, true );
b2ConstraintGraph* graph = context->graph;
b2JointSim* joints = graph->colors[b2_overflowIndex].jointSims.data;
int jointCount = graph->colors[b2_overflowIndex].jointSims.count;
for ( int i = 0; i < jointCount; ++i )
{
b2JointSim* joint = joints + i;
b2PrepareJoint( joint, context );
}
b2TracyCZoneEnd( prepare_joints );
}
void b2WarmStartOverflowJoints( b2StepContext* context )
{
b2TracyCZoneNC( prepare_joints, "PrepJoints", b2_colorOldLace, true );
b2ConstraintGraph* graph = context->graph;
b2JointSim* joints = graph->colors[b2_overflowIndex].jointSims.data;
int jointCount = graph->colors[b2_overflowIndex].jointSims.count;
for ( int i = 0; i < jointCount; ++i )
{
b2JointSim* joint = joints + i;
b2WarmStartJoint( joint, context );
}
b2TracyCZoneEnd( prepare_joints );
}
void b2SolveOverflowJoints( b2StepContext* context, bool useBias )
{
b2TracyCZoneNC( solve_joints, "SolveJoints", b2_colorLemonChiffon, true );
b2ConstraintGraph* graph = context->graph;
b2JointSim* joints = graph->colors[b2_overflowIndex].jointSims.data;
int jointCount = graph->colors[b2_overflowIndex].jointSims.count;
for ( int i = 0; i < jointCount; ++i )
{
b2JointSim* joint = joints + i;
b2SolveJoint( joint, context, useBias );
}
b2TracyCZoneEnd( solve_joints );
}
extern void b2DrawDistanceJoint( b2DebugDraw* draw, b2JointSim* base, b2Transform transformA, b2Transform transformB );
extern void b2DrawPrismaticJoint( b2DebugDraw* draw, b2JointSim* base, b2Transform transformA, b2Transform transformB );
extern void b2DrawRevoluteJoint( b2DebugDraw* draw, b2JointSim* base, b2Transform transformA, b2Transform transformB,
float drawSize );
extern void b2DrawWheelJoint( b2DebugDraw* draw, b2JointSim* base, b2Transform transformA, b2Transform transformB );
void b2DrawJoint( b2DebugDraw* draw, b2World* world, b2Joint* joint )
{
b2Body* bodyA = b2BodyArray_Get( &world->bodies, joint->edges[0].bodyId );
b2Body* bodyB = b2BodyArray_Get( &world->bodies, joint->edges[1].bodyId );
if ( bodyA->setIndex == b2_disabledSet || bodyB->setIndex == b2_disabledSet )
{
return;
}
b2JointSim* jointSim = b2GetJointSim( world, joint );
b2Transform transformA = b2GetBodyTransformQuick( world, bodyA );
b2Transform transformB = b2GetBodyTransformQuick( world, bodyB );
b2Vec2 pA = b2TransformPoint( transformA, jointSim->localOriginAnchorA );
b2Vec2 pB = b2TransformPoint( transformB, jointSim->localOriginAnchorB );
b2HexColor color = b2_colorDarkSeaGreen;
switch ( joint->type )
{
case b2_distanceJoint:
b2DrawDistanceJoint( draw, jointSim, transformA, transformB );
break;
case b2_mouseJoint:
{
b2Vec2 target = jointSim->mouseJoint.targetA;
b2HexColor c1 = b2_colorGreen;
draw->DrawPoint( target, 4.0f, c1, draw->context );
draw->DrawPoint( pB, 4.0f, c1, draw->context );
b2HexColor c2 = b2_colorGray8;
draw->DrawSegment( target, pB, c2, draw->context );
}
break;
case b2_prismaticJoint:
b2DrawPrismaticJoint( draw, jointSim, transformA, transformB );
break;
case b2_revoluteJoint:
b2DrawRevoluteJoint( draw, jointSim, transformA, transformB, joint->drawSize );
break;
case b2_wheelJoint:
b2DrawWheelJoint( draw, jointSim, transformA, transformB );
break;
default:
draw->DrawSegment( transformA.p, pA, color, draw->context );
draw->DrawSegment( pA, pB, color, draw->context );
draw->DrawSegment( transformB.p, pB, color, draw->context );
}
if ( draw->drawGraphColors )
{
b2HexColor colors[b2_graphColorCount] = { b2_colorRed, b2_colorOrange, b2_colorYellow, b2_colorGreen,
b2_colorCyan, b2_colorBlue, b2_colorViolet, b2_colorPink,
b2_colorChocolate, b2_colorGoldenrod, b2_colorCoral, b2_colorBlack };
int colorIndex = joint->colorIndex;
if ( colorIndex != B2_NULL_INDEX )
{
b2Vec2 p = b2Lerp( pA, pB, 0.5f );
draw->DrawPoint( p, 5.0f, colors[colorIndex], draw->context );
}
}
}