mirror of https://github.com/axmolengine/axmol.git
issue #503, chippmunk library is upgraded from 5.1 to 5.3.4, which is usd in cocos2d-iphone 0.99.5 & 1.0.0-rc
This commit is contained in:
parent
57a14e90c9
commit
948c6d750e
|
@ -26,7 +26,10 @@ src/cpHashSet.c \
|
||||||
src/cpPolyShape.c \
|
src/cpPolyShape.c \
|
||||||
src/cpShape.c \
|
src/cpShape.c \
|
||||||
src/cpSpace.c \
|
src/cpSpace.c \
|
||||||
|
src/cpSpaceComponent.c \
|
||||||
src/cpSpaceHash.c \
|
src/cpSpaceHash.c \
|
||||||
|
src/cpSpaceQuery.c \
|
||||||
|
src/cpSpaceStep.c \
|
||||||
src/cpVect.c
|
src/cpVect.c
|
||||||
|
|
||||||
LOCAL_C_INCLUDES := $(LOCAL_PATH)/include/chipmunk
|
LOCAL_C_INCLUDES := $(LOCAL_PATH)/include/chipmunk
|
||||||
|
|
|
@ -26,6 +26,16 @@
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifndef CP_ALLOW_PRIVATE_ACCESS
|
||||||
|
#define CP_ALLOW_PRIVATE_ACCESS 1
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if CP_ALLOW_PRIVATE_ACCESS == 1
|
||||||
|
#define CP_PRIVATE(symbol) symbol
|
||||||
|
#else
|
||||||
|
#define CP_PRIVATE(symbol) symbol##_private
|
||||||
|
#endif
|
||||||
|
|
||||||
void cpMessage(const char *message, const char *condition, const char *file, int line, int isError);
|
void cpMessage(const char *message, const char *condition, const char *file, int line, int isError);
|
||||||
#ifdef NDEBUG
|
#ifdef NDEBUG
|
||||||
#define cpAssertWarn(condition, message)
|
#define cpAssertWarn(condition, message)
|
||||||
|
@ -40,43 +50,7 @@ void cpMessage(const char *message, const char *condition, const char *file, int
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "chipmunk_types.h"
|
#include "chipmunk_types.h"
|
||||||
|
|
||||||
static inline cpFloat
|
|
||||||
cpfmax(cpFloat a, cpFloat b)
|
|
||||||
{
|
|
||||||
return (a > b) ? a : b;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline cpFloat
|
|
||||||
cpfmin(cpFloat a, cpFloat b)
|
|
||||||
{
|
|
||||||
return (a < b) ? a : b;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline cpFloat
|
|
||||||
cpfabs(cpFloat n)
|
|
||||||
{
|
|
||||||
return (n < 0) ? -n : n;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline cpFloat
|
|
||||||
cpfclamp(cpFloat f, cpFloat min, cpFloat max)
|
|
||||||
{
|
|
||||||
return cpfmin(cpfmax(f, min), max);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline cpFloat
|
|
||||||
cpflerp(cpFloat f1, cpFloat f2, cpFloat t)
|
|
||||||
{
|
|
||||||
return f1*(1.0f - t) + f2*t;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline cpFloat
|
|
||||||
cpflerpconst(cpFloat f1, cpFloat f2, cpFloat d)
|
|
||||||
{
|
|
||||||
return f1 + cpfclamp(f2 - f1, -d, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifndef INFINITY
|
#ifndef INFINITY
|
||||||
#ifdef _MSC_VER
|
#ifdef _MSC_VER
|
||||||
union MSVC_EVIL_FLOAT_HACK
|
union MSVC_EVIL_FLOAT_HACK
|
||||||
|
@ -97,7 +71,7 @@ cpflerpconst(cpFloat f1, cpFloat f2, cpFloat d)
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Maximum allocated size for various Chipmunk buffer sizes
|
// Maximum allocated size for various Chipmunk buffers
|
||||||
#define CP_BUFFER_BYTES (32*1024)
|
#define CP_BUFFER_BYTES (32*1024)
|
||||||
|
|
||||||
#define cpmalloc malloc
|
#define cpmalloc malloc
|
||||||
|
@ -107,11 +81,11 @@ cpflerpconst(cpFloat f1, cpFloat f2, cpFloat d)
|
||||||
|
|
||||||
#include "cpVect.h"
|
#include "cpVect.h"
|
||||||
#include "cpBB.h"
|
#include "cpBB.h"
|
||||||
#include "cpBody.h"
|
|
||||||
#include "cpArray.h"
|
#include "cpArray.h"
|
||||||
#include "cpHashSet.h"
|
#include "cpHashSet.h"
|
||||||
#include "cpSpaceHash.h"
|
#include "cpSpaceHash.h"
|
||||||
|
|
||||||
|
#include "cpBody.h"
|
||||||
#include "cpShape.h"
|
#include "cpShape.h"
|
||||||
#include "cpPolyShape.h"
|
#include "cpPolyShape.h"
|
||||||
|
|
||||||
|
@ -125,24 +99,65 @@ cpflerpconst(cpFloat f1, cpFloat f2, cpFloat d)
|
||||||
#define CP_HASH_COEF (3344921057ul)
|
#define CP_HASH_COEF (3344921057ul)
|
||||||
#define CP_HASH_PAIR(A, B) ((cpHashValue)(A)*CP_HASH_COEF ^ (cpHashValue)(B)*CP_HASH_COEF)
|
#define CP_HASH_PAIR(A, B) ((cpHashValue)(A)*CP_HASH_COEF ^ (cpHashValue)(B)*CP_HASH_COEF)
|
||||||
|
|
||||||
extern char *cpVersionString;
|
extern const char *cpVersionString;
|
||||||
void cpInitChipmunk(void);
|
void cpInitChipmunk(void);
|
||||||
|
|
||||||
// Calculate the moment of inertia for a circle, r1 and r2 are the inner and outer diameters.
|
/**
|
||||||
// (A solid circle has an inner diameter of 0)
|
Calculate the moment of inertia for a circle.
|
||||||
|
r1 and r2 are the inner and outer diameters. A solid circle has an inner diameter of 0.
|
||||||
|
*/
|
||||||
cpFloat cpMomentForCircle(cpFloat m, cpFloat r1, cpFloat r2, cpVect offset);
|
cpFloat cpMomentForCircle(cpFloat m, cpFloat r1, cpFloat r2, cpVect offset);
|
||||||
|
|
||||||
// Calculate the moment of inertia for a line segment. (beveling radius not supported)
|
/**
|
||||||
|
Calculate area of a hollow circle.
|
||||||
|
*/
|
||||||
|
cpFloat cpAreaForCircle(cpFloat r1, cpFloat r2);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Calculate the moment of inertia for a line segment.
|
||||||
|
Beveling radius is not supported.
|
||||||
|
*/
|
||||||
cpFloat cpMomentForSegment(cpFloat m, cpVect a, cpVect b);
|
cpFloat cpMomentForSegment(cpFloat m, cpVect a, cpVect b);
|
||||||
|
|
||||||
// Calculate the moment of inertia for a solid polygon shape.
|
/**
|
||||||
cpFloat cpMomentForPoly(cpFloat m, int numVerts, cpVect *verts, cpVect offset);
|
Calculate the area of a fattened (capsule shaped) line segment.
|
||||||
|
*/
|
||||||
|
cpFloat cpAreaForSegment(cpVect a, cpVect b, cpFloat r);
|
||||||
|
|
||||||
// Calculate the moment of inertia for a solid box.
|
/**
|
||||||
|
Calculate the moment of inertia for a solid polygon shape assuming it's center of gravity is at it's centroid. The offset is added to each vertex.
|
||||||
|
*/
|
||||||
|
cpFloat cpMomentForPoly(cpFloat m, int numVerts, const cpVect *verts, cpVect offset);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Calculate the signed area of a polygon.
|
||||||
|
*/
|
||||||
|
cpFloat cpAreaForPoly(const int numVerts, const cpVect *verts);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Calculate the natural centroid of a polygon.
|
||||||
|
*/
|
||||||
|
cpVect cpCentroidForPoly(const int numVerts, const cpVect *verts);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Center the polygon on the origin. (Subtracts the centroid of the polygon from each vertex)
|
||||||
|
*/
|
||||||
|
void cpRecenterPoly(const int numVerts, cpVect *verts);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Calculate the moment of inertia for a solid box.
|
||||||
|
*/
|
||||||
cpFloat cpMomentForBox(cpFloat m, cpFloat width, cpFloat height);
|
cpFloat cpMomentForBox(cpFloat m, cpFloat width, cpFloat height);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline cpVect operator *(const cpVect v, const cpFloat s){return cpvmult(v, s);}
|
||||||
|
static inline cpVect operator +(const cpVect v1, const cpVect v2){return cpvadd(v1, v2);}
|
||||||
|
static inline cpVect operator -(const cpVect v1, const cpVect v2){return cpvsub(v1, v2);}
|
||||||
|
static inline cpBool operator ==(const cpVect v1, const cpVect v2){return cpveql(v1, v2);}
|
||||||
|
static inline cpVect operator -(const cpVect v){return cpvneg(v);}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
MAKE_REF(cpv); // makes a variable named _cpv that contains the function pointer for cpv()
|
MAKE_REF(cpv); // makes a variable named _cpv that contains the function pointer for cpv()
|
||||||
|
MAKE_REF(cpveql);
|
||||||
MAKE_REF(cpvadd);
|
MAKE_REF(cpvadd);
|
||||||
MAKE_REF(cpvneg);
|
MAKE_REF(cpvneg);
|
||||||
MAKE_REF(cpvsub);
|
MAKE_REF(cpvsub);
|
||||||
|
@ -33,8 +34,8 @@ MAKE_REF(cpvnormalize_safe);
|
||||||
MAKE_REF(cpvclamp);
|
MAKE_REF(cpvclamp);
|
||||||
MAKE_REF(cpvlerpconst);
|
MAKE_REF(cpvlerpconst);
|
||||||
MAKE_REF(cpvdist);
|
MAKE_REF(cpvdist);
|
||||||
MAKE_REF(cpvnear);
|
|
||||||
MAKE_REF(cpvdistsq);
|
MAKE_REF(cpvdistsq);
|
||||||
|
MAKE_REF(cpvnear);
|
||||||
|
|
||||||
MAKE_REF(cpBBNew);
|
MAKE_REF(cpBBNew);
|
||||||
MAKE_REF(cpBBintersects);
|
MAKE_REF(cpBBintersects);
|
||||||
|
@ -46,10 +47,18 @@ MAKE_REF(cpBBexpand);
|
||||||
MAKE_REF(cpBodyWorld2Local);
|
MAKE_REF(cpBodyWorld2Local);
|
||||||
MAKE_REF(cpBodyLocal2World);
|
MAKE_REF(cpBodyLocal2World);
|
||||||
MAKE_REF(cpBodyApplyImpulse);
|
MAKE_REF(cpBodyApplyImpulse);
|
||||||
|
MAKE_REF(cpBodyIsSleeping);
|
||||||
|
MAKE_REF(cpBodyIsRogue);
|
||||||
|
MAKE_REF(cpBodyKineticEnergy);
|
||||||
|
|
||||||
MAKE_REF(cpArbiterIsFirstContact);
|
MAKE_REF(cpArbiterIsFirstContact);
|
||||||
MAKE_REF(cpArbiterGetShapes);
|
MAKE_REF(cpArbiterGetShapes);
|
||||||
MAKE_REF(cpArbiterGetNormal);
|
MAKE_REF(cpArbiterGetNormal);
|
||||||
MAKE_REF(cpArbiterGetPoint);
|
MAKE_REF(cpArbiterGetPoint);
|
||||||
|
|
||||||
|
MAKE_REF(cpConstraintGetImpulse);
|
||||||
|
|
||||||
|
MAKE_REF(cpSegmentQueryHitPoint);
|
||||||
|
MAKE_REF(cpSegmentQueryHitDist);
|
||||||
|
|
||||||
#endif // _CHIPMUNK_FFI_H_
|
#endif // _CHIPMUNK_FFI_H_
|
|
@ -3,14 +3,29 @@
|
||||||
|
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
#ifndef AIRPLAYUSECHIPMUNK
|
#ifndef AIRPLAYUSECHIPMUNK
|
||||||
#import "TargetConditionals.h"
|
#import "TargetConditionals.h"
|
||||||
#endif
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Use single precision floats on the iPhone.
|
// dont' use CGPoints in cocos2d-x to make your code multi-platform
|
||||||
#if TARGET_OS_IPHONE==1
|
// #if (defined TARGET_OS_IPHONE) && (!defined CP_USE_CGPOINTS)
|
||||||
#define CP_USE_DOUBLES 0
|
// #define CP_USE_CGPOINTS
|
||||||
#else
|
// #endif
|
||||||
|
|
||||||
|
#ifdef CP_USE_CGPOINTS
|
||||||
|
#if TARGET_OS_IPHONE
|
||||||
|
#import <CoreGraphics/CGGeometry.h>
|
||||||
|
#elif TARGET_OS_MAC
|
||||||
|
#import <ApplicationServices/ApplicationServices.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__LP64__) && __LP64__
|
||||||
|
#define CP_USE_DOUBLES 1
|
||||||
|
#else
|
||||||
|
#define CP_USE_DOUBLES 0
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef CP_USE_DOUBLES
|
||||||
// use doubles by default for higher precision
|
// use doubles by default for higher precision
|
||||||
#define CP_USE_DOUBLES 1
|
#define CP_USE_DOUBLES 1
|
||||||
#endif
|
#endif
|
||||||
|
@ -41,17 +56,67 @@
|
||||||
#define cpfceil ceilf
|
#define cpfceil ceilf
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
//#if TARGET_OS_IPHONE
|
static inline cpFloat
|
||||||
// CCPoints are structurally the same, and allow
|
cpfmax(cpFloat a, cpFloat b)
|
||||||
// easy interoperability with other iPhone libraries
|
{
|
||||||
//#import <CoreGraphics/CCGeometry.h>
|
return (a > b) ? a : b;
|
||||||
//typedef CCPoint cpVect;
|
}
|
||||||
//#else
|
|
||||||
|
static inline cpFloat
|
||||||
|
cpfmin(cpFloat a, cpFloat b)
|
||||||
|
{
|
||||||
|
return (a < b) ? a : b;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline cpFloat
|
||||||
|
cpfabs(cpFloat n)
|
||||||
|
{
|
||||||
|
return (n < 0) ? -n : n;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline cpFloat
|
||||||
|
cpfclamp(cpFloat f, cpFloat min, cpFloat max)
|
||||||
|
{
|
||||||
|
return cpfmin(cpfmax(f, min), max);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline cpFloat
|
||||||
|
cpflerp(cpFloat f1, cpFloat f2, cpFloat t)
|
||||||
|
{
|
||||||
|
return f1*(1.0f - t) + f2*t;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline cpFloat
|
||||||
|
cpflerpconst(cpFloat f1, cpFloat f2, cpFloat d)
|
||||||
|
{
|
||||||
|
return f1 + cpfclamp(f2 - f1, -d, d);
|
||||||
|
}
|
||||||
|
|
||||||
|
// CGPoints are structurally the same, and allow
|
||||||
|
// easy interoperability with other Cocoa libraries
|
||||||
|
#ifdef CP_USE_CGPOINTS
|
||||||
|
typedef CGPoint cpVect;
|
||||||
|
#else
|
||||||
typedef struct cpVect{cpFloat x,y;} cpVect;
|
typedef struct cpVect{cpFloat x,y;} cpVect;
|
||||||
//#endif
|
#endif
|
||||||
|
|
||||||
typedef unsigned int cpHashValue;
|
typedef unsigned int cpHashValue;
|
||||||
|
|
||||||
|
// Oh C, how we love to define our own boolean types to get compiler compatibility
|
||||||
|
#ifdef CP_BOOL_TYPE
|
||||||
|
typedef CP_BOOL_TYPE cpBool;
|
||||||
|
#else
|
||||||
|
typedef int cpBool;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef cpTrue
|
||||||
|
#define cpTrue 1
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef cpFalse
|
||||||
|
#define cpFalse 0
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef CP_DATA_POINTER_TYPE
|
#ifdef CP_DATA_POINTER_TYPE
|
||||||
typedef CP_DATA_POINTER_TYPE cpDataPointer;
|
typedef CP_DATA_POINTER_TYPE cpDataPointer;
|
||||||
#else
|
#else
|
||||||
|
@ -76,6 +141,12 @@ typedef unsigned int cpHashValue;
|
||||||
typedef unsigned int cpLayers;
|
typedef unsigned int cpLayers;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef CP_TIMESTAMP_TYPE
|
||||||
|
typedef CP_TIMESTAMP_TYPE cpTimestamp;
|
||||||
|
#else
|
||||||
|
typedef unsigned int cpTimestamp;
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifndef CP_NO_GROUP
|
#ifndef CP_NO_GROUP
|
||||||
#define CP_NO_GROUP ((cpGroup)0)
|
#define CP_NO_GROUP ((cpGroup)0)
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -39,7 +39,7 @@ typedef struct cpConstraintClass {
|
||||||
|
|
||||||
|
|
||||||
typedef struct cpConstraint {
|
typedef struct cpConstraint {
|
||||||
const cpConstraintClass *klass;
|
CP_PRIVATE(const cpConstraintClass *klass);
|
||||||
|
|
||||||
cpBody *a, *b;
|
cpBody *a, *b;
|
||||||
cpFloat maxForce;
|
cpFloat maxForce;
|
||||||
|
@ -56,14 +56,26 @@ typedef cpConstraint cpJoint;
|
||||||
void cpConstraintDestroy(cpConstraint *constraint);
|
void cpConstraintDestroy(cpConstraint *constraint);
|
||||||
void cpConstraintFree(cpConstraint *constraint);
|
void cpConstraintFree(cpConstraint *constraint);
|
||||||
|
|
||||||
|
static inline void
|
||||||
|
cpConstraintActivateBodies(cpConstraint *constraint)
|
||||||
|
{
|
||||||
|
cpBody *a = constraint->a; if(a) cpBodyActivate(a);
|
||||||
|
cpBody *b = constraint->b; if(b) cpBodyActivate(b);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline cpFloat
|
||||||
|
cpConstraintGetImpulse(cpConstraint *constraint)
|
||||||
|
{
|
||||||
|
return constraint->CP_PRIVATE(klass)->getImpulse(constraint);
|
||||||
|
}
|
||||||
|
|
||||||
#define cpConstraintCheckCast(constraint, struct) \
|
#define cpConstraintCheckCast(constraint, struct) \
|
||||||
cpAssert(constraint->klass == struct##GetClass(), "Constraint is not a "#struct);
|
cpAssert(constraint->CP_PRIVATE(klass) == struct##GetClass(), "Constraint is not a "#struct);
|
||||||
|
|
||||||
|
|
||||||
#define CP_DefineConstraintGetter(struct, type, member, name) \
|
#define CP_DefineConstraintGetter(struct, type, member, name) \
|
||||||
static inline type \
|
static inline type \
|
||||||
struct##Get##name(cpConstraint *constraint){ \
|
struct##Get##name(const cpConstraint *constraint){ \
|
||||||
cpConstraintCheckCast(constraint, struct); \
|
cpConstraintCheckCast(constraint, struct); \
|
||||||
return ((struct *)constraint)->member; \
|
return ((struct *)constraint)->member; \
|
||||||
} \
|
} \
|
||||||
|
@ -72,6 +84,7 @@ struct##Get##name(cpConstraint *constraint){ \
|
||||||
static inline void \
|
static inline void \
|
||||||
struct##Set##name(cpConstraint *constraint, type value){ \
|
struct##Set##name(cpConstraint *constraint, type value){ \
|
||||||
cpConstraintCheckCast(constraint, struct); \
|
cpConstraintCheckCast(constraint, struct); \
|
||||||
|
cpConstraintActivateBodies(constraint); \
|
||||||
((struct *)constraint)->member = value; \
|
((struct *)constraint)->member = value; \
|
||||||
} \
|
} \
|
||||||
|
|
||||||
|
|
|
@ -30,8 +30,8 @@ typedef struct cpDampedRotarySpring {
|
||||||
cpFloat damping;
|
cpFloat damping;
|
||||||
cpDampedRotarySpringTorqueFunc springTorqueFunc;
|
cpDampedRotarySpringTorqueFunc springTorqueFunc;
|
||||||
|
|
||||||
cpFloat dt;
|
|
||||||
cpFloat target_wrn;
|
cpFloat target_wrn;
|
||||||
|
cpFloat w_coef;
|
||||||
|
|
||||||
cpFloat iSum;
|
cpFloat iSum;
|
||||||
} cpDampedRotarySpring;
|
} cpDampedRotarySpring;
|
||||||
|
|
|
@ -33,8 +33,8 @@ typedef struct cpDampedSpring {
|
||||||
cpFloat damping;
|
cpFloat damping;
|
||||||
cpDampedSpringForceFunc springForceFunc;
|
cpDampedSpringForceFunc springForceFunc;
|
||||||
|
|
||||||
cpFloat dt;
|
|
||||||
cpFloat target_vrn;
|
cpFloat target_vrn;
|
||||||
|
cpFloat v_coef;
|
||||||
|
|
||||||
cpVect r1, r2;
|
cpVect r1, r2;
|
||||||
cpFloat nMass;
|
cpFloat nMass;
|
||||||
|
|
|
@ -40,5 +40,9 @@ cpGrooveJoint *cpGrooveJointAlloc(void);
|
||||||
cpGrooveJoint *cpGrooveJointInit(cpGrooveJoint *joint, cpBody *a, cpBody *b, cpVect groove_a, cpVect groove_b, cpVect anchr2);
|
cpGrooveJoint *cpGrooveJointInit(cpGrooveJoint *joint, cpBody *a, cpBody *b, cpVect groove_a, cpVect groove_b, cpVect anchr2);
|
||||||
cpConstraint *cpGrooveJointNew(cpBody *a, cpBody *b, cpVect groove_a, cpVect groove_b, cpVect anchr2);
|
cpConstraint *cpGrooveJointNew(cpBody *a, cpBody *b, cpVect groove_a, cpVect groove_b, cpVect anchr2);
|
||||||
|
|
||||||
// TODO setters for the groove.
|
|
||||||
|
CP_DefineConstraintGetter(cpGrooveJoint, cpVect, grv_a, GrooveA);
|
||||||
|
void cpGrooveJointSetGrooveA(cpConstraint *constraint, cpVect value);
|
||||||
|
CP_DefineConstraintGetter(cpGrooveJoint, cpVect, grv_b, GrooveB);
|
||||||
|
void cpGrooveJointSetGrooveB(cpConstraint *constraint, cpVect value);
|
||||||
CP_DefineConstraintProperty(cpGrooveJoint, cpVect, anchr2, Anchr2);
|
CP_DefineConstraintProperty(cpGrooveJoint, cpVect, anchr2, Anchr2);
|
||||||
|
|
|
@ -25,6 +25,17 @@ void cpConstraintInit(cpConstraint *constraint, const cpConstraintClass *klass,
|
||||||
|
|
||||||
#define J_MAX(constraint, dt) (((cpConstraint *)constraint)->maxForce*(dt))
|
#define J_MAX(constraint, dt) (((cpConstraint *)constraint)->maxForce*(dt))
|
||||||
|
|
||||||
|
// Get valid body pointers and exit early if the bodies are idle
|
||||||
|
#define CONSTRAINT_BEGIN(constraint, a_var, b_var) \
|
||||||
|
cpBody *a_var, *b_var; { \
|
||||||
|
a_var = ((cpConstraint *)constraint)->a; \
|
||||||
|
b_var = ((cpConstraint *)constraint)->b; \
|
||||||
|
if( \
|
||||||
|
(cpBodyIsSleeping(a_var) || cpBodyIsStatic(a_var)) && \
|
||||||
|
(cpBodyIsSleeping(b_var) || cpBodyIsStatic(b_var)) \
|
||||||
|
) return; \
|
||||||
|
}
|
||||||
|
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
relative_velocity(cpBody *a, cpBody *b, cpVect r1, cpVect r2){
|
relative_velocity(cpBody *a, cpBody *b, cpVect r1, cpVect r2){
|
||||||
cpVect v1_sum = cpvadd(a->v, cpvmult(cpvperp(r1), a->w));
|
cpVect v1_sum = cpvadd(a->v, cpvmult(cpvperp(r1), a->w));
|
||||||
|
@ -45,11 +56,18 @@ apply_impulses(cpBody *a , cpBody *b, cpVect r1, cpVect r2, cpVect j)
|
||||||
cpBodyApplyImpulse(b, j, r2);
|
cpBodyApplyImpulse(b, j, r2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void
|
||||||
|
apply_bias_impulse(cpBody *body, cpVect j, cpVect r)
|
||||||
|
{
|
||||||
|
body->v_bias = cpvadd(body->v_bias, cpvmult(j, body->m_inv));
|
||||||
|
body->w_bias += body->i_inv*cpvcross(r, j);
|
||||||
|
}
|
||||||
|
|
||||||
static inline void
|
static inline void
|
||||||
apply_bias_impulses(cpBody *a , cpBody *b, cpVect r1, cpVect r2, cpVect j)
|
apply_bias_impulses(cpBody *a , cpBody *b, cpVect r1, cpVect r2, cpVect j)
|
||||||
{
|
{
|
||||||
cpBodyApplyBiasImpulse(a, cpvneg(j), r1);
|
apply_bias_impulse(a, cpvneg(j), r1);
|
||||||
cpBodyApplyBiasImpulse(b, j, r2);
|
apply_bias_impulse(b, j, r2);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
|
|
|
@ -31,73 +31,79 @@ extern cpFloat cp_collision_slop;
|
||||||
// Data structure for contact points.
|
// Data structure for contact points.
|
||||||
typedef struct cpContact {
|
typedef struct cpContact {
|
||||||
// Contact point and normal.
|
// Contact point and normal.
|
||||||
cpVect p, n;
|
cpVect CP_PRIVATE(p), CP_PRIVATE(n);
|
||||||
// Penetration distance.
|
// Penetration distance.
|
||||||
cpFloat dist;
|
CP_PRIVATE(cpFloat dist);
|
||||||
|
|
||||||
// Calculated by cpArbiterPreStep().
|
// Calculated by cpArbiterPreStep().
|
||||||
cpVect r1, r2;
|
cpVect CP_PRIVATE(r1), CP_PRIVATE(r2);
|
||||||
cpFloat nMass, tMass, bounce;
|
cpFloat CP_PRIVATE(nMass), CP_PRIVATE(tMass), CP_PRIVATE(bounce);
|
||||||
|
|
||||||
// Persistant contact information.
|
// Persistant contact information.
|
||||||
cpFloat jnAcc, jtAcc, jBias;
|
cpFloat CP_PRIVATE(jnAcc), CP_PRIVATE(jtAcc), CP_PRIVATE(jBias);
|
||||||
cpFloat bias;
|
CP_PRIVATE(cpFloat bias);
|
||||||
|
|
||||||
// Hash value used to (mostly) uniquely identify a contact.
|
// Hash value used to (mostly) uniquely identify a contact.
|
||||||
cpHashValue hash;
|
CP_PRIVATE(cpHashValue hash);
|
||||||
} cpContact;
|
} cpContact;
|
||||||
|
|
||||||
// Contacts are always allocated in groups.
|
// Contacts are always allocated in groups.
|
||||||
cpContact* cpContactInit(cpContact *con, cpVect p, cpVect n, cpFloat dist, cpHashValue hash);
|
cpContact* cpContactInit(cpContact *con, cpVect p, cpVect n, cpFloat dist, cpHashValue hash);
|
||||||
|
|
||||||
// Sum the contact impulses. (Can be used after cpSpaceStep() returns)
|
// Sum the contact impulses. (Can be used after cpSpaceStep() returns)
|
||||||
cpVect cpContactsSumImpulses(cpContact *contacts, int numContacts);
|
cpVect CP_PRIVATE(cpContactsSumImpulses)(cpContact *contacts, int numContacts);
|
||||||
cpVect cpContactsSumImpulsesWithFriction(cpContact *contacts, int numContacts);
|
cpVect CP_PRIVATE(cpContactsSumImpulsesWithFriction)(cpContact *contacts, int numContacts);
|
||||||
|
|
||||||
|
#define CP_MAX_CONTACTS_PER_ARBITER 6
|
||||||
|
|
||||||
typedef enum cpArbiterState {
|
typedef enum cpArbiterState {
|
||||||
cpArbiterStateNormal,
|
cpArbiterStateNormal,
|
||||||
cpArbiterStateFirstColl,
|
cpArbiterStateFirstColl,
|
||||||
cpArbiterStateIgnore,
|
cpArbiterStateIgnore,
|
||||||
|
cpArbiterStateSleep,
|
||||||
|
cpArbiterStateCached,
|
||||||
} cpArbiterState;
|
} cpArbiterState;
|
||||||
|
|
||||||
// Data structure for tracking collisions between shapes.
|
// Data structure for tracking collisions between shapes.
|
||||||
typedef struct cpArbiter {
|
typedef struct cpArbiter {
|
||||||
// Information on the contact points between the objects.
|
// Information on the contact points between the objects.
|
||||||
int numContacts;
|
CP_PRIVATE(int numContacts);
|
||||||
cpContact *contacts;
|
CP_PRIVATE(cpContact *contacts);
|
||||||
|
|
||||||
// The two shapes involved in the collision.
|
// The two shapes and bodies involved in the collision.
|
||||||
// These variables are NOT in the order defined by the collision handler.
|
// These variables are NOT in the order defined by the collision handler.
|
||||||
cpShape *private_a, *private_b;
|
// Using CP_ARBITER_GET_SHAPES and CP_ARBITER_GET_BODIES will save you from
|
||||||
|
// many headaches
|
||||||
|
cpShape CP_PRIVATE(*a), CP_PRIVATE(*b);
|
||||||
|
|
||||||
// Calculated before calling the pre-solve collision handler
|
// Calculated before calling the pre-solve collision handler
|
||||||
// Override them with custom values if you want specialized behavior
|
// Override them with custom values if you want specialized behavior
|
||||||
cpFloat e;
|
CP_PRIVATE(cpFloat e);
|
||||||
cpFloat u;
|
CP_PRIVATE(cpFloat u);
|
||||||
// Used for surface_v calculations, implementation may change
|
// Used for surface_v calculations, implementation may change
|
||||||
cpVect surface_vr;
|
CP_PRIVATE(cpVect surface_vr);
|
||||||
|
|
||||||
// Time stamp of the arbiter. (from cpSpace)
|
// Time stamp of the arbiter. (from cpSpace)
|
||||||
int stamp;
|
CP_PRIVATE(cpTimestamp stamp);
|
||||||
|
|
||||||
struct cpCollisionHandler *handler;
|
CP_PRIVATE(struct cpCollisionHandler *handler);
|
||||||
|
|
||||||
// Are the shapes swapped in relation to the collision handler?
|
// Are the shapes swapped in relation to the collision handler?
|
||||||
char swappedColl;
|
CP_PRIVATE(cpBool swappedColl);
|
||||||
char state;
|
CP_PRIVATE(cpArbiterState state);
|
||||||
} cpArbiter;
|
} cpArbiter;
|
||||||
|
|
||||||
// Arbiters are allocated in large buffers by the space and don't require a destroy function
|
// Arbiters are allocated in large buffers by the space and don't require a destroy function
|
||||||
cpArbiter* cpArbiterInit(cpArbiter *arb, cpShape *a, cpShape *b);
|
cpArbiter* CP_PRIVATE(cpArbiterInit)(cpArbiter *arb, cpShape *a, cpShape *b);
|
||||||
|
|
||||||
// These functions are all intended to be used internally.
|
// These functions are all intended to be used internally.
|
||||||
// Inject new contact points into the arbiter while preserving contact history.
|
// Inject new contact points into the arbiter while preserving contact history.
|
||||||
void cpArbiterUpdate(cpArbiter *arb, cpContact *contacts, int numContacts, struct cpCollisionHandler *handler, cpShape *a, cpShape *b);
|
void CP_PRIVATE(cpArbiterUpdate)(cpArbiter *arb, cpContact *contacts, int numContacts, struct cpCollisionHandler *handler, cpShape *a, cpShape *b);
|
||||||
// Precalculate values used by the solver.
|
// Precalculate values used by the solver.
|
||||||
void cpArbiterPreStep(cpArbiter *arb, cpFloat dt_inv);
|
void CP_PRIVATE(cpArbiterPreStep)(cpArbiter *arb, cpFloat dt_inv);
|
||||||
void cpArbiterApplyCachedImpulse(cpArbiter *arb);
|
void CP_PRIVATE(cpArbiterApplyCachedImpulse)(cpArbiter *arb);
|
||||||
// Run an iteration of the solver on the arbiter.
|
// Run an iteration of the solver on the arbiter.
|
||||||
void cpArbiterApplyImpulse(cpArbiter *arb, cpFloat eCoef);
|
void CP_PRIVATE(cpArbiterApplyImpulse)(cpArbiter *arb, cpFloat eCoef);
|
||||||
|
|
||||||
// Arbiter Helper Functions
|
// Arbiter Helper Functions
|
||||||
cpVect cpArbiterTotalImpulse(cpArbiter *arb);
|
cpVect cpArbiterTotalImpulse(cpArbiter *arb);
|
||||||
|
@ -106,31 +112,77 @@ void cpArbiterIgnore(cpArbiter *arb);
|
||||||
|
|
||||||
|
|
||||||
static inline void
|
static inline void
|
||||||
cpArbiterGetShapes(cpArbiter *arb, cpShape **a, cpShape **b)
|
cpArbiterGetShapes(const cpArbiter *arb, cpShape **a, cpShape **b)
|
||||||
{
|
{
|
||||||
if(arb->swappedColl){
|
if(arb->CP_PRIVATE(swappedColl)){
|
||||||
(*a) = arb->private_b, (*b) = arb->private_a;
|
(*a) = arb->CP_PRIVATE(b), (*b) = arb->CP_PRIVATE(a);
|
||||||
} else {
|
} else {
|
||||||
(*a) = arb->private_a, (*b) = arb->private_b;
|
(*a) = arb->CP_PRIVATE(a), (*b) = arb->CP_PRIVATE(b);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#define CP_ARBITER_GET_SHAPES(arb, a, b) cpShape *a, *b; cpArbiterGetShapes(arb, &a, &b);
|
#define CP_ARBITER_GET_SHAPES(arb, a, b) cpShape *a, *b; cpArbiterGetShapes(arb, &a, &b);
|
||||||
|
|
||||||
|
static inline void
|
||||||
|
cpArbiterGetBodies(const cpArbiter *arb, cpBody **a, cpBody **b)
|
||||||
|
{
|
||||||
|
CP_ARBITER_GET_SHAPES(arb, shape_a, shape_b);
|
||||||
|
(*a) = shape_a->body;
|
||||||
|
(*b) = shape_b->body;
|
||||||
|
}
|
||||||
|
#define CP_ARBITER_GET_BODIES(arb, a, b) cpBody *a, *b; cpArbiterGetBodies(arb, &a, &b);
|
||||||
|
|
||||||
|
static inline cpBool
|
||||||
|
cpArbiterIsFirstContact(const cpArbiter *arb)
|
||||||
|
{
|
||||||
|
return arb->CP_PRIVATE(state) == cpArbiterStateFirstColl;
|
||||||
|
}
|
||||||
|
|
||||||
static inline int
|
static inline int
|
||||||
cpArbiterIsFirstContact(cpArbiter *arb)
|
cpArbiterGetCount(const cpArbiter *arb)
|
||||||
{
|
{
|
||||||
return arb->state == cpArbiterStateFirstColl;
|
return arb->CP_PRIVATE(numContacts);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpArbiterGetNormal(cpArbiter *arb, int i)
|
cpArbiterGetNormal(const cpArbiter *arb, int i)
|
||||||
{
|
{
|
||||||
cpVect n = arb->contacts[i].n;
|
cpVect n = arb->CP_PRIVATE(contacts)[i].CP_PRIVATE(n);
|
||||||
return arb->swappedColl ? cpvneg(n) : n;
|
return arb->CP_PRIVATE(swappedColl) ? cpvneg(n) : n;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpArbiterGetPoint(cpArbiter *arb, int i)
|
cpArbiterGetPoint(const cpArbiter *arb, int i)
|
||||||
{
|
{
|
||||||
return arb->contacts[i].p;
|
return arb->CP_PRIVATE(contacts)[i].CP_PRIVATE(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline cpFloat
|
||||||
|
cpArbiteGetDepth(const cpArbiter *arb, int i)
|
||||||
|
{
|
||||||
|
return arb->CP_PRIVATE(contacts)[i].CP_PRIVATE(dist);
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef struct cpContactPointSet {
|
||||||
|
int count;
|
||||||
|
|
||||||
|
struct {
|
||||||
|
cpVect point, normal;
|
||||||
|
cpFloat dist;
|
||||||
|
} points[CP_MAX_CONTACTS_PER_ARBITER];
|
||||||
|
} cpContactPointSet;
|
||||||
|
|
||||||
|
static inline cpContactPointSet
|
||||||
|
cpArbiterGetContactPointSet(const cpArbiter *arb)
|
||||||
|
{
|
||||||
|
cpContactPointSet set;
|
||||||
|
set.count = cpArbiterGetCount(arb);
|
||||||
|
|
||||||
|
int i;
|
||||||
|
for(i=0; i<set.count; i++){
|
||||||
|
set.points[i].point = arb->CP_PRIVATE(contacts)[i].CP_PRIVATE(p);
|
||||||
|
set.points[i].normal = arb->CP_PRIVATE(contacts)[i].CP_PRIVATE(n);
|
||||||
|
set.points[i].dist = arb->CP_PRIVATE(contacts)[i].CP_PRIVATE(dist);
|
||||||
|
}
|
||||||
|
|
||||||
|
return set;
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,8 +22,9 @@
|
||||||
// NOTE: cpArray is rarely used and will probably go away.
|
// NOTE: cpArray is rarely used and will probably go away.
|
||||||
|
|
||||||
typedef struct cpArray{
|
typedef struct cpArray{
|
||||||
int num, max;
|
CP_PRIVATE(int num);
|
||||||
void **arr;
|
CP_PRIVATE(int max);
|
||||||
|
CP_PRIVATE(void **arr);
|
||||||
} cpArray;
|
} cpArray;
|
||||||
|
|
||||||
typedef void (*cpArrayIter)(void *ptr, void *data);
|
typedef void (*cpArrayIter)(void *ptr, void *data);
|
||||||
|
@ -42,5 +43,7 @@ void *cpArrayPop(cpArray *arr);
|
||||||
void cpArrayDeleteIndex(cpArray *arr, int idx);
|
void cpArrayDeleteIndex(cpArray *arr, int idx);
|
||||||
void cpArrayDeleteObj(cpArray *arr, void *obj);
|
void cpArrayDeleteObj(cpArray *arr, void *obj);
|
||||||
|
|
||||||
|
void cpArrayAppend(cpArray *arr, cpArray *other);
|
||||||
|
|
||||||
void cpArrayEach(cpArray *arr, cpArrayIter iterFunc, void *data);
|
void cpArrayEach(cpArray *arr, cpArrayIter iterFunc, void *data);
|
||||||
int cpArrayContains(cpArray *arr, void *ptr);
|
cpBool cpArrayContains(cpArray *arr, void *ptr);
|
||||||
|
|
|
@ -31,19 +31,19 @@ cpBBNew(const cpFloat l, const cpFloat b,
|
||||||
return bb;
|
return bb;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int
|
static inline cpBool
|
||||||
cpBBintersects(const cpBB a, const cpBB b)
|
cpBBintersects(const cpBB a, const cpBB b)
|
||||||
{
|
{
|
||||||
return (a.l<=b.r && b.l<=a.r && a.b<=b.t && b.b<=a.t);
|
return (a.l<=b.r && b.l<=a.r && a.b<=b.t && b.b<=a.t);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int
|
static inline cpBool
|
||||||
cpBBcontainsBB(const cpBB bb, const cpBB other)
|
cpBBcontainsBB(const cpBB bb, const cpBB other)
|
||||||
{
|
{
|
||||||
return (bb.l < other.l && bb.r > other.r && bb.b < other.b && bb.t > other.t);
|
return (bb.l < other.l && bb.r > other.r && bb.b < other.b && bb.t > other.t);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int
|
static inline cpBool
|
||||||
cpBBcontainsVect(const cpBB bb, const cpVect v)
|
cpBBcontainsVect(const cpBB bb, const cpVect v)
|
||||||
{
|
{
|
||||||
return (bb.l < v.x && bb.r > v.x && bb.b < v.y && bb.t > v.y);
|
return (bb.l < v.x && bb.r > v.x && bb.b < v.y && bb.t > v.y);
|
||||||
|
|
|
@ -20,14 +20,27 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
struct cpBody;
|
struct cpBody;
|
||||||
|
struct cpShape;
|
||||||
|
struct cpSpace;
|
||||||
|
|
||||||
typedef void (*cpBodyVelocityFunc)(struct cpBody *body, cpVect gravity, cpFloat damping, cpFloat dt);
|
typedef void (*cpBodyVelocityFunc)(struct cpBody *body, cpVect gravity, cpFloat damping, cpFloat dt);
|
||||||
typedef void (*cpBodyPositionFunc)(struct cpBody *body, cpFloat dt);
|
typedef void (*cpBodyPositionFunc)(struct cpBody *body, cpFloat dt);
|
||||||
|
|
||||||
extern cpBodyVelocityFunc cpBodyUpdateVelocityDefault;
|
extern cpBodyVelocityFunc cpBodyUpdateVelocityDefault;
|
||||||
extern cpBodyPositionFunc cpBodyUpdatePositionDefault;
|
extern cpBodyPositionFunc cpBodyUpdatePositionDefault;
|
||||||
|
|
||||||
|
// Structure to hold information about the contact graph components
|
||||||
|
// when putting groups of objects to sleep.
|
||||||
|
// No interesting user accessible fields.
|
||||||
|
typedef struct cpComponentNode {
|
||||||
|
struct cpBody *parent;
|
||||||
|
struct cpBody *next;
|
||||||
|
int rank;
|
||||||
|
cpFloat idleTime;
|
||||||
|
} cpComponentNode;
|
||||||
|
|
||||||
typedef struct cpBody{
|
typedef struct cpBody{
|
||||||
// *** Integration Functions.ntoehu
|
// *** Integration Functions.
|
||||||
|
|
||||||
// Function that is called to integrate the body's velocity. (Defaults to cpBodyUpdateVelocity)
|
// Function that is called to integrate the body's velocity. (Defaults to cpBodyUpdateVelocity)
|
||||||
cpBodyVelocityFunc velocity_func;
|
cpBodyVelocityFunc velocity_func;
|
||||||
|
@ -63,16 +76,26 @@ typedef struct cpBody{
|
||||||
// User defined data pointer.
|
// User defined data pointer.
|
||||||
cpDataPointer data;
|
cpDataPointer data;
|
||||||
|
|
||||||
|
// *** Other Fields
|
||||||
|
|
||||||
// Maximum velocities this body can move at after integrating velocity
|
// Maximum velocities this body can move at after integrating velocity
|
||||||
cpFloat v_limit, w_limit;
|
cpFloat v_limit, w_limit;
|
||||||
|
|
||||||
// *** Internally Used Fields
|
// *** Internally Used Fields
|
||||||
|
|
||||||
// Velocity bias values used when solving penetrations and correcting constraints.
|
// Velocity bias values used when solving penetrations and correcting constraints.
|
||||||
cpVect v_bias;
|
CP_PRIVATE(cpVect v_bias);
|
||||||
cpFloat w_bias;
|
CP_PRIVATE(cpFloat w_bias);
|
||||||
|
|
||||||
// int active;
|
// Space this body has been added to
|
||||||
|
CP_PRIVATE(struct cpSpace *space);
|
||||||
|
|
||||||
|
// Pointer to the shape list.
|
||||||
|
// Shapes form a linked list using cpShape.next when added to a space.
|
||||||
|
CP_PRIVATE(struct cpShape *shapesList);
|
||||||
|
|
||||||
|
// Used by cpSpaceStep() to store contact graph information.
|
||||||
|
CP_PRIVATE(cpComponentNode node);
|
||||||
} cpBody;
|
} cpBody;
|
||||||
|
|
||||||
// Basic allocation/destruction functions
|
// Basic allocation/destruction functions
|
||||||
|
@ -80,11 +103,48 @@ cpBody *cpBodyAlloc(void);
|
||||||
cpBody *cpBodyInit(cpBody *body, cpFloat m, cpFloat i);
|
cpBody *cpBodyInit(cpBody *body, cpFloat m, cpFloat i);
|
||||||
cpBody *cpBodyNew(cpFloat m, cpFloat i);
|
cpBody *cpBodyNew(cpFloat m, cpFloat i);
|
||||||
|
|
||||||
|
cpBody *cpBodyInitStatic(cpBody *body);
|
||||||
|
cpBody *cpBodyNewStatic();
|
||||||
|
|
||||||
void cpBodyDestroy(cpBody *body);
|
void cpBodyDestroy(cpBody *body);
|
||||||
void cpBodyFree(cpBody *body);
|
void cpBodyFree(cpBody *body);
|
||||||
|
|
||||||
#define CP_DefineBodyGetter(type, member, name) static inline type cpBodyGet##name(cpBody *body){return body->member;}
|
// Wake up a sleeping or idle body. (defined in cpSpace.c)
|
||||||
#define CP_DefineBodySetter(type, member, name) static inline void cpBodySet##name(cpBody *body, type value){body->member = value;}
|
void cpBodyActivate(cpBody *body);
|
||||||
|
|
||||||
|
// Force a body to sleep;
|
||||||
|
// defined in cpSpaceComponent.c
|
||||||
|
void cpBodySleep(cpBody *body);
|
||||||
|
void cpBodySleepWithGroup(cpBody *body, cpBody *group);
|
||||||
|
|
||||||
|
static inline cpBool
|
||||||
|
cpBodyIsSleeping(const cpBody *body)
|
||||||
|
{
|
||||||
|
return (CP_PRIVATE(body->node).next != ((cpBody*)0));
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline cpBool
|
||||||
|
cpBodyIsStatic(const cpBody *body)
|
||||||
|
{
|
||||||
|
return CP_PRIVATE(body->node).idleTime == INFINITY;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline cpBool
|
||||||
|
cpBodyIsRogue(const cpBody *body)
|
||||||
|
{
|
||||||
|
return (body->CP_PRIVATE(space) == ((struct cpSpace*)0));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#define CP_DefineBodyGetter(type, member, name) \
|
||||||
|
static inline type cpBodyGet##name(const cpBody *body){return body->member;}
|
||||||
|
|
||||||
|
#define CP_DefineBodySetter(type, member, name) \
|
||||||
|
static inline void \
|
||||||
|
cpBodySet##name(cpBody *body, const type value){ \
|
||||||
|
cpBodyActivate(body); \
|
||||||
|
body->member = value; \
|
||||||
|
} \
|
||||||
|
|
||||||
#define CP_DefineBodyProperty(type, member, name) \
|
#define CP_DefineBodyProperty(type, member, name) \
|
||||||
CP_DefineBodyGetter(type, member, name) \
|
CP_DefineBodyGetter(type, member, name) \
|
||||||
|
@ -120,41 +180,40 @@ void cpBodyUpdatePosition(cpBody *body, cpFloat dt);
|
||||||
|
|
||||||
// Convert body local to world coordinates
|
// Convert body local to world coordinates
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpBodyLocal2World(cpBody *body, cpVect v)
|
cpBodyLocal2World(const cpBody *body, const cpVect v)
|
||||||
{
|
{
|
||||||
return cpvadd(body->p, cpvrotate(v, body->rot));
|
return cpvadd(body->p, cpvrotate(v, body->rot));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert world to body local coordinates
|
// Convert world to body local coordinates
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpBodyWorld2Local(cpBody *body, cpVect v)
|
cpBodyWorld2Local(const cpBody *body, const cpVect v)
|
||||||
{
|
{
|
||||||
return cpvunrotate(cpvsub(v, body->p), body->rot);
|
return cpvunrotate(cpvsub(v, body->p), body->rot);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Apply an impulse (in world coordinates) to the body at a point relative to the center of gravity (also in world coordinates).
|
// Apply an impulse (in world coordinates) to the body at a point relative to the center of gravity (also in world coordinates).
|
||||||
static inline void
|
static inline void
|
||||||
cpBodyApplyImpulse(cpBody *body, cpVect j, cpVect r)
|
cpBodyApplyImpulse(cpBody *body, const cpVect j, const cpVect r)
|
||||||
{
|
{
|
||||||
body->v = cpvadd(body->v, cpvmult(j, body->m_inv));
|
body->v = cpvadd(body->v, cpvmult(j, body->m_inv));
|
||||||
body->w += body->i_inv*cpvcross(r, j);
|
body->w += body->i_inv*cpvcross(r, j);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Not intended for external use. Used by cpArbiter.c and cpConstraint.c.
|
|
||||||
static inline void
|
|
||||||
cpBodyApplyBiasImpulse(cpBody *body, cpVect j, cpVect r)
|
|
||||||
{
|
|
||||||
body->v_bias = cpvadd(body->v_bias, cpvmult(j, body->m_inv));
|
|
||||||
body->w_bias += body->i_inv*cpvcross(r, j);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Zero the forces on a body.
|
// Zero the forces on a body.
|
||||||
void cpBodyResetForces(cpBody *body);
|
void cpBodyResetForces(cpBody *body);
|
||||||
// Apply a force (in world coordinates) to a body at a point relative to the center of gravity (also in world coordinates).
|
// Apply a force (in world coordinates) to a body at a point relative to the center of gravity (also in world coordinates).
|
||||||
void cpBodyApplyForce(cpBody *body, cpVect f, cpVect r);
|
void cpBodyApplyForce(cpBody *body, const cpVect f, const cpVect r);
|
||||||
|
|
||||||
|
static inline cpFloat
|
||||||
|
cpBodyKineticEnergy(const cpBody *body)
|
||||||
|
{
|
||||||
|
// Need to do some fudging to avoid NaNs
|
||||||
|
cpFloat vsq = cpvdot(body->v, body->v);
|
||||||
|
cpFloat wsq = body->w*body->w;
|
||||||
|
return (vsq ? vsq*body->m : 0.0f) + (wsq ? wsq*body->i : 0.0f);
|
||||||
|
}
|
||||||
|
|
||||||
// Apply a damped spring force between two bodies.
|
// Apply a damped spring force between two bodies.
|
||||||
// Warning: Large damping values can be unstable. Use a cpDampedSpring constraint for this instead.
|
// Warning: Large damping values can be unstable. Use a cpDampedSpring constraint for this instead.
|
||||||
void cpApplyDampedSpring(cpBody *a, cpBody *b, cpVect anchr1, cpVect anchr2, cpFloat rlen, cpFloat k, cpFloat dmp, cpFloat dt);
|
void cpApplyDampedSpring(cpBody *a, cpBody *b, cpVect anchr1, cpVect anchr2, cpFloat rlen, cpFloat k, cpFloat dmp, cpFloat dt);
|
||||||
|
|
||||||
//int cpBodyMarkLowEnergy(cpBody *body, cpFloat dvsq, int max);
|
|
||||||
|
|
|
@ -19,5 +19,10 @@
|
||||||
* SOFTWARE.
|
* SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// Collides two cpShape structures. (this function is lonely :( )
|
//TODO delete this header?
|
||||||
int cpCollideShapes(cpShape *a, cpShape *b, cpContact *arr);
|
|
||||||
|
// Collides two cpShape structures.
|
||||||
|
// Returns the number of contact points added to arr
|
||||||
|
// which should be at least CP_MAX_CONTACTS_PER_ARBITER in length.
|
||||||
|
// This function is very lonely in this header :(
|
||||||
|
int cpCollideShapes(const cpShape *a, const cpShape *b, cpContact *arr);
|
||||||
|
|
|
@ -25,39 +25,36 @@
|
||||||
// cpHashSetBin's form the linked lists in the chained hash table.
|
// cpHashSetBin's form the linked lists in the chained hash table.
|
||||||
typedef struct cpHashSetBin {
|
typedef struct cpHashSetBin {
|
||||||
// Pointer to the element.
|
// Pointer to the element.
|
||||||
void *elt;
|
CP_PRIVATE(void *elt);
|
||||||
// Hash value of the element.
|
// Hash value of the element.
|
||||||
cpHashValue hash;
|
CP_PRIVATE(cpHashValue hash);
|
||||||
// Next element in the chain.
|
// Next element in the chain.
|
||||||
struct cpHashSetBin *next;
|
CP_PRIVATE(struct cpHashSetBin *next);
|
||||||
} cpHashSetBin;
|
} cpHashSetBin;
|
||||||
|
|
||||||
// Equality function. Returns true if ptr is equal to elt.
|
// Equality function. Returns true if ptr is equal to elt.
|
||||||
typedef int (*cpHashSetEqlFunc)(void *ptr, void *elt);
|
typedef cpBool (*cpHashSetEqlFunc)(void *ptr, void *elt);
|
||||||
// Used by cpHashSetInsert(). Called to transform the ptr into an element.
|
// Used by cpHashSetInsert(). Called to transform the ptr into an element.
|
||||||
typedef void *(*cpHashSetTransFunc)(void *ptr, void *data);
|
typedef void *(*cpHashSetTransFunc)(void *ptr, void *data);
|
||||||
// Iterator function for a hashset.
|
|
||||||
typedef void (*cpHashSetIterFunc)(void *elt, void *data);
|
|
||||||
// Filter function. Returns false if elt should be dropped.
|
|
||||||
typedef int (*cpHashSetFilterFunc)(void *elt, void *data);
|
|
||||||
|
|
||||||
typedef struct cpHashSet {
|
typedef struct cpHashSet {
|
||||||
// Number of elements stored in the table.
|
// Number of elements stored in the table.
|
||||||
int entries;
|
CP_PRIVATE(int entries);
|
||||||
// Number of cells in the table.
|
// Number of cells in the table.
|
||||||
int size;
|
CP_PRIVATE(int size);
|
||||||
|
|
||||||
cpHashSetEqlFunc eql;
|
CP_PRIVATE(cpHashSetEqlFunc eql);
|
||||||
cpHashSetTransFunc trans;
|
CP_PRIVATE(cpHashSetTransFunc trans);
|
||||||
|
|
||||||
// Default value returned by cpHashSetFind() when no element is found.
|
// Default value returned by cpHashSetFind() when no element is found.
|
||||||
// Defaults to NULL.
|
// Defaults to NULL.
|
||||||
void *default_value;
|
CP_PRIVATE(void *default_value);
|
||||||
|
|
||||||
// The table and recycled bins
|
// The table and recycled bins
|
||||||
cpHashSetBin **table, *pooledBins;
|
CP_PRIVATE(cpHashSetBin **table);
|
||||||
|
CP_PRIVATE(cpHashSetBin *pooledBins);
|
||||||
|
|
||||||
cpArray *allocatedBuffers;
|
CP_PRIVATE(cpArray *allocatedBuffers);
|
||||||
} cpHashSet;
|
} cpHashSet;
|
||||||
|
|
||||||
// Basic allocation/destruction functions.
|
// Basic allocation/destruction functions.
|
||||||
|
@ -77,6 +74,9 @@ void *cpHashSetRemove(cpHashSet *set, cpHashValue hash, void *ptr);
|
||||||
void *cpHashSetFind(cpHashSet *set, cpHashValue hash, void *ptr);
|
void *cpHashSetFind(cpHashSet *set, cpHashValue hash, void *ptr);
|
||||||
|
|
||||||
// Iterate over a hashset.
|
// Iterate over a hashset.
|
||||||
|
typedef void (*cpHashSetIterFunc)(void *elt, void *data);
|
||||||
void cpHashSetEach(cpHashSet *set, cpHashSetIterFunc func, void *data);
|
void cpHashSetEach(cpHashSet *set, cpHashSetIterFunc func, void *data);
|
||||||
// Iterate over a hashset, retain .
|
|
||||||
|
// Iterate over a hashset, drop the element if the func returns false.
|
||||||
|
typedef cpBool (*cpHashSetFilterFunc)(void *elt, void *data);
|
||||||
void cpHashSetFilter(cpHashSet *set, cpHashSetFilterFunc func, void *data);
|
void cpHashSetFilter(cpHashSet *set, cpHashSetFilterFunc func, void *data);
|
||||||
|
|
|
@ -29,16 +29,16 @@ typedef struct cpPolyShapeAxis{
|
||||||
|
|
||||||
// Convex polygon shape structure.
|
// Convex polygon shape structure.
|
||||||
typedef struct cpPolyShape{
|
typedef struct cpPolyShape{
|
||||||
cpShape shape;
|
CP_PRIVATE(cpShape shape);
|
||||||
|
|
||||||
// Vertex and axis lists.
|
// Vertex and axis lists.
|
||||||
int numVerts;
|
CP_PRIVATE(int numVerts);
|
||||||
cpVect *verts;
|
CP_PRIVATE(cpVect *verts);
|
||||||
cpPolyShapeAxis *axes;
|
CP_PRIVATE(cpPolyShapeAxis *axes);
|
||||||
|
|
||||||
// Transformed vertex and axis lists.
|
// Transformed vertex and axis lists.
|
||||||
cpVect *tVerts;
|
CP_PRIVATE(cpVect *tVerts);
|
||||||
cpPolyShapeAxis *tAxes;
|
CP_PRIVATE(cpPolyShapeAxis *tAxes);
|
||||||
} cpPolyShape;
|
} cpPolyShape;
|
||||||
|
|
||||||
// Basic allocation functions.
|
// Basic allocation functions.
|
||||||
|
@ -50,7 +50,7 @@ cpPolyShape *cpBoxShapeInit(cpPolyShape *poly, cpBody *body, cpFloat width, cpFl
|
||||||
cpShape *cpBoxShapeNew(cpBody *body, cpFloat width, cpFloat height);
|
cpShape *cpBoxShapeNew(cpBody *body, cpFloat width, cpFloat height);
|
||||||
|
|
||||||
// Check that a set of vertexes has a correct winding and that they are convex
|
// Check that a set of vertexes has a correct winding and that they are convex
|
||||||
int cpPolyValidate(cpVect *verts, int numVerts);
|
cpBool cpPolyValidate(const cpVect *verts, const int numVerts);
|
||||||
|
|
||||||
int cpPolyShapeGetNumVerts(cpShape *shape);
|
int cpPolyShapeGetNumVerts(cpShape *shape);
|
||||||
cpVect cpPolyShapeGetVert(cpShape *shape, int idx);
|
cpVect cpPolyShapeGetVert(cpShape *shape, int idx);
|
||||||
|
@ -61,43 +61,43 @@ cpVect cpPolyShapeGetVert(cpShape *shape, int idx);
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
cpPolyShapeValueOnAxis(const cpPolyShape *poly, const cpVect n, const cpFloat d)
|
cpPolyShapeValueOnAxis(const cpPolyShape *poly, const cpVect n, const cpFloat d)
|
||||||
{
|
{
|
||||||
cpVect *verts = poly->tVerts;
|
cpVect *verts = poly->CP_PRIVATE(tVerts);
|
||||||
cpFloat min = cpvdot(n, verts[0]);
|
cpFloat min = cpvdot(n, verts[0]);
|
||||||
|
|
||||||
int i;
|
int i;
|
||||||
for(i=1; i<poly->numVerts; i++)
|
for(i=1; i<poly->CP_PRIVATE(numVerts); i++)
|
||||||
min = cpfmin(min, cpvdot(n, verts[i]));
|
min = cpfmin(min, cpvdot(n, verts[i]));
|
||||||
|
|
||||||
return min - d;
|
return min - d;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns true if the polygon contains the vertex.
|
// Returns true if the polygon contains the vertex.
|
||||||
static inline int
|
static inline cpBool
|
||||||
cpPolyShapeContainsVert(cpPolyShape *poly, cpVect v)
|
cpPolyShapeContainsVert(const cpPolyShape *poly, const cpVect v)
|
||||||
{
|
{
|
||||||
cpPolyShapeAxis *axes = poly->tAxes;
|
cpPolyShapeAxis *axes = poly->CP_PRIVATE(tAxes);
|
||||||
|
|
||||||
int i;
|
int i;
|
||||||
for(i=0; i<poly->numVerts; i++){
|
for(i=0; i<poly->CP_PRIVATE(numVerts); i++){
|
||||||
cpFloat dist = cpvdot(axes[i].n, v) - axes[i].d;
|
cpFloat dist = cpvdot(axes[i].n, v) - axes[i].d;
|
||||||
if(dist > 0.0f) return 0;
|
if(dist > 0.0f) return cpFalse;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return cpTrue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as cpPolyShapeContainsVert() but ignores faces pointing away from the normal.
|
// Same as cpPolyShapeContainsVert() but ignores faces pointing away from the normal.
|
||||||
static inline int
|
static inline cpBool
|
||||||
cpPolyShapeContainsVertPartial(cpPolyShape *poly, cpVect v, cpVect n)
|
cpPolyShapeContainsVertPartial(const cpPolyShape *poly, const cpVect v, const cpVect n)
|
||||||
{
|
{
|
||||||
cpPolyShapeAxis *axes = poly->tAxes;
|
cpPolyShapeAxis *axes = poly->CP_PRIVATE(tAxes);
|
||||||
|
|
||||||
int i;
|
int i;
|
||||||
for(i=0; i<poly->numVerts; i++){
|
for(i=0; i<poly->CP_PRIVATE(numVerts); i++){
|
||||||
if(cpvdot(axes[i].n, n) < 0.0f) continue;
|
if(cpvdot(axes[i].n, n) < 0.0f) continue;
|
||||||
cpFloat dist = cpvdot(axes[i].n, v) - axes[i].d;
|
cpFloat dist = cpvdot(axes[i].n, v) - axes[i].d;
|
||||||
if(dist > 0.0f) return 0;
|
if(dist > 0.0f) return cpFalse;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return cpTrue;
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,7 +47,7 @@ typedef struct cpShapeClass {
|
||||||
void (*destroy)(struct cpShape *shape);
|
void (*destroy)(struct cpShape *shape);
|
||||||
|
|
||||||
// called by cpShapePointQuery().
|
// called by cpShapePointQuery().
|
||||||
int (*pointQuery)(struct cpShape *shape, cpVect p);
|
cpBool (*pointQuery)(struct cpShape *shape, cpVect p);
|
||||||
|
|
||||||
// called by cpShapeSegmentQuery()
|
// called by cpShapeSegmentQuery()
|
||||||
void (*segmentQuery)(struct cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInfo *info);
|
void (*segmentQuery)(struct cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInfo *info);
|
||||||
|
@ -56,7 +56,7 @@ typedef struct cpShapeClass {
|
||||||
// Basic shape struct that the others inherit from.
|
// Basic shape struct that the others inherit from.
|
||||||
typedef struct cpShape{
|
typedef struct cpShape{
|
||||||
// The "class" of a shape as defined above
|
// The "class" of a shape as defined above
|
||||||
const cpShapeClass *klass;
|
CP_PRIVATE(const cpShapeClass *klass);
|
||||||
|
|
||||||
// cpBody that the shape is attached to.
|
// cpBody that the shape is attached to.
|
||||||
cpBody *body;
|
cpBody *body;
|
||||||
|
@ -65,7 +65,7 @@ typedef struct cpShape{
|
||||||
cpBB bb;
|
cpBB bb;
|
||||||
|
|
||||||
// Sensors invoke callbacks, but do not generate collisions
|
// Sensors invoke callbacks, but do not generate collisions
|
||||||
int sensor;
|
cpBool sensor;
|
||||||
|
|
||||||
// *** Surface properties.
|
// *** Surface properties.
|
||||||
|
|
||||||
|
@ -90,8 +90,11 @@ typedef struct cpShape{
|
||||||
|
|
||||||
// *** Internally Used Fields
|
// *** Internally Used Fields
|
||||||
|
|
||||||
|
// Shapes form a linked list when added to space on a non-NULL body
|
||||||
|
CP_PRIVATE(struct cpShape *next);
|
||||||
|
|
||||||
// Unique id used as the hash value.
|
// Unique id used as the hash value.
|
||||||
cpHashValue hashid;
|
CP_PRIVATE(cpHashValue hashid);
|
||||||
} cpShape;
|
} cpShape;
|
||||||
|
|
||||||
// Low level shape initialization func.
|
// Low level shape initialization func.
|
||||||
|
@ -105,21 +108,21 @@ void cpShapeFree(cpShape *shape);
|
||||||
cpBB cpShapeCacheBB(cpShape *shape);
|
cpBB cpShapeCacheBB(cpShape *shape);
|
||||||
|
|
||||||
// Test if a point lies within a shape.
|
// Test if a point lies within a shape.
|
||||||
int cpShapePointQuery(cpShape *shape, cpVect p);
|
cpBool cpShapePointQuery(cpShape *shape, cpVect p);
|
||||||
|
|
||||||
#define CP_DeclareShapeGetter(struct, type, name) type struct##Get##name(cpShape *shape)
|
#define CP_DeclareShapeGetter(struct, type, name) type struct##Get##name(cpShape *shape)
|
||||||
|
|
||||||
// Circle shape structure.
|
// Circle shape structure.
|
||||||
typedef struct cpCircleShape{
|
typedef struct cpCircleShape{
|
||||||
cpShape shape;
|
CP_PRIVATE(cpShape shape);
|
||||||
|
|
||||||
// Center in body space coordinates
|
// Center in body space coordinates
|
||||||
cpVect c;
|
CP_PRIVATE(cpVect c);
|
||||||
// Radius.
|
// Radius.
|
||||||
cpFloat r;
|
CP_PRIVATE(cpFloat r);
|
||||||
|
|
||||||
// Transformed center. (world space coordinates)
|
// Transformed center. (world space coordinates)
|
||||||
cpVect tc;
|
CP_PRIVATE(cpVect tc);
|
||||||
} cpCircleShape;
|
} cpCircleShape;
|
||||||
|
|
||||||
// Basic allocation functions for cpCircleShape.
|
// Basic allocation functions for cpCircleShape.
|
||||||
|
@ -132,15 +135,15 @@ CP_DeclareShapeGetter(cpCircleShape, cpFloat, Radius);
|
||||||
|
|
||||||
// Segment shape structure.
|
// Segment shape structure.
|
||||||
typedef struct cpSegmentShape{
|
typedef struct cpSegmentShape{
|
||||||
cpShape shape;
|
CP_PRIVATE(cpShape shape);
|
||||||
|
|
||||||
// Endpoints and normal of the segment. (body space coordinates)
|
// Endpoints and normal of the segment. (body space coordinates)
|
||||||
cpVect a, b, n;
|
cpVect CP_PRIVATE(a), CP_PRIVATE(b), CP_PRIVATE(n);
|
||||||
// Radius of the segment. (Thickness)
|
// Radius of the segment. (Thickness)
|
||||||
cpFloat r;
|
cpFloat CP_PRIVATE(r);
|
||||||
|
|
||||||
// Transformed endpoints and normal. (world space coordinates)
|
// Transformed endpoints and normal. (world space coordinates)
|
||||||
cpVect ta, tb, tn;
|
cpVect CP_PRIVATE(ta), CP_PRIVATE(tb), CP_PRIVATE(tn);
|
||||||
} cpSegmentShape;
|
} cpSegmentShape;
|
||||||
|
|
||||||
// Basic allocation functions for cpSegmentShape.
|
// Basic allocation functions for cpSegmentShape.
|
||||||
|
@ -159,16 +162,16 @@ void cpResetShapeIdCounter(void);
|
||||||
// Directed segment queries against individual shapes.
|
// Directed segment queries against individual shapes.
|
||||||
void cpSegmentQueryInfoPrint(cpSegmentQueryInfo *info);
|
void cpSegmentQueryInfoPrint(cpSegmentQueryInfo *info);
|
||||||
|
|
||||||
int cpShapeSegmentQuery(cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInfo *info);
|
cpBool cpShapeSegmentQuery(cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInfo *info);
|
||||||
|
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpSegmentQueryHitPoint(cpVect start, cpVect end, cpSegmentQueryInfo info)
|
cpSegmentQueryHitPoint(const cpVect start, const cpVect end, const cpSegmentQueryInfo info)
|
||||||
{
|
{
|
||||||
return cpvlerp(start, end, info.t);
|
return cpvlerp(start, end, info.t);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
cpSegmentQueryHitDist(cpVect start, cpVect end, cpSegmentQueryInfo info)
|
cpSegmentQueryHitDist(const cpVect start, const cpVect end, const cpSegmentQueryInfo info)
|
||||||
{
|
{
|
||||||
return cpvdist(start, end)*info.t;
|
return cpvdist(start, end)*info.t;
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,11 +22,11 @@
|
||||||
struct cpSpace;
|
struct cpSpace;
|
||||||
|
|
||||||
// Number of frames that contact information should persist.
|
// Number of frames that contact information should persist.
|
||||||
extern int cp_contact_persistence;
|
extern cpTimestamp cp_contact_persistence;
|
||||||
|
|
||||||
// User collision handler function types.
|
// User collision handler function types.
|
||||||
typedef int (*cpCollisionBeginFunc)(cpArbiter *arb, struct cpSpace *space, void *data);
|
typedef cpBool (*cpCollisionBeginFunc)(cpArbiter *arb, struct cpSpace *space, void *data);
|
||||||
typedef int (*cpCollisionPreSolveFunc)(cpArbiter *arb, struct cpSpace *space, void *data);
|
typedef cpBool (*cpCollisionPreSolveFunc)(cpArbiter *arb, struct cpSpace *space, void *data);
|
||||||
typedef void (*cpCollisionPostSolveFunc)(cpArbiter *arb, struct cpSpace *space, void *data);
|
typedef void (*cpCollisionPostSolveFunc)(cpArbiter *arb, struct cpSpace *space, void *data);
|
||||||
typedef void (*cpCollisionSeparateFunc)(cpArbiter *arb, struct cpSpace *space, void *data);
|
typedef void (*cpCollisionSeparateFunc)(cpArbiter *arb, struct cpSpace *space, void *data);
|
||||||
|
|
||||||
|
@ -42,9 +42,8 @@ typedef struct cpCollisionHandler {
|
||||||
void *data;
|
void *data;
|
||||||
} cpCollisionHandler;
|
} cpCollisionHandler;
|
||||||
|
|
||||||
#define CP_MAX_CONTACTS_PER_ARBITER 6
|
|
||||||
typedef struct cpContactBufferHeader {
|
typedef struct cpContactBufferHeader {
|
||||||
int stamp;
|
cpTimestamp stamp;
|
||||||
struct cpContactBufferHeader *next;
|
struct cpContactBufferHeader *next;
|
||||||
unsigned int numContacts;
|
unsigned int numContacts;
|
||||||
} cpContactBufferHeader;
|
} cpContactBufferHeader;
|
||||||
|
@ -64,44 +63,62 @@ typedef struct cpSpace{
|
||||||
// Default damping to supply when integrating rigid body motions.
|
// Default damping to supply when integrating rigid body motions.
|
||||||
cpFloat damping;
|
cpFloat damping;
|
||||||
|
|
||||||
|
// Speed threshold for a body to be considered idle.
|
||||||
|
// The default value of 0 means to let the space guess a good threshold based on gravity.
|
||||||
|
cpFloat idleSpeedThreshold;
|
||||||
|
|
||||||
|
// Time a group of bodies must remain idle in order to fall asleep
|
||||||
|
// The default value of INFINITY disables the sleeping algorithm.
|
||||||
|
cpFloat sleepTimeThreshold;
|
||||||
|
|
||||||
// *** Internally Used Fields
|
// *** Internally Used Fields
|
||||||
|
|
||||||
// When the space is locked, you should not add or remove objects;
|
// When the space lock count is non zero you cannot add or remove objects
|
||||||
int locked;
|
CP_PRIVATE(int locked);
|
||||||
|
|
||||||
// Time stamp. Is incremented on every call to cpSpaceStep().
|
// Time stamp. Is incremented on every call to cpSpaceStep().
|
||||||
int stamp;
|
CP_PRIVATE(cpTimestamp stamp);
|
||||||
|
|
||||||
// The static and active shape spatial hashes.
|
// The static and active shape spatial hashes.
|
||||||
cpSpaceHash *staticShapes;
|
CP_PRIVATE(cpSpaceHash *staticShapes);
|
||||||
cpSpaceHash *activeShapes;
|
CP_PRIVATE(cpSpaceHash *activeShapes);
|
||||||
|
|
||||||
// List of bodies in the system.
|
// List of bodies in the system.
|
||||||
cpArray *bodies;
|
CP_PRIVATE(cpArray *bodies);
|
||||||
|
|
||||||
|
// List of groups of sleeping bodies.
|
||||||
|
CP_PRIVATE(cpArray *sleepingComponents);
|
||||||
|
|
||||||
|
// List of bodies that have been flagged to be awoken.
|
||||||
|
CP_PRIVATE(cpArray *rousedBodies);
|
||||||
|
|
||||||
// List of active arbiters for the impulse solver.
|
// List of active arbiters for the impulse solver.
|
||||||
cpArray *arbiters, *pooledArbiters;
|
CP_PRIVATE(cpArray *arbiters);
|
||||||
|
CP_PRIVATE(cpArray *pooledArbiters);
|
||||||
|
|
||||||
// Linked list ring of contact buffers.
|
// Linked list ring of contact buffers.
|
||||||
// Head is the current buffer. Tail is the oldest buffer.
|
// Head is the newest buffer, and each buffer points to a newer buffer.
|
||||||
// The list points in the direction of tail->head.
|
// Head wraps around and points to the oldest (tail) buffer.
|
||||||
cpContactBufferHeader *contactBuffersHead, *contactBuffersTail;
|
CP_PRIVATE(cpContactBufferHeader *contactBuffersHead);
|
||||||
|
CP_PRIVATE(cpContactBufferHeader *_contactBuffersTail_Deprecated);
|
||||||
|
|
||||||
// List of buffers to be free()ed when destroying the space.
|
// List of buffers to be free()ed when destroying the space.
|
||||||
cpArray *allocatedBuffers;
|
CP_PRIVATE(cpArray *allocatedBuffers);
|
||||||
|
|
||||||
// Persistant contact set.
|
// Persistant contact set.
|
||||||
cpHashSet *contactSet;
|
CP_PRIVATE(cpHashSet *contactSet);
|
||||||
|
|
||||||
// List of constraints in the system.
|
// List of constraints in the system.
|
||||||
cpArray *constraints;
|
CP_PRIVATE(cpArray *constraints);
|
||||||
|
|
||||||
// Set of collisionpair functions.
|
// Set of collisionpair functions.
|
||||||
cpHashSet *collFuncSet;
|
CP_PRIVATE(cpHashSet *collFuncSet);
|
||||||
// Default collision handler.
|
// Default collision handler.
|
||||||
cpCollisionHandler defaultHandler;
|
CP_PRIVATE(cpCollisionHandler defaultHandler);
|
||||||
|
|
||||||
cpHashSet *postStepCallbacks;
|
CP_PRIVATE(cpHashSet *postStepCallbacks);
|
||||||
|
|
||||||
|
cpBody staticBody;
|
||||||
} cpSpace;
|
} cpSpace;
|
||||||
|
|
||||||
// Basic allocation/destruction functions.
|
// Basic allocation/destruction functions.
|
||||||
|
@ -159,13 +176,20 @@ cpShape *cpSpacePointQueryFirst(cpSpace *space, cpVect point, cpLayers layers, c
|
||||||
|
|
||||||
// Segment query callback function
|
// Segment query callback function
|
||||||
typedef void (*cpSpaceSegmentQueryFunc)(cpShape *shape, cpFloat t, cpVect n, void *data);
|
typedef void (*cpSpaceSegmentQueryFunc)(cpShape *shape, cpFloat t, cpVect n, void *data);
|
||||||
int cpSpaceSegmentQuery(cpSpace *space, cpVect start, cpVect end, cpLayers layers, cpGroup group, cpSpaceSegmentQueryFunc func, void *data);
|
void cpSpaceSegmentQuery(cpSpace *space, cpVect start, cpVect end, cpLayers layers, cpGroup group, cpSpaceSegmentQueryFunc func, void *data);
|
||||||
cpShape *cpSpaceSegmentQueryFirst(cpSpace *space, cpVect start, cpVect end, cpLayers layers, cpGroup group, cpSegmentQueryInfo *out);
|
cpShape *cpSpaceSegmentQueryFirst(cpSpace *space, cpVect start, cpVect end, cpLayers layers, cpGroup group, cpSegmentQueryInfo *out);
|
||||||
|
|
||||||
// BB query callback function
|
// BB query callback function
|
||||||
typedef void (*cpSpaceBBQueryFunc)(cpShape *shape, void *data);
|
typedef void (*cpSpaceBBQueryFunc)(cpShape *shape, void *data);
|
||||||
void cpSpaceBBQuery(cpSpace *space, cpBB bb, cpLayers layers, cpGroup group, cpSpaceBBQueryFunc func, void *data);
|
void cpSpaceBBQuery(cpSpace *space, cpBB bb, cpLayers layers, cpGroup group, cpSpaceBBQueryFunc func, void *data);
|
||||||
|
|
||||||
|
// Shape query callback function
|
||||||
|
typedef void (*cpSpaceShapeQueryFunc)(cpShape *shape, cpContactPointSet *points, void *data);
|
||||||
|
cpBool cpSpaceShapeQuery(cpSpace *space, cpShape *shape, cpSpaceShapeQueryFunc func, void *data);
|
||||||
|
|
||||||
|
|
||||||
|
void cpSpaceActivateShapesTouchingShape(cpSpace *space, cpShape *shape);
|
||||||
|
|
||||||
|
|
||||||
// Iterator function for iterating the bodies in a space.
|
// Iterator function for iterating the bodies in a space.
|
||||||
typedef void (*cpSpaceBodyIterator)(cpBody *body, void *data);
|
typedef void (*cpSpaceBodyIterator)(cpBody *body, void *data);
|
||||||
|
@ -176,5 +200,7 @@ void cpSpaceResizeStaticHash(cpSpace *space, cpFloat dim, int count);
|
||||||
void cpSpaceResizeActiveHash(cpSpace *space, cpFloat dim, int count);
|
void cpSpaceResizeActiveHash(cpSpace *space, cpFloat dim, int count);
|
||||||
void cpSpaceRehashStatic(cpSpace *space);
|
void cpSpaceRehashStatic(cpSpace *space);
|
||||||
|
|
||||||
|
void cpSpaceRehashShape(cpSpace *space, cpShape *shape);
|
||||||
|
|
||||||
// Update the space.
|
// Update the space.
|
||||||
void cpSpaceStep(cpSpace *space, cpFloat dt);
|
void cpSpaceStep(cpSpace *space, cpFloat dt);
|
||||||
|
|
|
@ -30,7 +30,7 @@ typedef struct cpHandle{
|
||||||
int retain;
|
int retain;
|
||||||
// Query stamp. Used to make sure two objects
|
// Query stamp. Used to make sure two objects
|
||||||
// aren't identified twice in the same query.
|
// aren't identified twice in the same query.
|
||||||
int stamp;
|
cpTimestamp stamp;
|
||||||
} cpHandle;
|
} cpHandle;
|
||||||
|
|
||||||
// Linked list element for in the chains.
|
// Linked list element for in the chains.
|
||||||
|
@ -44,25 +44,26 @@ typedef cpBB (*cpSpaceHashBBFunc)(void *obj);
|
||||||
|
|
||||||
typedef struct cpSpaceHash{
|
typedef struct cpSpaceHash{
|
||||||
// Number of cells in the table.
|
// Number of cells in the table.
|
||||||
int numcells;
|
CP_PRIVATE(int numcells);
|
||||||
// Dimentions of the cells.
|
// Dimentions of the cells.
|
||||||
cpFloat celldim;
|
CP_PRIVATE(cpFloat celldim);
|
||||||
|
|
||||||
// BBox callback.
|
// BBox callback.
|
||||||
cpSpaceHashBBFunc bbfunc;
|
CP_PRIVATE(cpSpaceHashBBFunc bbfunc);
|
||||||
|
|
||||||
// Hashset of the handles and the recycled ones.
|
// Hashset of the handles and the recycled ones.
|
||||||
cpHashSet *handleSet;
|
CP_PRIVATE(cpHashSet *handleSet);
|
||||||
cpArray *pooledHandles;
|
CP_PRIVATE(cpArray *pooledHandles);
|
||||||
|
|
||||||
// The table and the recycled bins.
|
// The table and the recycled bins.
|
||||||
cpSpaceHashBin **table, *pooledBins;
|
CP_PRIVATE(cpSpaceHashBin **table);
|
||||||
|
CP_PRIVATE(cpSpaceHashBin *pooledBins);
|
||||||
|
|
||||||
// list of buffers to free on destruction.
|
// list of buffers to free on destruction.
|
||||||
cpArray *allocatedBuffers;
|
CP_PRIVATE(cpArray *allocatedBuffers);
|
||||||
|
|
||||||
// Incremented on each query. See cpHandle.stamp.
|
// Incremented on each query. See cpHandle.stamp.
|
||||||
int stamp;
|
CP_PRIVATE(cpTimestamp stamp);
|
||||||
} cpSpaceHash;
|
} cpSpaceHash;
|
||||||
|
|
||||||
//Basic allocation/destruction functions.
|
//Basic allocation/destruction functions.
|
||||||
|
@ -77,7 +78,7 @@ void cpSpaceHashFree(cpSpaceHash *hash);
|
||||||
void cpSpaceHashResize(cpSpaceHash *hash, cpFloat celldim, int numcells);
|
void cpSpaceHashResize(cpSpaceHash *hash, cpFloat celldim, int numcells);
|
||||||
|
|
||||||
// Add an object to the hash.
|
// Add an object to the hash.
|
||||||
void cpSpaceHashInsert(cpSpaceHash *hash, void *obj, cpHashValue id, cpBB bb);
|
void cpSpaceHashInsert(cpSpaceHash *hash, void *obj, cpHashValue id, cpBB _deprecated_ignored);
|
||||||
// Remove an object from the hash.
|
// Remove an object from the hash.
|
||||||
void cpSpaceHashRemove(cpSpaceHash *hash, void *obj, cpHashValue id);
|
void cpSpaceHashRemove(cpSpaceHash *hash, void *obj, cpHashValue id);
|
||||||
|
|
||||||
|
|
|
@ -19,8 +19,10 @@
|
||||||
* SOFTWARE.
|
* SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static const cpVect cpvzero={0.0f,0.0f};
|
/// Constant for the zero vector.
|
||||||
|
static const cpVect cpvzero = {0.0f,0.0f};
|
||||||
|
|
||||||
|
/// Convenience constructor for cpVect structs.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpv(const cpFloat x, const cpFloat y)
|
cpv(const cpFloat x, const cpFloat y)
|
||||||
{
|
{
|
||||||
|
@ -29,128 +31,176 @@ cpv(const cpFloat x, const cpFloat y)
|
||||||
}
|
}
|
||||||
|
|
||||||
// non-inlined functions
|
// non-inlined functions
|
||||||
cpFloat cpvlength(const cpVect v);
|
|
||||||
cpVect cpvslerp(const cpVect v1, const cpVect v2, const cpFloat t);
|
|
||||||
cpVect cpvslerpconst(const cpVect v1, const cpVect v2, const cpFloat a);
|
|
||||||
cpVect cpvforangle(const cpFloat a); // convert radians to a normalized vector
|
|
||||||
cpFloat cpvtoangle(const cpVect v); // convert a vector to radians
|
|
||||||
char *cpvstr(const cpVect v); // get a string representation of a vector
|
|
||||||
|
|
||||||
|
/// Returns the length of v.
|
||||||
|
cpFloat cpvlength(const cpVect v);
|
||||||
|
|
||||||
|
/// Spherical linearly interpolate between v1 and v2.
|
||||||
|
cpVect cpvslerp(const cpVect v1, const cpVect v2, const cpFloat t);
|
||||||
|
|
||||||
|
/// Spherical linearly interpolate between v1 towards v2 by no more than angle a radians
|
||||||
|
cpVect cpvslerpconst(const cpVect v1, const cpVect v2, const cpFloat a);
|
||||||
|
|
||||||
|
/// Returns the unit length vector for the given angle (in radians).
|
||||||
|
cpVect cpvforangle(const cpFloat a);
|
||||||
|
|
||||||
|
/// Returns the angular direction v is pointing in (in radians).
|
||||||
|
cpFloat cpvtoangle(const cpVect v);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Returns a string representation of v. Intended mostly for debugging purposes and not production use.
|
||||||
|
|
||||||
|
@attention The string points to a static local and is reset every time the function is called.
|
||||||
|
If you want to print more than one vector you will have to split up your printing onto separate lines.
|
||||||
|
*/
|
||||||
|
char *cpvstr(const cpVect v);
|
||||||
|
|
||||||
|
/// Check if two vectors are equal. (Be careful when comparing floating point numbers!)
|
||||||
|
static inline cpBool
|
||||||
|
cpveql(const cpVect v1, const cpVect v2)
|
||||||
|
{
|
||||||
|
return (v1.x == v2.x && v1.y == v2.y);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Add two vectors
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvadd(const cpVect v1, const cpVect v2)
|
cpvadd(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return cpv(v1.x + v2.x, v1.y + v2.y);
|
return cpv(v1.x + v2.x, v1.y + v2.y);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Negate a vector.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvneg(const cpVect v)
|
cpvneg(const cpVect v)
|
||||||
{
|
{
|
||||||
return cpv(-v.x, -v.y);
|
return cpv(-v.x, -v.y);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Subtract two vectors.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvsub(const cpVect v1, const cpVect v2)
|
cpvsub(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return cpv(v1.x - v2.x, v1.y - v2.y);
|
return cpv(v1.x - v2.x, v1.y - v2.y);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Scalar multiplication.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvmult(const cpVect v, const cpFloat s)
|
cpvmult(const cpVect v, const cpFloat s)
|
||||||
{
|
{
|
||||||
return cpv(v.x*s, v.y*s);
|
return cpv(v.x*s, v.y*s);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Vector dot product.
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
cpvdot(const cpVect v1, const cpVect v2)
|
cpvdot(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return v1.x*v2.x + v1.y*v2.y;
|
return v1.x*v2.x + v1.y*v2.y;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
2D vector cross product analog.
|
||||||
|
The cross product of 2D vectors results in a 3D vector with only a z component.
|
||||||
|
This function returns the magnitude of the z value.
|
||||||
|
*/
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
cpvcross(const cpVect v1, const cpVect v2)
|
cpvcross(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return v1.x*v2.y - v1.y*v2.x;
|
return v1.x*v2.y - v1.y*v2.x;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns a perpendicular vector. (90 degree rotation)
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvperp(const cpVect v)
|
cpvperp(const cpVect v)
|
||||||
{
|
{
|
||||||
return cpv(-v.y, v.x);
|
return cpv(-v.y, v.x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns a perpendicular vector. (-90 degree rotation)
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvrperp(const cpVect v)
|
cpvrperp(const cpVect v)
|
||||||
{
|
{
|
||||||
return cpv(v.y, -v.x);
|
return cpv(v.y, -v.x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the vector projection of v1 onto v2.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvproject(const cpVect v1, const cpVect v2)
|
cpvproject(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return cpvmult(v2, cpvdot(v1, v2)/cpvdot(v2, v2));
|
return cpvmult(v2, cpvdot(v1, v2)/cpvdot(v2, v2));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Uses complex number multiplication to rotate v1 by v2. Scaling will occur if v1 is not a unit vector.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvrotate(const cpVect v1, const cpVect v2)
|
cpvrotate(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return cpv(v1.x*v2.x - v1.y*v2.y, v1.x*v2.y + v1.y*v2.x);
|
return cpv(v1.x*v2.x - v1.y*v2.y, v1.x*v2.y + v1.y*v2.x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Inverse of cpvrotate().
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvunrotate(const cpVect v1, const cpVect v2)
|
cpvunrotate(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return cpv(v1.x*v2.x + v1.y*v2.y, v1.y*v2.x - v1.x*v2.y);
|
return cpv(v1.x*v2.x + v1.y*v2.y, v1.y*v2.x - v1.x*v2.y);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the squared length of v. Faster than cpvlength() when you only need to compare lengths.
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
cpvlengthsq(const cpVect v)
|
cpvlengthsq(const cpVect v)
|
||||||
{
|
{
|
||||||
return cpvdot(v, v);
|
return cpvdot(v, v);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Linearly interpolate between v1 and v2.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvlerp(const cpVect v1, const cpVect v2, const cpFloat t)
|
cpvlerp(const cpVect v1, const cpVect v2, const cpFloat t)
|
||||||
{
|
{
|
||||||
return cpvadd(cpvmult(v1, 1.0f - t), cpvmult(v2, t));
|
return cpvadd(cpvmult(v1, 1.0f - t), cpvmult(v2, t));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns a normalized copy of v.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvnormalize(const cpVect v)
|
cpvnormalize(const cpVect v)
|
||||||
{
|
{
|
||||||
return cpvmult(v, 1.0f/cpvlength(v));
|
return cpvmult(v, 1.0f/cpvlength(v));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns a normalized copy of v or cpvzero if v was already cpvzero. Protects against divide by zero errors.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvnormalize_safe(const cpVect v)
|
cpvnormalize_safe(const cpVect v)
|
||||||
{
|
{
|
||||||
return (v.x == 0.0f && v.y == 0.0f ? cpvzero : cpvnormalize(v));
|
return (v.x == 0.0f && v.y == 0.0f ? cpvzero : cpvnormalize(v));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Clamp v to length len.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvclamp(const cpVect v, const cpFloat len)
|
cpvclamp(const cpVect v, const cpFloat len)
|
||||||
{
|
{
|
||||||
return (cpvdot(v,v) > len*len) ? cpvmult(cpvnormalize(v), len) : v;
|
return (cpvdot(v,v) > len*len) ? cpvmult(cpvnormalize(v), len) : v;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Linearly interpolate between v1 towards v2 by distance d.
|
||||||
static inline cpVect
|
static inline cpVect
|
||||||
cpvlerpconst(cpVect v1, cpVect v2, cpFloat d)
|
cpvlerpconst(cpVect v1, cpVect v2, cpFloat d)
|
||||||
{
|
{
|
||||||
return cpvadd(v1, cpvclamp(cpvsub(v2, v1), d));
|
return cpvadd(v1, cpvclamp(cpvsub(v2, v1), d));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the distance between v1 and v2.
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
cpvdist(const cpVect v1, const cpVect v2)
|
cpvdist(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return cpvlength(cpvsub(v1, v2));
|
return cpvlength(cpvsub(v1, v2));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the squared distance between v1 and v2. Faster than cpvdist() when you only need to compare distances.
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
cpvdistsq(const cpVect v1, const cpVect v2)
|
cpvdistsq(const cpVect v1, const cpVect v2)
|
||||||
{
|
{
|
||||||
return cpvlengthsq(cpvsub(v1, v2));
|
return cpvlengthsq(cpvsub(v1, v2));
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int
|
/// Returns true if the distance between v1 and v2 is less than dist.
|
||||||
|
static inline cpBool
|
||||||
cpvnear(const cpVect v1, const cpVect v2, const cpFloat dist)
|
cpvnear(const cpVect v1, const cpVect v2, const cpFloat dist)
|
||||||
{
|
{
|
||||||
return cpvdistsq(v1, v2) < dist*dist;
|
return cpvdistsq(v1, v2) < dist*dist;
|
||||||
|
|
|
@ -38,8 +38,11 @@ OBJECTS = \
|
||||||
$(OBJECTS_DIR)/cpHashSet.o \
|
$(OBJECTS_DIR)/cpHashSet.o \
|
||||||
$(OBJECTS_DIR)/cpPolyShape.o \
|
$(OBJECTS_DIR)/cpPolyShape.o \
|
||||||
$(OBJECTS_DIR)/cpShape.o \
|
$(OBJECTS_DIR)/cpShape.o \
|
||||||
$(OBJECTS_DIR)/cpSpace.o \
|
$(OBJECTS_DIR)/cpSpace.o \
|
||||||
$(OBJECTS_DIR)/cpSpaceHash.o \
|
$(OBJECTS_DIR)/cpSpaceComponent.o \
|
||||||
|
$(OBJECTS_DIR)/cpSpaceHash.o \
|
||||||
|
$(OBJECTS_DIR)/cpSpaceQuery.o \
|
||||||
|
$(OBJECTS_DIR)/cpSpaceStep.o \
|
||||||
$(OBJECTS_DIR)/cpVect.o \
|
$(OBJECTS_DIR)/cpVect.o \
|
||||||
$(OBJECTS_DIR)/cpConstraint.o \
|
$(OBJECTS_DIR)/cpConstraint.o \
|
||||||
$(OBJECTS_DIR)/cpDampedRotarySpring.o \
|
$(OBJECTS_DIR)/cpDampedRotarySpring.o \
|
||||||
|
@ -99,9 +102,18 @@ $(OBJECTS_DIR)/cpShape.o : ../src/cpShape.c
|
||||||
|
|
||||||
$(OBJECTS_DIR)/cpSpace.o : ../src/cpSpace.c
|
$(OBJECTS_DIR)/cpSpace.o : ../src/cpSpace.c
|
||||||
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpSpace.o ../src/cpSpace.c
|
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpSpace.o ../src/cpSpace.c
|
||||||
|
|
||||||
|
$(OBJECTS_DIR)/cpSpace.o : ../src/cpSpaceComponent.c
|
||||||
|
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpSpace.o ../src/cpSpaceComponent.c
|
||||||
|
|
||||||
$(OBJECTS_DIR)/cpSpaceHash.o : ../src/cpSpaceHash.c
|
$(OBJECTS_DIR)/cpSpaceHash.o : ../src/cpSpaceHash.c
|
||||||
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpSpaceHash.o ../src/cpSpaceHash.c
|
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpSpaceHash.o ../src/cpSpaceHash.c
|
||||||
|
|
||||||
|
$(OBJECTS_DIR)/cpSpace.o : ../src/cpSpaceQuery.c
|
||||||
|
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpSpace.o ../src/cpSpaceQuery.c
|
||||||
|
|
||||||
|
$(OBJECTS_DIR)/cpSpace.o : ../src/cpSpaceStep.c
|
||||||
|
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpSpace.o ../src/cpSpaceStep.c
|
||||||
|
|
||||||
$(OBJECTS_DIR)/cpVect.o : ../src/cpVect.c
|
$(OBJECTS_DIR)/cpVect.o : ../src/cpVect.c
|
||||||
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpVect.o ../src/cpVect.c
|
$(CC) -c $(CC_FLAGS) $(INCLUDE_PATH) $(LAST_INCLUDE_PATH) -o $(OBJECTS_DIR)/cpVect.o ../src/cpVect.c
|
||||||
|
|
|
@ -9,7 +9,7 @@ if(BUILD_SHARED)
|
||||||
${chipmunk_source_files}
|
${chipmunk_source_files}
|
||||||
)
|
)
|
||||||
# set the lib's version number
|
# set the lib's version number
|
||||||
set_target_properties(chipmunk PROPERTIES VERSION 5.1)
|
set_target_properties(chipmunk PROPERTIES VERSION 5.3.4)
|
||||||
install(TARGETS chipmunk RUNTIME DESTINATION lib LIBRARY DESTINATION lib)
|
install(TARGETS chipmunk RUNTIME DESTINATION lib LIBRARY DESTINATION lib)
|
||||||
endif(BUILD_SHARED)
|
endif(BUILD_SHARED)
|
||||||
|
|
||||||
|
|
|
@ -18,9 +18,11 @@
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
* SOFTWARE.
|
* SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#define _USE_MATH_DEFINES
|
||||||
|
#include <math.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk.h"
|
||||||
|
|
||||||
|
@ -43,7 +45,7 @@ cpMessage(const char *message, const char *condition, const char *file, int line
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
char *cpVersionString = "5.x.x";
|
const char *cpVersionString = "5.3.4";
|
||||||
|
|
||||||
void
|
void
|
||||||
cpInitChipmunk(void)
|
cpInitChipmunk(void)
|
||||||
|
@ -59,7 +61,13 @@ cpInitChipmunk(void)
|
||||||
cpFloat
|
cpFloat
|
||||||
cpMomentForCircle(cpFloat m, cpFloat r1, cpFloat r2, cpVect offset)
|
cpMomentForCircle(cpFloat m, cpFloat r1, cpFloat r2, cpVect offset)
|
||||||
{
|
{
|
||||||
return (1.0f/2.0f)*m*(r1*r1 + r2*r2) + m*cpvdot(offset, offset);
|
return m*(0.5f*(r1*r1 + r2*r2) + cpvlengthsq(offset));
|
||||||
|
}
|
||||||
|
|
||||||
|
cpFloat
|
||||||
|
cpAreaForCircle(cpFloat r1, cpFloat r2)
|
||||||
|
{
|
||||||
|
return 2.0f*(cpFloat)M_PI*cpfabs(r1*r1 - r2*r2);
|
||||||
}
|
}
|
||||||
|
|
||||||
cpFloat
|
cpFloat
|
||||||
|
@ -68,21 +76,23 @@ cpMomentForSegment(cpFloat m, cpVect a, cpVect b)
|
||||||
cpFloat length = cpvlength(cpvsub(b, a));
|
cpFloat length = cpvlength(cpvsub(b, a));
|
||||||
cpVect offset = cpvmult(cpvadd(a, b), 1.0f/2.0f);
|
cpVect offset = cpvmult(cpvadd(a, b), 1.0f/2.0f);
|
||||||
|
|
||||||
return m*length*length/12.0f + m*cpvdot(offset, offset);
|
return m*(length*length/12.0f + cpvlengthsq(offset));
|
||||||
}
|
}
|
||||||
|
|
||||||
cpFloat
|
cpFloat
|
||||||
cpMomentForPoly(cpFloat m, const int numVerts, cpVect *verts, cpVect offset)
|
cpAreaForSegment(cpVect a, cpVect b, cpFloat r)
|
||||||
|
{
|
||||||
|
return 2.0f*r*((cpFloat)M_PI*r + cpvdist(a, b));
|
||||||
|
}
|
||||||
|
|
||||||
|
cpFloat
|
||||||
|
cpMomentForPoly(cpFloat m, const int numVerts, const cpVect *verts, cpVect offset)
|
||||||
{
|
{
|
||||||
cpVect *tVerts = (cpVect *)cpcalloc(numVerts, sizeof(cpVect));
|
|
||||||
for(int i=0; i<numVerts; i++)
|
|
||||||
tVerts[i] = cpvadd(verts[i], offset);
|
|
||||||
|
|
||||||
cpFloat sum1 = 0.0f;
|
cpFloat sum1 = 0.0f;
|
||||||
cpFloat sum2 = 0.0f;
|
cpFloat sum2 = 0.0f;
|
||||||
for(int i=0; i<numVerts; i++){
|
for(int i=0; i<numVerts; i++){
|
||||||
cpVect v1 = tVerts[i];
|
cpVect v1 = cpvadd(verts[i], offset);
|
||||||
cpVect v2 = tVerts[(i+1)%numVerts];
|
cpVect v2 = cpvadd(verts[(i+1)%numVerts], offset);
|
||||||
|
|
||||||
cpFloat a = cpvcross(v2, v1);
|
cpFloat a = cpvcross(v2, v1);
|
||||||
cpFloat b = cpvdot(v1, v1) + cpvdot(v1, v2) + cpvdot(v2, v2);
|
cpFloat b = cpvdot(v1, v1) + cpvdot(v1, v2) + cpvdot(v2, v2);
|
||||||
|
@ -91,15 +101,51 @@ cpMomentForPoly(cpFloat m, const int numVerts, cpVect *verts, cpVect offset)
|
||||||
sum2 += a;
|
sum2 += a;
|
||||||
}
|
}
|
||||||
|
|
||||||
cpfree(tVerts);
|
|
||||||
return (m*sum1)/(6.0f*sum2);
|
return (m*sum1)/(6.0f*sum2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cpFloat
|
||||||
|
cpAreaForPoly(const int numVerts, const cpVect *verts)
|
||||||
|
{
|
||||||
|
cpFloat area = 0.0f;
|
||||||
|
for(int i=0; i<numVerts; i++){
|
||||||
|
area += cpvcross(verts[i], verts[(i+1)%numVerts]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return area/2.0f;
|
||||||
|
}
|
||||||
|
|
||||||
|
cpVect
|
||||||
|
cpCentroidForPoly(const int numVerts, const cpVect *verts)
|
||||||
|
{
|
||||||
|
cpFloat sum = 0.0f;
|
||||||
|
cpVect vsum = cpvzero;
|
||||||
|
|
||||||
|
for(int i=0; i<numVerts; i++){
|
||||||
|
cpVect v1 = verts[i];
|
||||||
|
cpVect v2 = verts[(i+1)%numVerts];
|
||||||
|
cpFloat cross = cpvcross(v1, v2);
|
||||||
|
|
||||||
|
sum += cross;
|
||||||
|
vsum = cpvadd(vsum, cpvmult(cpvadd(v1, v2), cross));
|
||||||
|
}
|
||||||
|
|
||||||
|
return cpvmult(vsum, 1.0f/(3.0f*sum));
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
cpRecenterPoly(const int numVerts, cpVect *verts){
|
||||||
|
cpVect centroid = cpCentroidForPoly(numVerts, verts);
|
||||||
|
|
||||||
|
for(int i=0; i<numVerts; i++){
|
||||||
|
verts[i] = cpvsub(verts[i], centroid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
cpFloat
|
cpFloat
|
||||||
cpMomentForBox(cpFloat m, cpFloat width, cpFloat height)
|
cpMomentForBox(cpFloat m, cpFloat width, cpFloat height)
|
||||||
{
|
{
|
||||||
return m*(width*width + height*height)/12;
|
return m*(width*width + height*height)/12.0f;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#include "chipmunk_ffi.h"
|
#include "chipmunk_ffi.h"
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
// TODO: Comment me!
|
// TODO: Comment me!
|
||||||
|
|
|
@ -22,7 +22,7 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static cpFloat
|
static cpFloat
|
||||||
|
@ -33,12 +33,12 @@ defaultSpringTorque(cpDampedRotarySpring *spring, cpFloat relativeAngle){
|
||||||
static void
|
static void
|
||||||
preStep(cpDampedRotarySpring *spring, cpFloat dt, cpFloat dt_inv)
|
preStep(cpDampedRotarySpring *spring, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = spring->constraint.a;
|
CONSTRAINT_BEGIN(spring, a, b);
|
||||||
cpBody *b = spring->constraint.b;
|
|
||||||
|
|
||||||
spring->iSum = 1.0f/(a->i_inv + b->i_inv);
|
cpFloat moment = a->i_inv + b->i_inv;
|
||||||
|
spring->iSum = 1.0f/moment;
|
||||||
|
|
||||||
spring->dt = dt;
|
spring->w_coef = 1.0f - cpfexp(-spring->damping*dt*moment);
|
||||||
spring->target_wrn = 0.0f;
|
spring->target_wrn = 0.0f;
|
||||||
|
|
||||||
// apply spring torque
|
// apply spring torque
|
||||||
|
@ -50,15 +50,14 @@ preStep(cpDampedRotarySpring *spring, cpFloat dt, cpFloat dt_inv)
|
||||||
static void
|
static void
|
||||||
applyImpulse(cpDampedRotarySpring *spring)
|
applyImpulse(cpDampedRotarySpring *spring)
|
||||||
{
|
{
|
||||||
cpBody *a = spring->constraint.a;
|
CONSTRAINT_BEGIN(spring, a, b);
|
||||||
cpBody *b = spring->constraint.b;
|
|
||||||
|
|
||||||
// compute relative velocity
|
// compute relative velocity
|
||||||
cpFloat wrn = a->w - b->w;//normal_relative_velocity(a, b, r1, r2, n) - spring->target_vrn;
|
cpFloat wrn = a->w - b->w;//normal_relative_velocity(a, b, r1, r2, n) - spring->target_vrn;
|
||||||
|
|
||||||
// compute velocity loss from drag
|
// compute velocity loss from drag
|
||||||
// not 100% certain this is derived correctly, though it makes sense
|
// not 100% certain this is derived correctly, though it makes sense
|
||||||
cpFloat w_damp = wrn*(1.0f - cpfexp(-spring->damping*spring->dt/spring->iSum));
|
cpFloat w_damp = wrn*spring->w_coef;
|
||||||
spring->target_wrn = wrn - w_damp;
|
spring->target_wrn = wrn - w_damp;
|
||||||
|
|
||||||
//apply_impulses(a, b, spring->r1, spring->r2, cpvmult(spring->n, v_damp*spring->nMass));
|
//apply_impulses(a, b, spring->r1, spring->r2, cpvmult(spring->n, v_damp*spring->nMass));
|
||||||
|
|
|
@ -22,7 +22,7 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static cpFloat
|
static cpFloat
|
||||||
|
@ -33,8 +33,7 @@ defaultSpringForce(cpDampedSpring *spring, cpFloat dist){
|
||||||
static void
|
static void
|
||||||
preStep(cpDampedSpring *spring, cpFloat dt, cpFloat dt_inv)
|
preStep(cpDampedSpring *spring, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = spring->constraint.a;
|
CONSTRAINT_BEGIN(spring, a, b);
|
||||||
cpBody *b = spring->constraint.b;
|
|
||||||
|
|
||||||
spring->r1 = cpvrotate(spring->anchr1, a->rot);
|
spring->r1 = cpvrotate(spring->anchr1, a->rot);
|
||||||
spring->r2 = cpvrotate(spring->anchr2, b->rot);
|
spring->r2 = cpvrotate(spring->anchr2, b->rot);
|
||||||
|
@ -43,11 +42,11 @@ preStep(cpDampedSpring *spring, cpFloat dt, cpFloat dt_inv)
|
||||||
cpFloat dist = cpvlength(delta);
|
cpFloat dist = cpvlength(delta);
|
||||||
spring->n = cpvmult(delta, 1.0f/(dist ? dist : INFINITY));
|
spring->n = cpvmult(delta, 1.0f/(dist ? dist : INFINITY));
|
||||||
|
|
||||||
// calculate mass normal
|
cpFloat k = k_scalar(a, b, spring->r1, spring->r2, spring->n);
|
||||||
spring->nMass = 1.0f/k_scalar(a, b, spring->r1, spring->r2, spring->n);
|
spring->nMass = 1.0f/k;
|
||||||
|
|
||||||
spring->dt = dt;
|
|
||||||
spring->target_vrn = 0.0f;
|
spring->target_vrn = 0.0f;
|
||||||
|
spring->v_coef = 1.0f - cpfexp(-spring->damping*dt*k);
|
||||||
|
|
||||||
// apply spring force
|
// apply spring force
|
||||||
cpFloat f_spring = spring->springForceFunc((cpConstraint *)spring, dist);
|
cpFloat f_spring = spring->springForceFunc((cpConstraint *)spring, dist);
|
||||||
|
@ -57,8 +56,7 @@ preStep(cpDampedSpring *spring, cpFloat dt, cpFloat dt_inv)
|
||||||
static void
|
static void
|
||||||
applyImpulse(cpDampedSpring *spring)
|
applyImpulse(cpDampedSpring *spring)
|
||||||
{
|
{
|
||||||
cpBody *a = spring->constraint.a;
|
CONSTRAINT_BEGIN(spring, a, b);
|
||||||
cpBody *b = spring->constraint.b;
|
|
||||||
|
|
||||||
cpVect n = spring->n;
|
cpVect n = spring->n;
|
||||||
cpVect r1 = spring->r1;
|
cpVect r1 = spring->r1;
|
||||||
|
@ -69,7 +67,7 @@ applyImpulse(cpDampedSpring *spring)
|
||||||
|
|
||||||
// compute velocity loss from drag
|
// compute velocity loss from drag
|
||||||
// not 100% certain this is derived correctly, though it makes sense
|
// not 100% certain this is derived correctly, though it makes sense
|
||||||
cpFloat v_damp = -vrn*(1.0f - cpfexp(-spring->damping*spring->dt/spring->nMass));
|
cpFloat v_damp = -vrn*spring->v_coef;
|
||||||
spring->target_vrn = vrn + v_damp;
|
spring->target_vrn = vrn + v_damp;
|
||||||
|
|
||||||
apply_impulses(a, b, spring->r1, spring->r2, cpvmult(spring->n, v_damp*spring->nMass));
|
apply_impulses(a, b, spring->r1, spring->r2, cpvmult(spring->n, v_damp*spring->nMass));
|
||||||
|
|
|
@ -21,14 +21,13 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
preStep(cpGearJoint *joint, cpFloat dt, cpFloat dt_inv)
|
preStep(cpGearJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
// calculate moment of inertia coefficient.
|
// calculate moment of inertia coefficient.
|
||||||
joint->iSum = 1.0f/(a->i_inv*joint->ratio_inv + joint->ratio*b->i_inv);
|
joint->iSum = 1.0f/(a->i_inv*joint->ratio_inv + joint->ratio*b->i_inv);
|
||||||
|
@ -49,8 +48,7 @@ preStep(cpGearJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
static void
|
static void
|
||||||
applyImpulse(cpGearJoint *joint)
|
applyImpulse(cpGearJoint *joint)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
// compute relative rotational velocity
|
// compute relative rotational velocity
|
||||||
cpFloat wr = b->w*joint->ratio - a->w;
|
cpFloat wr = b->w*joint->ratio - a->w;
|
||||||
|
@ -111,4 +109,5 @@ cpGearJointSetRatio(cpConstraint *constraint, cpFloat value)
|
||||||
cpConstraintCheckCast(constraint, cpGearJoint);
|
cpConstraintCheckCast(constraint, cpGearJoint);
|
||||||
((cpGearJoint *)constraint)->ratio = value;
|
((cpGearJoint *)constraint)->ratio = value;
|
||||||
((cpGearJoint *)constraint)->ratio_inv = 1.0f/value;
|
((cpGearJoint *)constraint)->ratio_inv = 1.0f/value;
|
||||||
|
cpConstraintActivateBodies(constraint);
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,14 +21,13 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
preStep(cpGrooveJoint *joint, cpFloat dt, cpFloat dt_inv)
|
preStep(cpGrooveJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
// calculate endpoints in worldspace
|
// calculate endpoints in worldspace
|
||||||
cpVect ta = cpBodyLocal2World(a, joint->grv_a);
|
cpVect ta = cpBodyLocal2World(a, joint->grv_a);
|
||||||
|
@ -79,8 +78,7 @@ grooveConstrain(cpGrooveJoint *joint, cpVect j){
|
||||||
static void
|
static void
|
||||||
applyImpulse(cpGrooveJoint *joint)
|
applyImpulse(cpGrooveJoint *joint)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
cpVect r1 = joint->r1;
|
cpVect r1 = joint->r1;
|
||||||
cpVect r2 = joint->r2;
|
cpVect r2 = joint->r2;
|
||||||
|
@ -136,3 +134,28 @@ cpGrooveJointNew(cpBody *a, cpBody *b, cpVect groove_a, cpVect groove_b, cpVect
|
||||||
{
|
{
|
||||||
return (cpConstraint *)cpGrooveJointInit(cpGrooveJointAlloc(), a, b, groove_a, groove_b, anchr2);
|
return (cpConstraint *)cpGrooveJointInit(cpGrooveJointAlloc(), a, b, groove_a, groove_b, anchr2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
cpGrooveJointSetGrooveA(cpConstraint *constraint, cpVect value)
|
||||||
|
{
|
||||||
|
cpGrooveJoint *g = (cpGrooveJoint *)constraint;
|
||||||
|
cpConstraintCheckCast(constraint, cpGrooveJoint);
|
||||||
|
|
||||||
|
g->grv_a = value;
|
||||||
|
g->grv_n = cpvperp(cpvnormalize(cpvsub(g->grv_b, value)));
|
||||||
|
|
||||||
|
cpConstraintActivateBodies(constraint);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
cpGrooveJointSetGrooveB(cpConstraint *constraint, cpVect value)
|
||||||
|
{
|
||||||
|
cpGrooveJoint *g = (cpGrooveJoint *)constraint;
|
||||||
|
cpConstraintCheckCast(constraint, cpGrooveJoint);
|
||||||
|
|
||||||
|
g->grv_b = value;
|
||||||
|
g->grv_n = cpvperp(cpvnormalize(cpvsub(value, g->grv_a)));
|
||||||
|
|
||||||
|
cpConstraintActivateBodies(constraint);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -22,14 +22,13 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
//#include <math.h>
|
//#include <math.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
preStep(cpPinJoint *joint, cpFloat dt, cpFloat dt_inv)
|
preStep(cpPinJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
joint->r1 = cpvrotate(joint->anchr1, a->rot);
|
joint->r1 = cpvrotate(joint->anchr1, a->rot);
|
||||||
joint->r2 = cpvrotate(joint->anchr2, b->rot);
|
joint->r2 = cpvrotate(joint->anchr2, b->rot);
|
||||||
|
@ -56,8 +55,7 @@ preStep(cpPinJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
static void
|
static void
|
||||||
applyImpulse(cpPinJoint *joint)
|
applyImpulse(cpPinJoint *joint)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
cpVect n = joint->n;
|
cpVect n = joint->n;
|
||||||
|
|
||||||
// compute relative velocity
|
// compute relative velocity
|
||||||
|
@ -101,8 +99,9 @@ cpPinJointInit(cpPinJoint *joint, cpBody *a, cpBody *b, cpVect anchr1, cpVect an
|
||||||
joint->anchr1 = anchr1;
|
joint->anchr1 = anchr1;
|
||||||
joint->anchr2 = anchr2;
|
joint->anchr2 = anchr2;
|
||||||
|
|
||||||
cpVect p1 = cpvadd(a->p, cpvrotate(anchr1, a->rot));
|
// STATIC_BODY_CHECK
|
||||||
cpVect p2 = cpvadd(b->p, cpvrotate(anchr2, b->rot));
|
cpVect p1 = (a ? cpvadd(a->p, cpvrotate(anchr1, a->rot)) : anchr1);
|
||||||
|
cpVect p2 = (b ? cpvadd(b->p, cpvrotate(anchr2, b->rot)) : anchr2);
|
||||||
joint->dist = cpvlength(cpvsub(p2, p1));
|
joint->dist = cpvlength(cpvsub(p2, p1));
|
||||||
|
|
||||||
joint->jnAcc = 0.0f;
|
joint->jnAcc = 0.0f;
|
||||||
|
|
|
@ -21,14 +21,13 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
preStep(cpPivotJoint *joint, cpFloat dt, cpFloat dt_inv)
|
preStep(cpPivotJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
joint->r1 = cpvrotate(joint->anchr1, a->rot);
|
joint->r1 = cpvrotate(joint->anchr1, a->rot);
|
||||||
joint->r2 = cpvrotate(joint->anchr2, b->rot);
|
joint->r2 = cpvrotate(joint->anchr2, b->rot);
|
||||||
|
@ -50,8 +49,7 @@ preStep(cpPivotJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
static void
|
static void
|
||||||
applyImpulse(cpPivotJoint *joint)
|
applyImpulse(cpPivotJoint *joint)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
cpVect r1 = joint->r1;
|
cpVect r1 = joint->r1;
|
||||||
cpVect r2 = joint->r2;
|
cpVect r2 = joint->r2;
|
||||||
|
@ -110,5 +108,7 @@ cpPivotJointNew2(cpBody *a, cpBody *b, cpVect anchr1, cpVect anchr2)
|
||||||
cpConstraint *
|
cpConstraint *
|
||||||
cpPivotJointNew(cpBody *a, cpBody *b, cpVect pivot)
|
cpPivotJointNew(cpBody *a, cpBody *b, cpVect pivot)
|
||||||
{
|
{
|
||||||
return cpPivotJointNew2(a, b, cpBodyWorld2Local(a, pivot), cpBodyWorld2Local(b, pivot));
|
cpVect anchr1 = (a ? cpBodyWorld2Local(a, pivot) : pivot);
|
||||||
|
cpVect anchr2 = (b ? cpBodyWorld2Local(b, pivot) : pivot);
|
||||||
|
return cpPivotJointNew2(a, b, anchr1, anchr2);
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,18 +19,16 @@
|
||||||
* SOFTWARE.
|
* SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
preStep(cpRatchetJoint *joint, cpFloat dt, cpFloat dt_inv)
|
preStep(cpRatchetJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
cpFloat angle = joint->angle;
|
cpFloat angle = joint->angle;
|
||||||
cpFloat phase = joint->phase;
|
cpFloat phase = joint->phase;
|
||||||
|
@ -70,8 +68,7 @@ applyImpulse(cpRatchetJoint *joint)
|
||||||
{
|
{
|
||||||
if(!joint->bias) return; // early exit
|
if(!joint->bias) return; // early exit
|
||||||
|
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
// compute relative rotational velocity
|
// compute relative rotational velocity
|
||||||
cpFloat wr = b->w - a->w;
|
cpFloat wr = b->w - a->w;
|
||||||
|
@ -116,7 +113,8 @@ cpRatchetJointInit(cpRatchetJoint *joint, cpBody *a, cpBody *b, cpFloat phase, c
|
||||||
joint->phase = phase;
|
joint->phase = phase;
|
||||||
joint->ratchet = ratchet;
|
joint->ratchet = ratchet;
|
||||||
|
|
||||||
joint->angle = b->a - a->a;
|
// STATIC_BODY_CHECK
|
||||||
|
joint->angle = (b ? b->a : 0.0f) - (a ? a->a : 0.0f);
|
||||||
|
|
||||||
return joint;
|
return joint;
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,14 +21,13 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
preStep(cpRotaryLimitJoint *joint, cpFloat dt, cpFloat dt_inv)
|
preStep(cpRotaryLimitJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
cpFloat dist = b->a - a->a;
|
cpFloat dist = b->a - a->a;
|
||||||
cpFloat pdist = 0.0f;
|
cpFloat pdist = 0.0f;
|
||||||
|
@ -62,8 +61,7 @@ applyImpulse(cpRotaryLimitJoint *joint)
|
||||||
{
|
{
|
||||||
if(!joint->bias) return; // early exit
|
if(!joint->bias) return; // early exit
|
||||||
|
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
// compute relative rotational velocity
|
// compute relative rotational velocity
|
||||||
cpFloat wr = b->w - a->w;
|
cpFloat wr = b->w - a->w;
|
||||||
|
|
|
@ -21,14 +21,13 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
preStep(cpSimpleMotor *joint, cpFloat dt, cpFloat dt_inv)
|
preStep(cpSimpleMotor *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
// calculate moment of inertia coefficient.
|
// calculate moment of inertia coefficient.
|
||||||
joint->iSum = 1.0f/(a->i_inv + b->i_inv);
|
joint->iSum = 1.0f/(a->i_inv + b->i_inv);
|
||||||
|
@ -44,8 +43,7 @@ preStep(cpSimpleMotor *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
static void
|
static void
|
||||||
applyImpulse(cpSimpleMotor *joint)
|
applyImpulse(cpSimpleMotor *joint)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
// compute relative rotational velocity
|
// compute relative rotational velocity
|
||||||
cpFloat wr = b->w - a->w + joint->rate;
|
cpFloat wr = b->w - a->w + joint->rate;
|
||||||
|
|
|
@ -21,14 +21,13 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
static void
|
static void
|
||||||
preStep(cpSlideJoint *joint, cpFloat dt, cpFloat dt_inv)
|
preStep(cpSlideJoint *joint, cpFloat dt, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
joint->r1 = cpvrotate(joint->anchr1, a->rot);
|
joint->r1 = cpvrotate(joint->anchr1, a->rot);
|
||||||
joint->r2 = cpvrotate(joint->anchr2, b->rot);
|
joint->r2 = cpvrotate(joint->anchr2, b->rot);
|
||||||
|
@ -69,8 +68,7 @@ applyImpulse(cpSlideJoint *joint)
|
||||||
{
|
{
|
||||||
if(!joint->bias) return; // early exit
|
if(!joint->bias) return; // early exit
|
||||||
|
|
||||||
cpBody *a = joint->constraint.a;
|
CONSTRAINT_BEGIN(joint, a, b);
|
||||||
cpBody *b = joint->constraint.b;
|
|
||||||
|
|
||||||
cpVect n = joint->n;
|
cpVect n = joint->n;
|
||||||
cpVect r1 = joint->r1;
|
cpVect r1 = joint->r1;
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "constraints/util.h"
|
#include "constraints/util.h"
|
||||||
|
|
||||||
cpFloat cp_bias_coef = 0.1f;
|
cpFloat cp_bias_coef = 0.1f;
|
||||||
|
@ -104,13 +104,20 @@ cpArbiterAlloc(void)
|
||||||
cpArbiter*
|
cpArbiter*
|
||||||
cpArbiterInit(cpArbiter *arb, cpShape *a, cpShape *b)
|
cpArbiterInit(cpArbiter *arb, cpShape *a, cpShape *b)
|
||||||
{
|
{
|
||||||
|
arb->handler = NULL;
|
||||||
|
arb->swappedColl = cpFalse;
|
||||||
|
|
||||||
|
arb->e = 0.0f;
|
||||||
|
arb->u = 0.0f;
|
||||||
|
arb->surface_vr = cpvzero;
|
||||||
|
|
||||||
arb->numContacts = 0;
|
arb->numContacts = 0;
|
||||||
arb->contacts = NULL;
|
arb->contacts = NULL;
|
||||||
|
|
||||||
arb->private_a = a;
|
arb->a = a;
|
||||||
arb->private_b = b;
|
arb->b = b;
|
||||||
|
|
||||||
arb->stamp = -1;
|
arb->stamp = 0;
|
||||||
arb->state = cpArbiterStateFirstColl;
|
arb->state = cpArbiterStateFirstColl;
|
||||||
|
|
||||||
return arb;
|
return arb;
|
||||||
|
@ -157,8 +164,6 @@ cpArbiterUpdate(cpArbiter *arb, cpContact *contacts, int numContacts, cpCollisio
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// cpfree(arb->contacts);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
arb->contacts = contacts;
|
arb->contacts = contacts;
|
||||||
|
@ -172,17 +177,18 @@ cpArbiterUpdate(cpArbiter *arb, cpContact *contacts, int numContacts, cpCollisio
|
||||||
arb->surface_vr = cpvsub(a->surface_v, b->surface_v);
|
arb->surface_vr = cpvsub(a->surface_v, b->surface_v);
|
||||||
|
|
||||||
// For collisions between two similar primitive types, the order could have been swapped.
|
// For collisions between two similar primitive types, the order could have been swapped.
|
||||||
arb->private_a = a; arb->private_b = b;
|
arb->a = a;
|
||||||
|
arb->b = b;
|
||||||
|
|
||||||
|
// mark it as new if it's been cached
|
||||||
|
if(arb->state == cpArbiterStateCached) arb->state = cpArbiterStateFirstColl;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
cpArbiterPreStep(cpArbiter *arb, cpFloat dt_inv)
|
cpArbiterPreStep(cpArbiter *arb, cpFloat dt_inv)
|
||||||
{
|
{
|
||||||
cpShape *shapea = arb->private_a;
|
cpBody *a = arb->a->body;
|
||||||
cpShape *shapeb = arb->private_b;
|
cpBody *b = arb->b->body;
|
||||||
|
|
||||||
cpBody *a = shapea->body;
|
|
||||||
cpBody *b = shapeb->body;
|
|
||||||
|
|
||||||
for(int i=0; i<arb->numContacts; i++){
|
for(int i=0; i<arb->numContacts; i++){
|
||||||
cpContact *con = &arb->contacts[i];
|
cpContact *con = &arb->contacts[i];
|
||||||
|
@ -207,8 +213,8 @@ cpArbiterPreStep(cpArbiter *arb, cpFloat dt_inv)
|
||||||
void
|
void
|
||||||
cpArbiterApplyCachedImpulse(cpArbiter *arb)
|
cpArbiterApplyCachedImpulse(cpArbiter *arb)
|
||||||
{
|
{
|
||||||
cpShape *shapea = arb->private_a;
|
cpShape *shapea = arb->a;
|
||||||
cpShape *shapeb = arb->private_b;
|
cpShape *shapeb = arb->b;
|
||||||
|
|
||||||
arb->u = shapea->u * shapeb->u;
|
arb->u = shapea->u * shapeb->u;
|
||||||
arb->surface_vr = cpvsub(shapeb->surface_v, shapea->surface_v);
|
arb->surface_vr = cpvsub(shapeb->surface_v, shapea->surface_v);
|
||||||
|
@ -225,8 +231,8 @@ cpArbiterApplyCachedImpulse(cpArbiter *arb)
|
||||||
void
|
void
|
||||||
cpArbiterApplyImpulse(cpArbiter *arb, cpFloat eCoef)
|
cpArbiterApplyImpulse(cpArbiter *arb, cpFloat eCoef)
|
||||||
{
|
{
|
||||||
cpBody *a = arb->private_a->body;
|
cpBody *a = arb->a->body;
|
||||||
cpBody *b = arb->private_b->body;
|
cpBody *b = arb->b->body;
|
||||||
|
|
||||||
for(int i=0; i<arb->numContacts; i++){
|
for(int i=0; i<arb->numContacts; i++){
|
||||||
cpContact *con = &arb->contacts[i];
|
cpContact *con = &arb->contacts[i];
|
||||||
|
|
|
@ -22,7 +22,7 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
|
|
||||||
|
|
||||||
//#define CP_ARRAY_INCREMENT 10
|
//#define CP_ARRAY_INCREMENT 10
|
||||||
|
@ -57,6 +57,7 @@ void
|
||||||
cpArrayDestroy(cpArray *arr)
|
cpArrayDestroy(cpArray *arr)
|
||||||
{
|
{
|
||||||
cpfree(arr->arr);
|
cpfree(arr->arr);
|
||||||
|
arr->arr = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -111,6 +112,20 @@ cpArrayDeleteObj(cpArray *arr, void *obj)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
cpArrayAppend(cpArray *arr, cpArray *other)
|
||||||
|
{
|
||||||
|
void *tail = &arr->arr[arr->num];
|
||||||
|
|
||||||
|
arr->num += other->num;
|
||||||
|
if(arr->num >= arr->max){
|
||||||
|
arr->max = arr->num;
|
||||||
|
arr->arr = (void **)cprealloc(arr->arr, arr->max*sizeof(void**));
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(tail, other->arr, other->num*sizeof(void**));
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
cpArrayEach(cpArray *arr, cpArrayIter iterFunc, void *data)
|
cpArrayEach(cpArray *arr, cpArrayIter iterFunc, void *data)
|
||||||
{
|
{
|
||||||
|
@ -118,11 +133,11 @@ cpArrayEach(cpArray *arr, cpArrayIter iterFunc, void *data)
|
||||||
iterFunc(arr->arr[i], data);
|
iterFunc(arr->arr[i], data);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
cpBool
|
||||||
cpArrayContains(cpArray *arr, void *ptr)
|
cpArrayContains(cpArray *arr, void *ptr)
|
||||||
{
|
{
|
||||||
for(int i=0; i<arr->num; i++)
|
for(int i=0; i<arr->num; i++)
|
||||||
if(arr->arr[i] == ptr) return 1;
|
if(arr->arr[i] == ptr) return cpTrue;
|
||||||
|
|
||||||
return 0;
|
return cpFalse;
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,8 +21,12 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <float.h>
|
#include <float.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
|
|
||||||
|
// initialized in cpInitChipmunk()
|
||||||
|
cpBody cpStaticBodySingleton;
|
||||||
|
|
||||||
cpBody*
|
cpBody*
|
||||||
cpBodyAlloc(void)
|
cpBodyAlloc(void)
|
||||||
|
@ -56,8 +60,13 @@ cpBodyInit(cpBody *body, cpFloat m, cpFloat i)
|
||||||
body->data = NULL;
|
body->data = NULL;
|
||||||
body->v_limit = (cpFloat)INFINITY;
|
body->v_limit = (cpFloat)INFINITY;
|
||||||
body->w_limit = (cpFloat)INFINITY;
|
body->w_limit = (cpFloat)INFINITY;
|
||||||
// body->active = 1;
|
|
||||||
|
body->space = NULL;
|
||||||
|
body->shapesList = NULL;
|
||||||
|
|
||||||
|
cpComponentNode node = {NULL, NULL, 0, 0.0f};
|
||||||
|
body->node = node;
|
||||||
|
|
||||||
return body;
|
return body;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -67,6 +76,21 @@ cpBodyNew(cpFloat m, cpFloat i)
|
||||||
return cpBodyInit(cpBodyAlloc(), m, i);
|
return cpBodyInit(cpBodyAlloc(), m, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cpBody *
|
||||||
|
cpBodyInitStatic(cpBody *body)
|
||||||
|
{
|
||||||
|
cpBodyInit(body, (cpFloat)INFINITY, (cpFloat)INFINITY);
|
||||||
|
body->node.idleTime = (cpFloat)INFINITY;
|
||||||
|
|
||||||
|
return body;
|
||||||
|
}
|
||||||
|
|
||||||
|
cpBody *
|
||||||
|
cpBodyNewStatic()
|
||||||
|
{
|
||||||
|
return cpBodyInitStatic(cpBodyAlloc());
|
||||||
|
}
|
||||||
|
|
||||||
void cpBodyDestroy(cpBody *body){}
|
void cpBodyDestroy(cpBody *body){}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -166,23 +190,3 @@ cpApplyDampedSpring(cpBody *a, cpBody *b, cpVect anchr1, cpVect anchr2, cpFloat
|
||||||
cpBodyApplyForce(a, f, r1);
|
cpBodyApplyForce(a, f, r1);
|
||||||
cpBodyApplyForce(b, cpvneg(f), r2);
|
cpBodyApplyForce(b, cpvneg(f), r2);
|
||||||
}
|
}
|
||||||
|
|
||||||
//int
|
|
||||||
//cpBodyMarkLowEnergy(cpBody *body, cpFloat dvsq, int max)
|
|
||||||
//{
|
|
||||||
// cpFloat ke = body->m*cpvdot(body->v, body->v);
|
|
||||||
// cpFloat re = body->i*body->w*body->w;
|
|
||||||
//
|
|
||||||
// if(ke + re > body->m*dvsq)
|
|
||||||
// body->active = 1;
|
|
||||||
// else if(body->active)
|
|
||||||
// body->active = (body->active + 1)%(max + 1);
|
|
||||||
// else {
|
|
||||||
// body->v = cpvzero;
|
|
||||||
// body->v_bias = cpvzero;
|
|
||||||
// body->w = 0.0f;
|
|
||||||
// body->w_bias = 0.0f;
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// return body->active;
|
|
||||||
//}
|
|
||||||
|
|
|
@ -21,16 +21,16 @@
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
#include <stdio.h>
|
//#include <stdio.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
|
|
||||||
typedef int (*collisionFunc)(cpShape *, cpShape *, cpContact *);
|
typedef int (*collisionFunc)(const cpShape *, const cpShape *, cpContact *);
|
||||||
|
|
||||||
// Add contact points for circle to circle collisions.
|
// Add contact points for circle to circle collisions.
|
||||||
// Used by several collision tests.
|
// Used by several collision tests.
|
||||||
static int
|
static int
|
||||||
circle2circleQuery(cpVect p1, cpVect p2, cpFloat r1, cpFloat r2, cpContact *con)
|
circle2circleQuery(const cpVect p1, const cpVect p2, const cpFloat r1, const cpFloat r2, cpContact *con)
|
||||||
{
|
{
|
||||||
cpFloat mindist = r1 + r2;
|
cpFloat mindist = r1 + r2;
|
||||||
cpVect delta = cpvsub(p2, p1);
|
cpVect delta = cpvsub(p2, p1);
|
||||||
|
@ -38,14 +38,12 @@ circle2circleQuery(cpVect p1, cpVect p2, cpFloat r1, cpFloat r2, cpContact *con)
|
||||||
if(distsq >= mindist*mindist) return 0;
|
if(distsq >= mindist*mindist) return 0;
|
||||||
|
|
||||||
cpFloat dist = cpfsqrt(distsq);
|
cpFloat dist = cpfsqrt(distsq);
|
||||||
// To avoid singularities, do nothing in the case of dist = 0.
|
|
||||||
cpFloat non_zero_dist = (dist ? dist : INFINITY);
|
|
||||||
|
|
||||||
// Allocate and initialize the contact.
|
// Allocate and initialize the contact.
|
||||||
cpContactInit(
|
cpContactInit(
|
||||||
con,
|
con,
|
||||||
cpvadd(p1, cpvmult(delta, 0.5f + (r1 - 0.5f*mindist)/non_zero_dist)),
|
cpvadd(p1, cpvmult(delta, 0.5f + (r1 - 0.5f*mindist)/(dist ? dist : INFINITY))),
|
||||||
cpvmult(delta, 1.0f/non_zero_dist),
|
(dist ? cpvmult(delta, 1.0f/dist) : cpv(1.0f, 0.0f)),
|
||||||
dist - mindist,
|
dist - mindist,
|
||||||
0
|
0
|
||||||
);
|
);
|
||||||
|
@ -55,7 +53,7 @@ circle2circleQuery(cpVect p1, cpVect p2, cpFloat r1, cpFloat r2, cpContact *con)
|
||||||
|
|
||||||
// Collide circle shapes.
|
// Collide circle shapes.
|
||||||
static int
|
static int
|
||||||
circle2circle(cpShape *shape1, cpShape *shape2, cpContact *arr)
|
circle2circle(const cpShape *shape1, const cpShape *shape2, cpContact *arr)
|
||||||
{
|
{
|
||||||
cpCircleShape *circ1 = (cpCircleShape *)shape1;
|
cpCircleShape *circ1 = (cpCircleShape *)shape1;
|
||||||
cpCircleShape *circ2 = (cpCircleShape *)shape2;
|
cpCircleShape *circ2 = (cpCircleShape *)shape2;
|
||||||
|
@ -65,7 +63,7 @@ circle2circle(cpShape *shape1, cpShape *shape2, cpContact *arr)
|
||||||
|
|
||||||
// Collide circles to segment shapes.
|
// Collide circles to segment shapes.
|
||||||
static int
|
static int
|
||||||
circle2segment(cpShape *circleShape, cpShape *segmentShape, cpContact *con)
|
circle2segment(const cpShape *circleShape, const cpShape *segmentShape, cpContact *con)
|
||||||
{
|
{
|
||||||
cpCircleShape *circ = (cpCircleShape *)circleShape;
|
cpCircleShape *circ = (cpCircleShape *)circleShape;
|
||||||
cpSegmentShape *seg = (cpSegmentShape *)segmentShape;
|
cpSegmentShape *seg = (cpSegmentShape *)segmentShape;
|
||||||
|
@ -118,17 +116,19 @@ circle2segment(cpShape *circleShape, cpShape *segmentShape, cpContact *con)
|
||||||
static cpContact *
|
static cpContact *
|
||||||
nextContactPoint(cpContact *arr, int *numPtr)
|
nextContactPoint(cpContact *arr, int *numPtr)
|
||||||
{
|
{
|
||||||
int num = *numPtr;
|
int index = *numPtr;
|
||||||
|
|
||||||
if(num <= CP_MAX_CONTACTS_PER_ARBITER)
|
if(index < CP_MAX_CONTACTS_PER_ARBITER){
|
||||||
(*numPtr) = num + 1;
|
(*numPtr) = index + 1;
|
||||||
|
return &arr[index];
|
||||||
return &arr[num];
|
} else {
|
||||||
|
return &arr[CP_MAX_CONTACTS_PER_ARBITER - 1];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find the minimum separating axis for the give poly and axis list.
|
// Find the minimum separating axis for the give poly and axis list.
|
||||||
static inline int
|
static inline int
|
||||||
findMSA(cpPolyShape *poly, cpPolyShapeAxis *axes, int num, cpFloat *min_out)
|
findMSA(const cpPolyShape *poly, const cpPolyShapeAxis *axes, const int num, cpFloat *min_out)
|
||||||
{
|
{
|
||||||
int min_index = 0;
|
int min_index = 0;
|
||||||
cpFloat min = cpPolyShapeValueOnAxis(poly, axes->n, axes->d);
|
cpFloat min = cpPolyShapeValueOnAxis(poly, axes->n, axes->d);
|
||||||
|
@ -148,9 +148,11 @@ findMSA(cpPolyShape *poly, cpPolyShapeAxis *axes, int num, cpFloat *min_out)
|
||||||
return min_index;
|
return min_index;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add contacts for penetrating vertexes.
|
// Add contacts for probably penetrating vertexes.
|
||||||
|
// This handles the degenerate case where an overlap was detected, but no vertexes fall inside
|
||||||
|
// the opposing polygon. (like a star of david)
|
||||||
static inline int
|
static inline int
|
||||||
findVerts(cpContact *arr, cpPolyShape *poly1, cpPolyShape *poly2, cpVect n, cpFloat dist)
|
findVertsFallback(cpContact *arr, const cpPolyShape *poly1, const cpPolyShape *poly2, const cpVect n, const cpFloat dist)
|
||||||
{
|
{
|
||||||
int num = 0;
|
int num = 0;
|
||||||
|
|
||||||
|
@ -166,15 +168,33 @@ findVerts(cpContact *arr, cpPolyShape *poly1, cpPolyShape *poly2, cpVect n, cpFl
|
||||||
cpContactInit(nextContactPoint(arr, &num), v, n, dist, CP_HASH_PAIR(poly2->shape.hashid, i));
|
cpContactInit(nextContactPoint(arr, &num), v, n, dist, CP_HASH_PAIR(poly2->shape.hashid, i));
|
||||||
}
|
}
|
||||||
|
|
||||||
// if(!num)
|
|
||||||
// addContactPoint(arr, &size, &num, cpContactNew(shape1->body->p, n, dist, 0));
|
|
||||||
|
|
||||||
return num;
|
return num;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Add contacts for penetrating vertexes.
|
||||||
|
static inline int
|
||||||
|
findVerts(cpContact *arr, const cpPolyShape *poly1, const cpPolyShape *poly2, const cpVect n, const cpFloat dist)
|
||||||
|
{
|
||||||
|
int num = 0;
|
||||||
|
|
||||||
|
for(int i=0; i<poly1->numVerts; i++){
|
||||||
|
cpVect v = poly1->tVerts[i];
|
||||||
|
if(cpPolyShapeContainsVert(poly2, v))
|
||||||
|
cpContactInit(nextContactPoint(arr, &num), v, n, dist, CP_HASH_PAIR(poly1->shape.hashid, i));
|
||||||
|
}
|
||||||
|
|
||||||
|
for(int i=0; i<poly2->numVerts; i++){
|
||||||
|
cpVect v = poly2->tVerts[i];
|
||||||
|
if(cpPolyShapeContainsVert(poly1, v))
|
||||||
|
cpContactInit(nextContactPoint(arr, &num), v, n, dist, CP_HASH_PAIR(poly2->shape.hashid, i));
|
||||||
|
}
|
||||||
|
|
||||||
|
return (num ? num : findVertsFallback(arr, poly1, poly2, n, dist));
|
||||||
|
}
|
||||||
|
|
||||||
// Collide poly shapes together.
|
// Collide poly shapes together.
|
||||||
static int
|
static int
|
||||||
poly2poly(cpShape *shape1, cpShape *shape2, cpContact *arr)
|
poly2poly(const cpShape *shape1, const cpShape *shape2, cpContact *arr)
|
||||||
{
|
{
|
||||||
cpPolyShape *poly1 = (cpPolyShape *)shape1;
|
cpPolyShape *poly1 = (cpPolyShape *)shape1;
|
||||||
cpPolyShape *poly2 = (cpPolyShape *)shape2;
|
cpPolyShape *poly2 = (cpPolyShape *)shape2;
|
||||||
|
@ -196,7 +216,7 @@ poly2poly(cpShape *shape1, cpShape *shape2, cpContact *arr)
|
||||||
|
|
||||||
// Like cpPolyValueOnAxis(), but for segments.
|
// Like cpPolyValueOnAxis(), but for segments.
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
segValueOnAxis(cpSegmentShape *seg, cpVect n, cpFloat d)
|
segValueOnAxis(const cpSegmentShape *seg, const cpVect n, const cpFloat d)
|
||||||
{
|
{
|
||||||
cpFloat a = cpvdot(n, seg->ta) - seg->r;
|
cpFloat a = cpvdot(n, seg->ta) - seg->r;
|
||||||
cpFloat b = cpvdot(n, seg->tb) - seg->r;
|
cpFloat b = cpvdot(n, seg->tb) - seg->r;
|
||||||
|
@ -205,7 +225,7 @@ segValueOnAxis(cpSegmentShape *seg, cpVect n, cpFloat d)
|
||||||
|
|
||||||
// Identify vertexes that have penetrated the segment.
|
// Identify vertexes that have penetrated the segment.
|
||||||
static inline void
|
static inline void
|
||||||
findPointsBehindSeg(cpContact *arr, int *num, cpSegmentShape *seg, cpPolyShape *poly, cpFloat pDist, cpFloat coef)
|
findPointsBehindSeg(cpContact *arr, int *num, const cpSegmentShape *seg, const cpPolyShape *poly, const cpFloat pDist, const cpFloat coef)
|
||||||
{
|
{
|
||||||
cpFloat dta = cpvcross(seg->tn, seg->ta);
|
cpFloat dta = cpvcross(seg->tn, seg->ta);
|
||||||
cpFloat dtb = cpvcross(seg->tn, seg->tb);
|
cpFloat dtb = cpvcross(seg->tn, seg->tb);
|
||||||
|
@ -225,7 +245,7 @@ findPointsBehindSeg(cpContact *arr, int *num, cpSegmentShape *seg, cpPolyShape *
|
||||||
// This one is complicated and gross. Just don't go there...
|
// This one is complicated and gross. Just don't go there...
|
||||||
// TODO: Comment me!
|
// TODO: Comment me!
|
||||||
static int
|
static int
|
||||||
seg2poly(cpShape *shape1, cpShape *shape2, cpContact *arr)
|
seg2poly(const cpShape *shape1, const cpShape *shape2, cpContact *arr)
|
||||||
{
|
{
|
||||||
cpSegmentShape *seg = (cpSegmentShape *)shape1;
|
cpSegmentShape *seg = (cpSegmentShape *)shape1;
|
||||||
cpPolyShape *poly = (cpPolyShape *)shape2;
|
cpPolyShape *poly = (cpPolyShape *)shape2;
|
||||||
|
@ -294,7 +314,7 @@ seg2poly(cpShape *shape1, cpShape *shape2, cpContact *arr)
|
||||||
// This one is less gross, but still gross.
|
// This one is less gross, but still gross.
|
||||||
// TODO: Comment me!
|
// TODO: Comment me!
|
||||||
static int
|
static int
|
||||||
circle2poly(cpShape *shape1, cpShape *shape2, cpContact *con)
|
circle2poly(const cpShape *shape1, const cpShape *shape2, cpContact *con)
|
||||||
{
|
{
|
||||||
cpCircleShape *circ = (cpCircleShape *)shape1;
|
cpCircleShape *circ = (cpCircleShape *)shape1;
|
||||||
cpPolyShape *poly = (cpPolyShape *)shape2;
|
cpPolyShape *poly = (cpPolyShape *)shape2;
|
||||||
|
@ -352,7 +372,7 @@ circle2poly(cpShape *shape1, cpShape *shape2, cpContact *con)
|
||||||
static collisionFunc *colfuncs = NULL;
|
static collisionFunc *colfuncs = NULL;
|
||||||
|
|
||||||
static void
|
static void
|
||||||
addColFunc(cpShapeType a, cpShapeType b, collisionFunc func)
|
addColFunc(const cpShapeType a, const cpShapeType b, const collisionFunc func)
|
||||||
{
|
{
|
||||||
colfuncs[a + b*CP_NUM_SHAPES] = func;
|
colfuncs[a + b*CP_NUM_SHAPES] = func;
|
||||||
}
|
}
|
||||||
|
@ -381,7 +401,7 @@ extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
int
|
int
|
||||||
cpCollideShapes(cpShape *a, cpShape *b, cpContact *arr)
|
cpCollideShapes(const cpShape *a, const cpShape *b, cpContact *arr)
|
||||||
{
|
{
|
||||||
// Their shape types must be in order.
|
// Their shape types must be in order.
|
||||||
cpAssert(a->klass->type <= b->klass->type, "Collision shapes passed to cpCollideShapes() are not sorted.");
|
cpAssert(a->klass->type <= b->klass->type, "Collision shapes passed to cpCollideShapes() are not sorted.");
|
||||||
|
|
|
@ -22,7 +22,7 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "prime.h"
|
#include "prime.h"
|
||||||
|
|
||||||
static void freeWrap(void *ptr, void *unused){cpfree(ptr);}
|
static void freeWrap(void *ptr, void *unused){cpfree(ptr);}
|
||||||
|
|
|
@ -20,9 +20,8 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "chipmunk_unsafe.h"
|
#include "chipmunk_unsafe.h"
|
||||||
|
|
||||||
cpPolyShape *
|
cpPolyShape *
|
||||||
|
@ -94,7 +93,7 @@ cpPolyShapeDestroy(cpShape *shape)
|
||||||
cpfree(poly->tAxes);
|
cpfree(poly->tAxes);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static cpBool
|
||||||
cpPolyShapePointQuery(cpShape *shape, cpVect p){
|
cpPolyShapePointQuery(cpShape *shape, cpVect p){
|
||||||
return cpBBcontainsVect(shape->bb, p) && cpPolyShapeContainsVert((cpPolyShape *)shape, p);
|
return cpBBcontainsVect(shape->bb, p) && cpPolyShapeContainsVert((cpPolyShape *)shape, p);
|
||||||
}
|
}
|
||||||
|
@ -137,8 +136,8 @@ static const cpShapeClass polyClass = {
|
||||||
cpPolyShapeSegmentQuery,
|
cpPolyShapeSegmentQuery,
|
||||||
};
|
};
|
||||||
|
|
||||||
int
|
cpBool
|
||||||
cpPolyValidate(cpVect *verts, int numVerts)
|
cpPolyValidate(const cpVect *verts, const int numVerts)
|
||||||
{
|
{
|
||||||
for(int i=0; i<numVerts; i++){
|
for(int i=0; i<numVerts; i++){
|
||||||
cpVect a = verts[i];
|
cpVect a = verts[i];
|
||||||
|
@ -146,10 +145,10 @@ cpPolyValidate(cpVect *verts, int numVerts)
|
||||||
cpVect c = verts[(i+2)%numVerts];
|
cpVect c = verts[(i+2)%numVerts];
|
||||||
|
|
||||||
if(cpvcross(cpvsub(b, a), cpvsub(c, b)) > 0.0f)
|
if(cpvcross(cpvsub(b, a), cpvsub(c, b)) > 0.0f)
|
||||||
return 0;
|
return cpFalse;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return cpTrue;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
|
@ -211,8 +210,8 @@ cpPolyShapeNew(cpBody *body, int numVerts, cpVect *verts, cpVect offset)
|
||||||
cpPolyShape *
|
cpPolyShape *
|
||||||
cpBoxShapeInit(cpPolyShape *poly, cpBody *body, cpFloat width, cpFloat height)
|
cpBoxShapeInit(cpPolyShape *poly, cpBody *body, cpFloat width, cpFloat height)
|
||||||
{
|
{
|
||||||
cpFloat hw = width/2;
|
cpFloat hw = width/2.0f;
|
||||||
cpFloat hh = height/2;
|
cpFloat hh = height/2.0f;
|
||||||
|
|
||||||
cpVect verts[] = {
|
cpVect verts[] = {
|
||||||
cpv(-hw,-hh),
|
cpv(-hw,-hh),
|
||||||
|
|
|
@ -23,7 +23,7 @@
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "chipmunk_unsafe.h"
|
#include "chipmunk_unsafe.h"
|
||||||
|
|
||||||
#define CP_DefineShapeGetter(struct, type, member, name) \
|
#define CP_DefineShapeGetter(struct, type, member, name) \
|
||||||
|
@ -60,8 +60,9 @@ cpShapeInit(cpShape *shape, const cpShapeClass *klass, cpBody *body)
|
||||||
shape->layers = CP_ALL_LAYERS;
|
shape->layers = CP_ALL_LAYERS;
|
||||||
|
|
||||||
shape->data = NULL;
|
shape->data = NULL;
|
||||||
|
shape->next = NULL;
|
||||||
|
|
||||||
cpShapeCacheBB(shape);
|
// cpShapeCacheBB(shape);
|
||||||
|
|
||||||
return shape;
|
return shape;
|
||||||
}
|
}
|
||||||
|
@ -81,6 +82,7 @@ cpShapeFree(cpShape *shape)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO this function should really take a position and rotation explicitly and be renamed
|
||||||
cpBB
|
cpBB
|
||||||
cpShapeCacheBB(cpShape *shape)
|
cpShapeCacheBB(cpShape *shape)
|
||||||
{
|
{
|
||||||
|
@ -90,12 +92,12 @@ cpShapeCacheBB(cpShape *shape)
|
||||||
return shape->bb;
|
return shape->bb;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
cpBool
|
||||||
cpShapePointQuery(cpShape *shape, cpVect p){
|
cpShapePointQuery(cpShape *shape, cpVect p){
|
||||||
return shape->klass->pointQuery(shape, p);
|
return shape->klass->pointQuery(shape, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
cpBool
|
||||||
cpShapeSegmentQuery(cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInfo *info){
|
cpShapeSegmentQuery(cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInfo *info){
|
||||||
cpSegmentQueryInfo blank = {NULL, 0.0f, cpvzero};
|
cpSegmentQueryInfo blank = {NULL, 0.0f, cpvzero};
|
||||||
(*info) = blank;
|
(*info) = blank;
|
||||||
|
@ -138,7 +140,7 @@ cpCircleShapeCacheData(cpShape *shape, cpVect p, cpVect rot)
|
||||||
return bbFromCircle(circle->tc, circle->r);
|
return bbFromCircle(circle->tc, circle->r);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static cpBool
|
||||||
cpCircleShapePointQuery(cpShape *shape, cpVect p){
|
cpCircleShapePointQuery(cpShape *shape, cpVect p){
|
||||||
cpCircleShape *circle = (cpCircleShape *)shape;
|
cpCircleShape *circle = (cpCircleShape *)shape;
|
||||||
return cpvnear(circle->tc, p, circle->r);
|
return cpvnear(circle->tc, p, circle->r);
|
||||||
|
@ -147,7 +149,7 @@ cpCircleShapePointQuery(cpShape *shape, cpVect p){
|
||||||
static void
|
static void
|
||||||
circleSegmentQuery(cpShape *shape, cpVect center, cpFloat r, cpVect a, cpVect b, cpSegmentQueryInfo *info)
|
circleSegmentQuery(cpShape *shape, cpVect center, cpFloat r, cpVect a, cpVect b, cpSegmentQueryInfo *info)
|
||||||
{
|
{
|
||||||
// umm... gross I normally frown upon such things
|
// offset the line to be relative to the circle
|
||||||
a = cpvsub(a, center);
|
a = cpvsub(a, center);
|
||||||
b = cpvsub(b, center);
|
b = cpvsub(b, center);
|
||||||
|
|
||||||
|
@ -239,16 +241,16 @@ cpSegmentShapeCacheData(cpShape *shape, cpVect p, cpVect rot)
|
||||||
return cpBBNew(l - rad, s - rad, r + rad, t + rad);
|
return cpBBNew(l - rad, s - rad, r + rad, t + rad);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static cpBool
|
||||||
cpSegmentShapePointQuery(cpShape *shape, cpVect p){
|
cpSegmentShapePointQuery(cpShape *shape, cpVect p){
|
||||||
if(!cpBBcontainsVect(shape->bb, p)) return 0;
|
if(!cpBBcontainsVect(shape->bb, p)) return cpFalse;
|
||||||
|
|
||||||
cpSegmentShape *seg = (cpSegmentShape *)shape;
|
cpSegmentShape *seg = (cpSegmentShape *)shape;
|
||||||
|
|
||||||
// Calculate normal distance from segment.
|
// Calculate normal distance from segment.
|
||||||
cpFloat dn = cpvdot(seg->tn, p) - cpvdot(seg->ta, seg->tn);
|
cpFloat dn = cpvdot(seg->tn, p) - cpvdot(seg->ta, seg->tn);
|
||||||
cpFloat dist = cpfabs(dn) - seg->r;
|
cpFloat dist = cpfabs(dn) - seg->r;
|
||||||
if(dist > 0.0f) return 0;
|
if(dist > 0.0f) return cpFalse;
|
||||||
|
|
||||||
// Calculate tangential distance along segment.
|
// Calculate tangential distance along segment.
|
||||||
cpFloat dt = -cpvcross(seg->tn, p);
|
cpFloat dt = -cpvcross(seg->tn, p);
|
||||||
|
@ -258,28 +260,32 @@ cpSegmentShapePointQuery(cpShape *shape, cpVect p){
|
||||||
// Decision tree to decide which feature of the segment to collide with.
|
// Decision tree to decide which feature of the segment to collide with.
|
||||||
if(dt <= dtMin){
|
if(dt <= dtMin){
|
||||||
if(dt < (dtMin - seg->r)){
|
if(dt < (dtMin - seg->r)){
|
||||||
return 0;
|
return cpFalse;
|
||||||
} else {
|
} else {
|
||||||
return cpvlengthsq(cpvsub(seg->ta, p)) < (seg->r*seg->r);
|
return cpvlengthsq(cpvsub(seg->ta, p)) < (seg->r*seg->r);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if(dt < dtMax){
|
if(dt < dtMax){
|
||||||
return 1;
|
return cpTrue;
|
||||||
} else {
|
} else {
|
||||||
if(dt < (dtMax + seg->r)) {
|
if(dt < (dtMax + seg->r)) {
|
||||||
return cpvlengthsq(cpvsub(seg->tb, p)) < (seg->r*seg->r);
|
return cpvlengthsq(cpvsub(seg->tb, p)) < (seg->r*seg->r);
|
||||||
} else {
|
} else {
|
||||||
return 0;
|
return cpFalse;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return cpTrue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline cpBool inUnitRange(cpFloat t){return (0.0f < t && t < 1.0f);}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
cpSegmentShapeSegmentQuery(cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInfo *info)
|
cpSegmentShapeSegmentQuery(cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInfo *info)
|
||||||
{
|
{
|
||||||
|
// TODO this function could be optimized better.
|
||||||
|
|
||||||
cpSegmentShape *seg = (cpSegmentShape *)shape;
|
cpSegmentShape *seg = (cpSegmentShape *)shape;
|
||||||
cpVect n = seg->tn;
|
cpVect n = seg->tn;
|
||||||
// flip n if a is behind the axis
|
// flip n if a is behind the axis
|
||||||
|
@ -288,40 +294,37 @@ cpSegmentShapeSegmentQuery(cpShape *shape, cpVect a, cpVect b, cpSegmentQueryInf
|
||||||
|
|
||||||
cpFloat an = cpvdot(a, n);
|
cpFloat an = cpvdot(a, n);
|
||||||
cpFloat bn = cpvdot(b, n);
|
cpFloat bn = cpvdot(b, n);
|
||||||
cpFloat d = cpvdot(seg->ta, n) + seg->r;
|
|
||||||
|
|
||||||
cpFloat t = (d - an)/(bn - an);
|
if(an != bn){
|
||||||
if(0.0f < t && t < 1.0f){
|
cpFloat d = cpvdot(seg->ta, n) + seg->r;
|
||||||
cpVect point = cpvlerp(a, b, t);
|
cpFloat t = (d - an)/(bn - an);
|
||||||
cpFloat dt = -cpvcross(seg->tn, point);
|
|
||||||
cpFloat dtMin = -cpvcross(seg->tn, seg->ta);
|
|
||||||
cpFloat dtMax = -cpvcross(seg->tn, seg->tb);
|
|
||||||
|
|
||||||
if(dtMin < dt && dt < dtMax){
|
if(0.0f < t && t < 1.0f){
|
||||||
info->shape = shape;
|
cpVect point = cpvlerp(a, b, t);
|
||||||
info->t = t;
|
cpFloat dt = -cpvcross(seg->tn, point);
|
||||||
info->n = n;
|
cpFloat dtMin = -cpvcross(seg->tn, seg->ta);
|
||||||
|
cpFloat dtMax = -cpvcross(seg->tn, seg->tb);
|
||||||
|
|
||||||
return; // don't continue on and check endcaps
|
if(dtMin < dt && dt < dtMax){
|
||||||
|
info->shape = shape;
|
||||||
|
info->t = t;
|
||||||
|
info->n = n;
|
||||||
|
|
||||||
|
return; // don't continue on and check endcaps
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(seg->r) {
|
if(seg->r) {
|
||||||
cpSegmentQueryInfo info1; info1.shape = NULL;
|
cpSegmentQueryInfo info1 = {NULL, 1.0f, cpvzero};
|
||||||
cpSegmentQueryInfo info2; info2.shape = NULL;
|
cpSegmentQueryInfo info2 = {NULL, 1.0f, cpvzero};
|
||||||
circleSegmentQuery(shape, seg->ta, seg->r, a, b, &info1);
|
circleSegmentQuery(shape, seg->ta, seg->r, a, b, &info1);
|
||||||
circleSegmentQuery(shape, seg->tb, seg->r, a, b, &info2);
|
circleSegmentQuery(shape, seg->tb, seg->r, a, b, &info2);
|
||||||
|
|
||||||
if(info1.shape && !info2.shape){
|
if(info1.t < info2.t){
|
||||||
(*info) = info1;
|
(*info) = info1;
|
||||||
} else if(info2.shape && !info1.shape){
|
} else {
|
||||||
(*info) = info2;
|
(*info) = info2;
|
||||||
} else if(info1.shape && info2.shape){
|
|
||||||
if(info1.t < info2.t){
|
|
||||||
(*info) = info1;
|
|
||||||
} else {
|
|
||||||
(*info) = info2;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,24 +20,24 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <stdio.h>
|
//#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
|
|
||||||
int cp_contact_persistence = 1;
|
cpTimestamp cp_contact_persistence = 3;
|
||||||
|
|
||||||
#pragma mark Contact Set Helpers
|
#pragma mark Contact Set Helpers
|
||||||
|
|
||||||
// Equal function for contactSet.
|
// Equal function for contactSet.
|
||||||
static int
|
static cpBool
|
||||||
contactSetEql(cpShape **shapes, cpArbiter *arb)
|
contactSetEql(cpShape **shapes, cpArbiter *arb)
|
||||||
{
|
{
|
||||||
cpShape *a = shapes[0];
|
cpShape *a = shapes[0];
|
||||||
cpShape *b = shapes[1];
|
cpShape *b = shapes[1];
|
||||||
|
|
||||||
return ((a == arb->private_a && b == arb->private_b) || (b == arb->private_a && a == arb->private_b));
|
return ((a == arb->a && b == arb->b) || (b == arb->a && a == arb->b));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Transformation function for contactSet.
|
// Transformation function for contactSet.
|
||||||
|
@ -55,13 +55,13 @@ contactSetTrans(cpShape **shapes, cpSpace *space)
|
||||||
for(int i=0; i<count; i++) cpArrayPush(space->pooledArbiters, buffer + i);
|
for(int i=0; i<count; i++) cpArrayPush(space->pooledArbiters, buffer + i);
|
||||||
}
|
}
|
||||||
|
|
||||||
return cpArbiterInit( (cpArbiter *)cpArrayPop(space->pooledArbiters), shapes[0], shapes[1]);
|
return cpArbiterInit((cpArbiter *) cpArrayPop(space->pooledArbiters), shapes[0], shapes[1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pragma mark Collision Pair Function Helpers
|
#pragma mark Collision Pair Function Helpers
|
||||||
|
|
||||||
// Equals function for collFuncSet.
|
// Equals function for collFuncSet.
|
||||||
static int
|
static cpBool
|
||||||
collFuncSetEql(cpCollisionHandler *check, cpCollisionHandler *pair)
|
collFuncSetEql(cpCollisionHandler *check, cpCollisionHandler *pair)
|
||||||
{
|
{
|
||||||
return ((check->a == pair->a && check->b == pair->b) || (check->b == pair->a && check->a == pair->b));
|
return ((check->a == pair->a && check->b == pair->b) || (check->b == pair->a && check->a == pair->b));
|
||||||
|
@ -77,32 +77,10 @@ collFuncSetTrans(cpCollisionHandler *handler, void *unused)
|
||||||
return copy;
|
return copy;
|
||||||
}
|
}
|
||||||
|
|
||||||
#pragma mark Post Step Function Helpers
|
|
||||||
|
|
||||||
typedef struct postStepCallback {
|
|
||||||
cpPostStepFunc func;
|
|
||||||
void *obj;
|
|
||||||
void *data;
|
|
||||||
} postStepCallback;
|
|
||||||
|
|
||||||
static int
|
|
||||||
postStepFuncSetEql(postStepCallback *a, postStepCallback *b){
|
|
||||||
return a->obj == b->obj;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void *
|
|
||||||
postStepFuncSetTrans(postStepCallback *callback, void *ignored)
|
|
||||||
{
|
|
||||||
postStepCallback *value = (postStepCallback *)cpmalloc(sizeof(postStepCallback));
|
|
||||||
(*value) = (*callback);
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
#pragma mark Misc Helper Funcs
|
#pragma mark Misc Helper Funcs
|
||||||
|
|
||||||
// Default collision functions.
|
// Default collision functions.
|
||||||
static int alwaysCollide(cpArbiter *arb, cpSpace *space, void *data){return 1;}
|
static cpBool alwaysCollide(cpArbiter *arb, cpSpace *space, void *data){return 1;}
|
||||||
static void nothing(cpArbiter *arb, cpSpace *space, void *data){}
|
static void nothing(cpArbiter *arb, cpSpace *space, void *data){}
|
||||||
|
|
||||||
// BBfunc callback for the spatial hash.
|
// BBfunc callback for the spatial hash.
|
||||||
|
@ -116,30 +94,6 @@ static void constraintFreeWrap(cpConstraint *ptr, void *unused){cpConstraintFr
|
||||||
|
|
||||||
#pragma mark Memory Management Functions
|
#pragma mark Memory Management Functions
|
||||||
|
|
||||||
#define CP_CONTACTS_BUFFER_SIZE ((CP_BUFFER_BYTES - sizeof(cpContactBufferHeader))/sizeof(cpContact))
|
|
||||||
typedef struct cpContactBuffer {
|
|
||||||
cpContactBufferHeader header;
|
|
||||||
cpContact contacts[CP_CONTACTS_BUFFER_SIZE];
|
|
||||||
} cpContactBuffer;
|
|
||||||
|
|
||||||
static cpContactBufferHeader *
|
|
||||||
cpSpaceAllocContactBuffer(cpSpace *space)
|
|
||||||
{
|
|
||||||
cpContactBuffer *buffer = (cpContactBuffer *)malloc(sizeof(cpContactBuffer));
|
|
||||||
cpArrayPush(space->allocatedBuffers, buffer);
|
|
||||||
return (cpContactBufferHeader *)buffer;
|
|
||||||
}
|
|
||||||
|
|
||||||
static cpContactBufferHeader *
|
|
||||||
cpContactBufferHeaderInit(cpContactBufferHeader *header, cpSpace *space)
|
|
||||||
{
|
|
||||||
header->stamp = space->stamp;
|
|
||||||
header->next = space->contactBuffersTail;
|
|
||||||
header->numContacts = 0;
|
|
||||||
|
|
||||||
return header;
|
|
||||||
}
|
|
||||||
|
|
||||||
cpSpace *
|
cpSpace *
|
||||||
cpSpaceAlloc(void)
|
cpSpaceAlloc(void)
|
||||||
{
|
{
|
||||||
|
@ -151,8 +105,6 @@ cpSpaceAlloc(void)
|
||||||
#define DEFAULT_ITERATIONS 10
|
#define DEFAULT_ITERATIONS 10
|
||||||
#define DEFAULT_ELASTIC_ITERATIONS 0
|
#define DEFAULT_ELASTIC_ITERATIONS 0
|
||||||
|
|
||||||
#define MAX_CONTACTS 10000
|
|
||||||
|
|
||||||
cpCollisionHandler defaultHandler = {0, 0, alwaysCollide, alwaysCollide, nothing, nothing, NULL};
|
cpCollisionHandler defaultHandler = {0, 0, alwaysCollide, alwaysCollide, nothing, nothing, NULL};
|
||||||
|
|
||||||
cpSpace*
|
cpSpace*
|
||||||
|
@ -174,14 +126,16 @@ cpSpaceInit(cpSpace *space)
|
||||||
space->allocatedBuffers = cpArrayNew(0);
|
space->allocatedBuffers = cpArrayNew(0);
|
||||||
|
|
||||||
space->bodies = cpArrayNew(0);
|
space->bodies = cpArrayNew(0);
|
||||||
|
space->sleepingComponents = cpArrayNew(0);
|
||||||
|
space->rousedBodies = cpArrayNew(0);
|
||||||
|
|
||||||
|
space->sleepTimeThreshold = INFINITY;
|
||||||
|
space->idleSpeedThreshold = 0.0f;
|
||||||
|
|
||||||
space->arbiters = cpArrayNew(0);
|
space->arbiters = cpArrayNew(0);
|
||||||
space->pooledArbiters = cpArrayNew(0);
|
space->pooledArbiters = cpArrayNew(0);
|
||||||
|
|
||||||
cpContactBufferHeader *header = cpContactBufferHeaderInit(cpSpaceAllocContactBuffer(space), space);
|
space->contactBuffersHead = NULL;
|
||||||
space->contactBuffersHead = header;
|
|
||||||
space->contactBuffersTail = header;
|
|
||||||
header->next = header; // Buffers will form a ring, start the ring explicitly
|
|
||||||
|
|
||||||
space->contactSet = cpHashSetNew(0, (cpHashSetEqlFunc)contactSetEql, (cpHashSetTransFunc)contactSetTrans);
|
space->contactSet = cpHashSetNew(0, (cpHashSetEqlFunc)contactSetEql, (cpHashSetTransFunc)contactSetTrans);
|
||||||
|
|
||||||
space->constraints = cpArrayNew(0);
|
space->constraints = cpArrayNew(0);
|
||||||
|
@ -190,7 +144,9 @@ cpSpaceInit(cpSpace *space)
|
||||||
space->collFuncSet = cpHashSetNew(0, (cpHashSetEqlFunc)collFuncSetEql, (cpHashSetTransFunc)collFuncSetTrans);
|
space->collFuncSet = cpHashSetNew(0, (cpHashSetEqlFunc)collFuncSetEql, (cpHashSetTransFunc)collFuncSetTrans);
|
||||||
space->collFuncSet->default_value = &space->defaultHandler;
|
space->collFuncSet->default_value = &space->defaultHandler;
|
||||||
|
|
||||||
space->postStepCallbacks = cpHashSetNew(0, (cpHashSetEqlFunc)postStepFuncSetEql, (cpHashSetTransFunc)postStepFuncSetTrans);
|
space->postStepCallbacks = NULL;
|
||||||
|
|
||||||
|
cpBodyInitStatic(&space->staticBody);
|
||||||
|
|
||||||
return space;
|
return space;
|
||||||
}
|
}
|
||||||
|
@ -208,6 +164,8 @@ cpSpaceDestroy(cpSpace *space)
|
||||||
cpSpaceHashFree(space->activeShapes);
|
cpSpaceHashFree(space->activeShapes);
|
||||||
|
|
||||||
cpArrayFree(space->bodies);
|
cpArrayFree(space->bodies);
|
||||||
|
cpArrayFree(space->sleepingComponents);
|
||||||
|
cpArrayFree(space->rousedBodies);
|
||||||
|
|
||||||
cpArrayFree(space->constraints);
|
cpArrayFree(space->constraints);
|
||||||
|
|
||||||
|
@ -244,6 +202,9 @@ cpSpaceFree(cpSpace *space)
|
||||||
void
|
void
|
||||||
cpSpaceFreeChildren(cpSpace *space)
|
cpSpaceFreeChildren(cpSpace *space)
|
||||||
{
|
{
|
||||||
|
cpArray *components = space->sleepingComponents;
|
||||||
|
while(components->num) cpBodyActivate((cpBody *)components->arr[0]);
|
||||||
|
|
||||||
cpSpaceHashEach(space->staticShapes, (cpSpaceHashIterator)&shapeFreeWrap, NULL);
|
cpSpaceHashEach(space->staticShapes, (cpSpaceHashIterator)&shapeFreeWrap, NULL);
|
||||||
cpSpaceHashEach(space->activeShapes, (cpSpaceHashIterator)&shapeFreeWrap, NULL);
|
cpSpaceHashEach(space->activeShapes, (cpSpaceHashIterator)&shapeFreeWrap, NULL);
|
||||||
cpArrayEach(space->bodies, (cpArrayIter)&bodyFreeWrap, NULL);
|
cpArrayEach(space->bodies, (cpArrayIter)&bodyFreeWrap, NULL);
|
||||||
|
@ -310,31 +271,62 @@ cpSpaceSetDefaultCollisionHandler(
|
||||||
|
|
||||||
#define cpAssertSpaceUnlocked(space) \
|
#define cpAssertSpaceUnlocked(space) \
|
||||||
cpAssert(!space->locked, \
|
cpAssert(!space->locked, \
|
||||||
"This addition/removal cannot be done safely during a call to cpSpaceStep(). " \
|
"This addition/removal cannot be done safely during a call to cpSpaceStep() or during a query. " \
|
||||||
"Put these calls into a Post Step Callback." \
|
"Put these calls into a post-step callback." \
|
||||||
);
|
);
|
||||||
|
|
||||||
|
static void
|
||||||
|
cpBodyAddShape(cpBody *body, cpShape *shape)
|
||||||
|
{
|
||||||
|
shape->next = shape->body->shapesList;
|
||||||
|
shape->body->shapesList = shape;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
cpBodyRemoveShape(cpBody *body, cpShape *shape)
|
||||||
|
{
|
||||||
|
cpShape **prev_ptr = &body->shapesList;
|
||||||
|
cpShape *node = body->shapesList;
|
||||||
|
|
||||||
|
while(node && node != shape){
|
||||||
|
prev_ptr = &node->next;
|
||||||
|
node = node->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
cpAssert(node, "Attempted to remove a shape from a body it was never attached to.");
|
||||||
|
(*prev_ptr) = node->next;
|
||||||
|
}
|
||||||
|
|
||||||
cpShape *
|
cpShape *
|
||||||
cpSpaceAddShape(cpSpace *space, cpShape *shape)
|
cpSpaceAddShape(cpSpace *space, cpShape *shape)
|
||||||
{
|
{
|
||||||
cpAssert(shape->body, "Cannot add a shape with a NULL body.");
|
cpBody *body = shape->body;
|
||||||
|
if(!body || cpBodyIsStatic(body)) return cpSpaceAddStaticShape(space, shape);
|
||||||
|
|
||||||
cpAssert(!cpHashSetFind(space->activeShapes->handleSet, shape->hashid, shape),
|
cpAssert(!cpHashSetFind(space->activeShapes->handleSet, shape->hashid, shape),
|
||||||
"Cannot add the same shape more than once.");
|
"Cannot add the same shape more than once.");
|
||||||
cpAssertSpaceUnlocked(space);
|
cpAssertSpaceUnlocked(space);
|
||||||
|
|
||||||
|
cpBodyActivate(body);
|
||||||
|
cpBodyAddShape(body, shape);
|
||||||
|
|
||||||
|
cpShapeCacheBB(shape);
|
||||||
cpSpaceHashInsert(space->activeShapes, shape, shape->hashid, shape->bb);
|
cpSpaceHashInsert(space->activeShapes, shape, shape->hashid, shape->bb);
|
||||||
|
|
||||||
return shape;
|
return shape;
|
||||||
}
|
}
|
||||||
|
|
||||||
cpShape *
|
cpShape *
|
||||||
cpSpaceAddStaticShape(cpSpace *space, cpShape *shape)
|
cpSpaceAddStaticShape(cpSpace *space, cpShape *shape)
|
||||||
{
|
{
|
||||||
cpAssert(shape->body, "Cannot add a static shape with a NULL body.");
|
|
||||||
cpAssert(!cpHashSetFind(space->staticShapes->handleSet, shape->hashid, shape),
|
cpAssert(!cpHashSetFind(space->staticShapes->handleSet, shape->hashid, shape),
|
||||||
"Cannot add the same static shape more than once.");
|
"Cannot add the same static shape more than once.");
|
||||||
cpAssertSpaceUnlocked(space);
|
cpAssertSpaceUnlocked(space);
|
||||||
|
|
||||||
|
if(!shape->body) shape->body = &space->staticBody;
|
||||||
|
|
||||||
cpShapeCacheBB(shape);
|
cpShapeCacheBB(shape);
|
||||||
|
cpSpaceActivateShapesTouchingShape(space, shape);
|
||||||
cpSpaceHashInsert(space->staticShapes, shape, shape->hashid, shape->bb);
|
cpSpaceHashInsert(space->staticShapes, shape, shape->hashid, shape->bb);
|
||||||
|
|
||||||
return shape;
|
return shape;
|
||||||
|
@ -343,10 +335,12 @@ cpSpaceAddStaticShape(cpSpace *space, cpShape *shape)
|
||||||
cpBody *
|
cpBody *
|
||||||
cpSpaceAddBody(cpSpace *space, cpBody *body)
|
cpSpaceAddBody(cpSpace *space, cpBody *body)
|
||||||
{
|
{
|
||||||
cpAssert(!cpArrayContains(space->bodies, body), "Cannot add the same body more than once.");
|
cpAssertWarn(!cpBodyIsStatic(body), "Static bodies cannot be added to a space as they are not meant to be simulated.");
|
||||||
|
cpAssert(!body->space, "Cannot add a body to a more than one space or to the same space twice.");
|
||||||
// cpAssertSpaceUnlocked(space); This should be safe as long as it's not from an integration callback
|
// cpAssertSpaceUnlocked(space); This should be safe as long as it's not from an integration callback
|
||||||
|
|
||||||
cpArrayPush(space->bodies, body);
|
cpArrayPush(space->bodies, body);
|
||||||
|
body->space = space;
|
||||||
|
|
||||||
return body;
|
return body;
|
||||||
}
|
}
|
||||||
|
@ -357,6 +351,11 @@ cpSpaceAddConstraint(cpSpace *space, cpConstraint *constraint)
|
||||||
cpAssert(!cpArrayContains(space->constraints, constraint), "Cannot add the same constraint more than once.");
|
cpAssert(!cpArrayContains(space->constraints, constraint), "Cannot add the same constraint more than once.");
|
||||||
// cpAssertSpaceUnlocked(space); This should be safe as long as its not from a constraint callback.
|
// cpAssertSpaceUnlocked(space); This should be safe as long as its not from a constraint callback.
|
||||||
|
|
||||||
|
if(!constraint->a) constraint->a = &space->staticBody;
|
||||||
|
if(!constraint->b) constraint->b = &space->staticBody;
|
||||||
|
|
||||||
|
cpBodyActivate(constraint->a);
|
||||||
|
cpBodyActivate(constraint->b);
|
||||||
cpArrayPush(space->constraints, constraint);
|
cpArrayPush(space->constraints, constraint);
|
||||||
|
|
||||||
return constraint;
|
return constraint;
|
||||||
|
@ -368,24 +367,37 @@ typedef struct removalContext {
|
||||||
} removalContext;
|
} removalContext;
|
||||||
|
|
||||||
// Hashset filter func to throw away old arbiters.
|
// Hashset filter func to throw away old arbiters.
|
||||||
static int
|
static cpBool
|
||||||
contactSetFilterRemovedShape(cpArbiter *arb, removalContext *context)
|
contactSetFilterRemovedShape(cpArbiter *arb, removalContext *context)
|
||||||
{
|
{
|
||||||
if(context->shape == arb->private_a || context->shape == arb->private_b){
|
if(context->shape == arb->a || context->shape == arb->b){
|
||||||
arb->handler->separate(arb, context->space, arb->handler->data);
|
if(arb->state != cpArbiterStateCached){
|
||||||
|
arb->handler->separate(arb, context->space, arb->handler->data);
|
||||||
|
}
|
||||||
|
|
||||||
cpArrayPush(context->space->pooledArbiters, arb);
|
cpArrayPush(context->space->pooledArbiters, arb);
|
||||||
return 0;
|
return cpFalse;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return cpTrue;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
cpSpaceRemoveShape(cpSpace *space, cpShape *shape)
|
cpSpaceRemoveShape(cpSpace *space, cpShape *shape)
|
||||||
{
|
{
|
||||||
cpAssertWarn(cpHashSetFind(space->activeShapes->handleSet, shape->hashid, shape),
|
cpBody *body = shape->body;
|
||||||
"Cannot remove a shape that was never added to the space. (Removed twice maybe?)");
|
if(cpBodyIsStatic(body)){
|
||||||
|
cpSpaceRemoveStaticShape(space, shape);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
cpBodyActivate(body);
|
||||||
|
|
||||||
cpAssertSpaceUnlocked(space);
|
cpAssertSpaceUnlocked(space);
|
||||||
|
cpAssertWarn(cpHashSetFind(space->activeShapes->handleSet, shape->hashid, shape),
|
||||||
|
"Cannot remove a shape that was not added to the space. (Removed twice maybe?)");
|
||||||
|
|
||||||
|
cpBodyRemoveShape(body, shape);
|
||||||
|
|
||||||
removalContext context = {space, shape};
|
removalContext context = {space, shape};
|
||||||
cpHashSetFilter(space->contactSet, (cpHashSetFilterFunc)contactSetFilterRemovedShape, &context);
|
cpHashSetFilter(space->contactSet, (cpHashSetFilterFunc)contactSetFilterRemovedShape, &context);
|
||||||
|
@ -396,238 +408,43 @@ void
|
||||||
cpSpaceRemoveStaticShape(cpSpace *space, cpShape *shape)
|
cpSpaceRemoveStaticShape(cpSpace *space, cpShape *shape)
|
||||||
{
|
{
|
||||||
cpAssertWarn(cpHashSetFind(space->staticShapes->handleSet, shape->hashid, shape),
|
cpAssertWarn(cpHashSetFind(space->staticShapes->handleSet, shape->hashid, shape),
|
||||||
"Cannot remove a static shape that was never added to the space. (Removed twice maybe?)");
|
"Cannot remove a static or sleeping shape that was not added to the space. (Removed twice maybe?)");
|
||||||
cpAssertSpaceUnlocked(space);
|
cpAssertSpaceUnlocked(space);
|
||||||
|
|
||||||
removalContext context = {space, shape};
|
removalContext context = {space, shape};
|
||||||
cpHashSetFilter(space->contactSet, (cpHashSetFilterFunc)contactSetFilterRemovedShape, &context);
|
cpHashSetFilter(space->contactSet, (cpHashSetFilterFunc)contactSetFilterRemovedShape, &context);
|
||||||
cpSpaceHashRemove(space->staticShapes, shape, shape->hashid);
|
cpSpaceHashRemove(space->staticShapes, shape, shape->hashid);
|
||||||
|
|
||||||
|
cpSpaceActivateShapesTouchingShape(space, shape);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
cpSpaceRemoveBody(cpSpace *space, cpBody *body)
|
cpSpaceRemoveBody(cpSpace *space, cpBody *body)
|
||||||
{
|
{
|
||||||
cpAssertWarn(cpArrayContains(space->bodies, body),
|
cpAssertWarn(body->space == space,
|
||||||
"Cannot remove a body that was never added to the space. (Removed twice maybe?)");
|
"Cannot remove a body that was not added to the space. (Removed twice maybe?)");
|
||||||
cpAssertSpaceUnlocked(space);
|
cpAssertSpaceUnlocked(space);
|
||||||
|
|
||||||
|
cpBodyActivate(body);
|
||||||
cpArrayDeleteObj(space->bodies, body);
|
cpArrayDeleteObj(space->bodies, body);
|
||||||
|
body->space = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
cpSpaceRemoveConstraint(cpSpace *space, cpConstraint *constraint)
|
cpSpaceRemoveConstraint(cpSpace *space, cpConstraint *constraint)
|
||||||
{
|
{
|
||||||
cpAssertWarn(cpArrayContains(space->constraints, constraint),
|
cpAssertWarn(cpArrayContains(space->constraints, constraint),
|
||||||
"Cannot remove a constraint that was never added to the space. (Removed twice maybe?)");
|
"Cannot remove a constraint that was not added to the space. (Removed twice maybe?)");
|
||||||
// cpAssertSpaceUnlocked(space); Should be safe as long as its not from a constraint callback.
|
// cpAssertSpaceUnlocked(space); Should be safe as long as its not from a constraint callback.
|
||||||
|
|
||||||
|
cpBodyActivate(constraint->a);
|
||||||
|
cpBodyActivate(constraint->b);
|
||||||
cpArrayDeleteObj(space->constraints, constraint);
|
cpArrayDeleteObj(space->constraints, constraint);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pragma mark Post Step Functions
|
|
||||||
|
|
||||||
void
|
|
||||||
cpSpaceAddPostStepCallback(cpSpace *space, cpPostStepFunc func, void *obj, void *data)
|
|
||||||
{
|
|
||||||
postStepCallback callback = {func, obj, data};
|
|
||||||
cpHashSetInsert(space->postStepCallbacks, (cpHashValue)(size_t)obj, &callback, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
removeAndFreeShapeAndBody(cpShape *shape, cpSpace *space)
|
|
||||||
{
|
|
||||||
cpSpaceRemoveShape(space, shape);
|
|
||||||
cpShapeFree(shape);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
cpSpacePostStepRemoveAndFreeShapeAndBody(cpSpace *space, cpShape *shape)
|
|
||||||
{
|
|
||||||
cpSpaceAddPostStepCallback(space, (cpPostStepFunc)removeAndFreeShapeAndBody, shape, space);
|
|
||||||
}
|
|
||||||
|
|
||||||
#pragma mark Point Query Functions
|
|
||||||
|
|
||||||
typedef struct pointQueryContext {
|
|
||||||
cpLayers layers;
|
|
||||||
cpGroup group;
|
|
||||||
cpSpacePointQueryFunc func;
|
|
||||||
void *data;
|
|
||||||
} pointQueryContext;
|
|
||||||
|
|
||||||
static void
|
|
||||||
pointQueryHelper(cpVect *point, cpShape *shape, pointQueryContext *context)
|
|
||||||
{
|
|
||||||
if(
|
|
||||||
!(shape->group && context->group == shape->group) && (context->layers&shape->layers) &&
|
|
||||||
cpShapePointQuery(shape, *point)
|
|
||||||
){
|
|
||||||
context->func(shape, context->data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
cpSpacePointQuery(cpSpace *space, cpVect point, cpLayers layers, cpGroup group, cpSpacePointQueryFunc func, void *data)
|
|
||||||
{
|
|
||||||
pointQueryContext context = {layers, group, func, data};
|
|
||||||
cpSpaceHashPointQuery(space->activeShapes, point, (cpSpaceHashQueryFunc)pointQueryHelper, &context);
|
|
||||||
cpSpaceHashPointQuery(space->staticShapes, point, (cpSpaceHashQueryFunc)pointQueryHelper, &context);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
rememberLastPointQuery(cpShape *shape, cpShape **outShape)
|
|
||||||
{
|
|
||||||
(*outShape) = shape;
|
|
||||||
}
|
|
||||||
|
|
||||||
cpShape *
|
|
||||||
cpSpacePointQueryFirst(cpSpace *space, cpVect point, cpLayers layers, cpGroup group)
|
|
||||||
{
|
|
||||||
cpShape *shape = NULL;
|
|
||||||
cpSpacePointQuery(space, point, layers, group, (cpSpacePointQueryFunc)rememberLastPointQuery, &shape);
|
|
||||||
|
|
||||||
return shape;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
cpSpaceEachBody(cpSpace *space, cpSpaceBodyIterator func, void *data)
|
|
||||||
{
|
|
||||||
cpArray *bodies = space->bodies;
|
|
||||||
|
|
||||||
for(int i=0; i<bodies->num; i++)
|
|
||||||
func((cpBody *)bodies->arr[i], data);
|
|
||||||
}
|
|
||||||
|
|
||||||
#pragma mark Segment Query Functions
|
|
||||||
|
|
||||||
typedef struct segQueryContext {
|
|
||||||
cpVect start, end;
|
|
||||||
cpLayers layers;
|
|
||||||
cpGroup group;
|
|
||||||
cpSpaceSegmentQueryFunc func;
|
|
||||||
int anyCollision;
|
|
||||||
} segQueryContext;
|
|
||||||
|
|
||||||
static cpFloat
|
|
||||||
segQueryFunc(segQueryContext *context, cpShape *shape, void *data)
|
|
||||||
{
|
|
||||||
cpSegmentQueryInfo info;
|
|
||||||
|
|
||||||
if(
|
|
||||||
!(shape->group && context->group == shape->group) && (context->layers&shape->layers) &&
|
|
||||||
cpShapeSegmentQuery(shape, context->start, context->end, &info)
|
|
||||||
){
|
|
||||||
if(context->func){
|
|
||||||
context->func(shape, info.t, info.n, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
context->anyCollision = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1.0f;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
cpSpaceSegmentQuery(cpSpace *space, cpVect start, cpVect end, cpLayers layers, cpGroup group, cpSpaceSegmentQueryFunc func, void *data)
|
|
||||||
{
|
|
||||||
segQueryContext context = {
|
|
||||||
start, end,
|
|
||||||
layers, group,
|
|
||||||
func,
|
|
||||||
0,
|
|
||||||
};
|
|
||||||
|
|
||||||
cpSpaceHashSegmentQuery(space->staticShapes, &context, start, end, 1.0f, (cpSpaceHashSegmentQueryFunc)segQueryFunc, data);
|
|
||||||
cpSpaceHashSegmentQuery(space->activeShapes, &context, start, end, 1.0f, (cpSpaceHashSegmentQueryFunc)segQueryFunc, data);
|
|
||||||
|
|
||||||
return context.anyCollision;
|
|
||||||
}
|
|
||||||
|
|
||||||
typedef struct segQueryFirstContext {
|
|
||||||
cpVect start, end;
|
|
||||||
cpLayers layers;
|
|
||||||
cpGroup group;
|
|
||||||
} segQueryFirstContext;
|
|
||||||
|
|
||||||
static cpFloat
|
|
||||||
segQueryFirst(segQueryFirstContext *context, cpShape *shape, cpSegmentQueryInfo *out)
|
|
||||||
{
|
|
||||||
cpSegmentQueryInfo info;// = {NULL, 1.0f, cpvzero};
|
|
||||||
if(
|
|
||||||
!(shape->group && context->group == shape->group) && (context->layers&shape->layers) &&
|
|
||||||
cpShapeSegmentQuery(shape, context->start, context->end, &info)
|
|
||||||
){
|
|
||||||
if(info.t < out->t){
|
|
||||||
out->shape = info.shape;
|
|
||||||
out->t = info.t;
|
|
||||||
out->n = info.n;
|
|
||||||
}
|
|
||||||
|
|
||||||
return info.t;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1.0f;
|
|
||||||
}
|
|
||||||
|
|
||||||
cpShape *
|
|
||||||
cpSpaceSegmentQueryFirst(cpSpace *space, cpVect start, cpVect end, cpLayers layers, cpGroup group, cpSegmentQueryInfo *out)
|
|
||||||
{
|
|
||||||
cpSegmentQueryInfo info = {NULL, 1.0f, cpvzero};
|
|
||||||
if(out){
|
|
||||||
(*out) = info;
|
|
||||||
} else {
|
|
||||||
out = &info;
|
|
||||||
}
|
|
||||||
|
|
||||||
out->t = 1.0f;
|
|
||||||
|
|
||||||
segQueryFirstContext context = {
|
|
||||||
start, end,
|
|
||||||
layers, group
|
|
||||||
};
|
|
||||||
|
|
||||||
cpSpaceHashSegmentQuery(space->staticShapes, &context, start, end, 1.0f, (cpSpaceHashSegmentQueryFunc)segQueryFirst, out);
|
|
||||||
cpSpaceHashSegmentQuery(space->activeShapes, &context, start, end, out->t, (cpSpaceHashSegmentQueryFunc)segQueryFirst, out);
|
|
||||||
|
|
||||||
return out->shape;
|
|
||||||
}
|
|
||||||
|
|
||||||
#pragma mark BB Query functions
|
|
||||||
|
|
||||||
typedef struct bbQueryContext {
|
|
||||||
cpLayers layers;
|
|
||||||
cpGroup group;
|
|
||||||
cpSpaceBBQueryFunc func;
|
|
||||||
void *data;
|
|
||||||
} bbQueryContext;
|
|
||||||
|
|
||||||
static void
|
|
||||||
bbQueryHelper(cpBB *bb, cpShape *shape, bbQueryContext *context)
|
|
||||||
{
|
|
||||||
if(
|
|
||||||
!(shape->group && context->group == shape->group) && (context->layers&shape->layers) &&
|
|
||||||
cpBBintersects(*bb, shape->bb)
|
|
||||||
){
|
|
||||||
context->func(shape, context->data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
cpSpaceBBQuery(cpSpace *space, cpBB bb, cpLayers layers, cpGroup group, cpSpaceBBQueryFunc func, void *data)
|
|
||||||
{
|
|
||||||
bbQueryContext context = {layers, group, func, data};
|
|
||||||
cpSpaceHashQuery(space->activeShapes, &bb, bb, (cpSpaceHashQueryFunc)bbQueryHelper, &context);
|
|
||||||
cpSpaceHashQuery(space->staticShapes, &bb, bb, (cpSpaceHashQueryFunc)bbQueryHelper, &context);
|
|
||||||
}
|
|
||||||
|
|
||||||
#pragma mark Spatial Hash Management
|
#pragma mark Spatial Hash Management
|
||||||
|
|
||||||
// Iterator function used for updating shape BBoxes.
|
static void updateBBCache(cpShape *shape, void *unused){cpShapeCacheBB(shape);}
|
||||||
static void
|
|
||||||
updateBBCache(cpShape *shape, void *unused)
|
|
||||||
{
|
|
||||||
cpShapeCacheBB(shape);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
cpSpaceResizeStaticHash(cpSpace *space, cpFloat dim, int count)
|
cpSpaceResizeStaticHash(cpSpace *space, cpFloat dim, int count)
|
||||||
|
@ -649,251 +466,34 @@ cpSpaceRehashStatic(cpSpace *space)
|
||||||
cpSpaceHashRehash(space->staticShapes);
|
cpSpaceHashRehash(space->staticShapes);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pragma mark Collision Detection Functions
|
void
|
||||||
|
cpSpaceRehashShape(cpSpace *space, cpShape *shape)
|
||||||
static cpContactBufferHeader *
|
|
||||||
cpSpaceGetFreeContactBuffer(cpSpace *space)
|
|
||||||
{
|
{
|
||||||
if(space->stamp - space->contactBuffersTail->stamp > cp_contact_persistence){
|
cpShapeCacheBB(shape);
|
||||||
cpContactBufferHeader *header = space->contactBuffersTail;
|
|
||||||
space->contactBuffersTail = header->next;
|
// attempt to rehash the shape in both hashes
|
||||||
|
cpSpaceHashRehashObject(space->activeShapes, shape, shape->hashid);
|
||||||
return cpContactBufferHeaderInit(header, space);
|
cpSpaceHashRehashObject(space->staticShapes, shape, shape->hashid);
|
||||||
} else {
|
|
||||||
cpContactBufferHeader *header = cpSpaceAllocContactBuffer(space);
|
|
||||||
return cpContactBufferHeaderInit(header, space);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
cpSpacePushNewContactBuffer(cpSpace *space)
|
|
||||||
{
|
|
||||||
// for(cpContactBuffer *buffer = space->contactBuffersTail; buffer != space->contactBuffersHead; buffer = buffer->next){
|
|
||||||
// printf("%p -> ", buffer);
|
|
||||||
// }
|
|
||||||
// printf("%p (head)\n", space->contactBuffersHead);
|
|
||||||
|
|
||||||
cpContactBufferHeader *buffer = cpSpaceGetFreeContactBuffer(space);
|
|
||||||
space->contactBuffersHead->next = buffer;
|
|
||||||
space->contactBuffersHead = buffer;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline int
|
|
||||||
queryReject(cpShape *a, cpShape *b)
|
|
||||||
{
|
|
||||||
return
|
|
||||||
// BBoxes must overlap
|
|
||||||
!cpBBintersects(a->bb, b->bb)
|
|
||||||
// Don't collide shapes attached to the same body.
|
|
||||||
|| a->body == b->body
|
|
||||||
// Don't collide objects in the same non-zero group
|
|
||||||
|| (a->group && b->group && a->group == b->group)
|
|
||||||
// Don't collide objects that don't share at least on layer.
|
|
||||||
|| !(a->layers & b->layers);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Callback from the spatial hash.
|
|
||||||
static void
|
|
||||||
queryFunc(cpShape *a, cpShape *b, cpSpace *space)
|
|
||||||
{
|
|
||||||
// Reject any of the simple cases
|
|
||||||
if(queryReject(a,b)) return;
|
|
||||||
|
|
||||||
// Find the collision pair function for the shapes.
|
|
||||||
struct{cpCollisionType a, b;} ids = {a->collision_type, b->collision_type};
|
|
||||||
cpHashValue collHashID = CP_HASH_PAIR(a->collision_type, b->collision_type);
|
|
||||||
cpCollisionHandler *handler = (cpCollisionHandler *)cpHashSetFind(space->collFuncSet, collHashID, &ids);
|
|
||||||
|
|
||||||
int sensor = a->sensor || b->sensor;
|
|
||||||
if(sensor && handler == &space->defaultHandler) return;
|
|
||||||
|
|
||||||
// Shape 'a' should have the lower shape type. (required by cpCollideShapes() )
|
|
||||||
if(a->klass->type > b->klass->type){
|
|
||||||
cpShape *temp = a;
|
|
||||||
a = b;
|
|
||||||
b = temp;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(space->contactBuffersHead->numContacts + CP_MAX_CONTACTS_PER_ARBITER > CP_CONTACTS_BUFFER_SIZE){
|
|
||||||
// contact buffer could overflow on the next collision, push a fresh one.
|
|
||||||
cpSpacePushNewContactBuffer(space);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Narrow-phase collision detection.
|
|
||||||
cpContact *contacts = ((cpContactBuffer *)(space->contactBuffersHead))->contacts + space->contactBuffersHead->numContacts;
|
|
||||||
int numContacts = cpCollideShapes(a, b, contacts);
|
|
||||||
if(!numContacts) return; // Shapes are not colliding.
|
|
||||||
space->contactBuffersHead->numContacts += numContacts;
|
|
||||||
|
|
||||||
// Get an arbiter from space->contactSet for the two shapes.
|
|
||||||
// This is where the persistant contact magic comes from.
|
|
||||||
cpShape *shape_pair[] = {a, b};
|
|
||||||
cpHashValue arbHashID = CP_HASH_PAIR((size_t)a, (size_t)b);
|
|
||||||
cpArbiter *arb = (cpArbiter *)cpHashSetInsert(space->contactSet, arbHashID, shape_pair, space);
|
|
||||||
cpArbiterUpdate(arb, contacts, numContacts, handler, a, b); // retains the contacts array
|
|
||||||
|
|
||||||
// Call the begin function first if it's the first step
|
|
||||||
if(arb->stamp == -1 && !handler->begin(arb, space, handler->data)){
|
|
||||||
cpArbiterIgnore(arb); // permanently ignore the collision until separation
|
|
||||||
}
|
|
||||||
|
|
||||||
if(
|
|
||||||
// Ignore the arbiter if it has been flagged
|
|
||||||
(arb->state != cpArbiterStateIgnore) &&
|
|
||||||
// Call preSolve
|
|
||||||
handler->preSolve(arb, space, handler->data) &&
|
|
||||||
// Process, but don't add collisions for sensors.
|
|
||||||
!sensor
|
|
||||||
){
|
|
||||||
cpArrayPush(space->arbiters, arb);
|
|
||||||
} else {
|
|
||||||
// cpfree(arb->contacts);
|
|
||||||
space->contactBuffersHead->numContacts -= numContacts;
|
|
||||||
arb->contacts = NULL;
|
|
||||||
arb->numContacts = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Time stamp the arbiter so we know it was used recently.
|
|
||||||
arb->stamp = space->stamp;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Iterator for active/static hash collisions.
|
|
||||||
static void
|
|
||||||
active2staticIter(cpShape *shape, cpSpace *space)
|
|
||||||
{
|
|
||||||
cpSpaceHashQuery(space->staticShapes, shape, shape->bb, (cpSpaceHashQueryFunc)queryFunc, space);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Hashset filter func to throw away old arbiters.
|
|
||||||
static int
|
|
||||||
contactSetFilter(cpArbiter *arb, cpSpace *space)
|
|
||||||
{
|
|
||||||
int ticks = space->stamp - arb->stamp;
|
|
||||||
|
|
||||||
// was used last frame, but not this one
|
|
||||||
if(ticks == 1){
|
|
||||||
arb->handler->separate(arb, space, arb->handler->data);
|
|
||||||
arb->stamp = -1; // mark it as a new pair again.
|
|
||||||
}
|
|
||||||
|
|
||||||
if(ticks >= cp_contact_persistence){
|
|
||||||
cpArrayPush(space->pooledArbiters, arb);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Hashset filter func to call and throw away post step callbacks.
|
|
||||||
static int
|
|
||||||
postStepCallbackSetFilter(postStepCallback *callback, cpSpace *space)
|
|
||||||
{
|
|
||||||
callback->func(space, callback->obj, callback->data);
|
|
||||||
cpfree(callback);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
#pragma mark All Important cpSpaceStep() Function
|
|
||||||
|
|
||||||
void
|
void
|
||||||
cpSpaceStep(cpSpace *space, cpFloat dt)
|
cpSpaceEachBody(cpSpace *space, cpSpaceBodyIterator func, void *data)
|
||||||
{
|
{
|
||||||
if(!dt) return; // don't step if the timestep is 0!
|
|
||||||
|
|
||||||
cpFloat dt_inv = 1.0f/dt;
|
|
||||||
|
|
||||||
cpArray *bodies = space->bodies;
|
cpArray *bodies = space->bodies;
|
||||||
cpArray *constraints = space->constraints;
|
|
||||||
|
|
||||||
space->locked = 1;
|
|
||||||
|
|
||||||
// Empty the arbiter list.
|
|
||||||
space->arbiters->num = 0;
|
|
||||||
|
|
||||||
// Integrate positions.
|
|
||||||
for(int i=0; i<bodies->num; i++){
|
for(int i=0; i<bodies->num; i++){
|
||||||
cpBody *body = (cpBody *)bodies->arr[i];
|
func((cpBody *)bodies->arr[i], data);
|
||||||
body->position_func(body, dt);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pre-cache BBoxes and shape data.
|
cpArray *components = space->sleepingComponents;
|
||||||
cpSpaceHashEach(space->activeShapes, (cpSpaceHashIterator)updateBBCache, NULL);
|
for(int i=0; i<components->num; i++){
|
||||||
|
cpBody *root = (cpBody *)components->arr[i];
|
||||||
// Collide!
|
cpBody *body = root, *next;
|
||||||
cpSpacePushNewContactBuffer(space);
|
do {
|
||||||
cpSpaceHashEach(space->activeShapes, (cpSpaceHashIterator)active2staticIter, space);
|
next = body->node.next;
|
||||||
cpSpaceHashQueryRehash(space->activeShapes, (cpSpaceHashQueryFunc)queryFunc, space);
|
func(body, data);
|
||||||
|
} while((body = next) != root);
|
||||||
// Clear out old cached arbiters and dispatch untouch functions
|
|
||||||
cpHashSetFilter(space->contactSet, (cpHashSetFilterFunc)contactSetFilter, space);
|
|
||||||
|
|
||||||
// Prestep the arbiters.
|
|
||||||
cpArray *arbiters = space->arbiters;
|
|
||||||
for(int i=0; i<arbiters->num; i++)
|
|
||||||
cpArbiterPreStep((cpArbiter *)arbiters->arr[i], dt_inv);
|
|
||||||
|
|
||||||
// Prestep the constraints.
|
|
||||||
for(int i=0; i<constraints->num; i++){
|
|
||||||
cpConstraint *constraint = (cpConstraint *)constraints->arr[i];
|
|
||||||
constraint->klass->preStep(constraint, dt, dt_inv);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for(int i=0; i<space->elasticIterations; i++){
|
|
||||||
for(int j=0; j<arbiters->num; j++)
|
|
||||||
cpArbiterApplyImpulse((cpArbiter *)arbiters->arr[j], 1.0f);
|
|
||||||
|
|
||||||
for(int j=0; j<constraints->num; j++){
|
|
||||||
cpConstraint *constraint = (cpConstraint *)constraints->arr[j];
|
|
||||||
constraint->klass->applyImpulse(constraint);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Integrate velocities.
|
|
||||||
cpFloat damping = cpfpow(1.0f/space->damping, -dt);
|
|
||||||
for(int i=0; i<bodies->num; i++){
|
|
||||||
cpBody *body = (cpBody *)bodies->arr[i];
|
|
||||||
body->velocity_func(body, space->gravity, damping, dt);
|
|
||||||
}
|
|
||||||
|
|
||||||
for(int i=0; i<arbiters->num; i++)
|
|
||||||
cpArbiterApplyCachedImpulse((cpArbiter *)arbiters->arr[i]);
|
|
||||||
|
|
||||||
// run the old-style elastic solver if elastic iterations are disabled
|
|
||||||
cpFloat elasticCoef = (space->elasticIterations ? 0.0f : 1.0f);
|
|
||||||
|
|
||||||
// Run the impulse solver.
|
|
||||||
for(int i=0; i<space->iterations; i++){
|
|
||||||
for(int j=0; j<arbiters->num; j++)
|
|
||||||
cpArbiterApplyImpulse((cpArbiter *)arbiters->arr[j], elasticCoef);
|
|
||||||
|
|
||||||
for(int j=0; j<constraints->num; j++){
|
|
||||||
cpConstraint *constraint = (cpConstraint *)constraints->arr[j];
|
|
||||||
constraint->klass->applyImpulse(constraint);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
space->locked = 0;
|
|
||||||
|
|
||||||
// run the post solve callbacks
|
|
||||||
for(int i=0; i<arbiters->num; i++){
|
|
||||||
cpArbiter *arb = (cpArbiter *) arbiters->arr[i];
|
|
||||||
|
|
||||||
cpCollisionHandler *handler = arb->handler;
|
|
||||||
handler->postSolve(arb, space, handler->data);
|
|
||||||
|
|
||||||
arb->state = cpArbiterStateNormal;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run the post step callbacks
|
|
||||||
// Use filter as an easy way to clear out the queue as it runs
|
|
||||||
cpHashSetFilter(space->postStepCallbacks, (cpHashSetFilterFunc)postStepCallbackSetFilter, space);
|
|
||||||
|
|
||||||
// cpFloat dvsq = cpvdot(space->gravity, space->gravity);
|
|
||||||
// dvsq *= dt*dt * space->damping*space->damping;
|
|
||||||
// for(int i=0; i<bodies->num; i++)
|
|
||||||
// cpBodyMarkLowEnergy(bodies->arr[i], dvsq, space->sleepTicks);
|
|
||||||
|
|
||||||
// Increment the stamp.
|
|
||||||
space->stamp++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -21,9 +21,8 @@
|
||||||
|
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
#include "chipmunk.h"
|
#include "chipmunk_private.h"
|
||||||
#include "prime.h"
|
#include "prime.h"
|
||||||
|
|
||||||
static cpHandle*
|
static cpHandle*
|
||||||
|
@ -36,11 +35,7 @@ cpHandleInit(cpHandle *hand, void *obj)
|
||||||
return hand;
|
return hand;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void
|
static inline void cpHandleRetain(cpHandle *hand){hand->retain++;}
|
||||||
cpHandleRetain(cpHandle *hand)
|
|
||||||
{
|
|
||||||
hand->retain++;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void
|
static inline void
|
||||||
cpHandleRelease(cpHandle *hand, cpArray *pooledHandles)
|
cpHandleRelease(cpHandle *hand, cpArray *pooledHandles)
|
||||||
|
@ -55,7 +50,7 @@ cpSpaceHashAlloc(void)
|
||||||
return (cpSpaceHash *)cpcalloc(1, sizeof(cpSpaceHash));
|
return (cpSpaceHash *)cpcalloc(1, sizeof(cpSpaceHash));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Frees the old table, and allocates a new one.
|
// Frees the old table, and allocate a new one.
|
||||||
static void
|
static void
|
||||||
cpSpaceHashAllocTable(cpSpaceHash *hash, int numcells)
|
cpSpaceHashAllocTable(cpSpaceHash *hash, int numcells)
|
||||||
{
|
{
|
||||||
|
@ -66,12 +61,7 @@ cpSpaceHashAllocTable(cpSpaceHash *hash, int numcells)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Equality function for the handleset.
|
// Equality function for the handleset.
|
||||||
static int
|
static int handleSetEql(void *obj, cpHandle *hand){return (obj == hand->obj);}
|
||||||
handleSetEql(void *obj, void *elt)
|
|
||||||
{
|
|
||||||
cpHandle *hand = (cpHandle *)elt;
|
|
||||||
return (obj == hand->obj);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Transformation function for the handleset.
|
// Transformation function for the handleset.
|
||||||
static void *
|
static void *
|
||||||
|
@ -101,7 +91,7 @@ cpSpaceHashInit(cpSpaceHash *hash, cpFloat celldim, int numcells, cpSpaceHashBBF
|
||||||
hash->celldim = celldim;
|
hash->celldim = celldim;
|
||||||
hash->bbfunc = bbfunc;
|
hash->bbfunc = bbfunc;
|
||||||
|
|
||||||
hash->handleSet = cpHashSetNew(0, handleSetEql, (cpHashSetTransFunc)handleSetTrans);
|
hash->handleSet = cpHashSetNew(0, (cpHashSetEqlFunc)handleSetEql, (cpHashSetTransFunc)handleSetTrans);
|
||||||
hash->pooledHandles = cpArrayNew(0);
|
hash->pooledHandles = cpArrayNew(0);
|
||||||
|
|
||||||
hash->pooledBins = NULL;
|
hash->pooledBins = NULL;
|
||||||
|
@ -132,7 +122,6 @@ clearHashCell(cpSpaceHash *hash, int idx)
|
||||||
while(bin){
|
while(bin){
|
||||||
cpSpaceHashBin *next = bin->next;
|
cpSpaceHashBin *next = bin->next;
|
||||||
|
|
||||||
// Release the lock on the handle.
|
|
||||||
cpHandleRelease(bin->handle, hash->pooledHandles);
|
cpHandleRelease(bin->handle, hash->pooledHandles);
|
||||||
recycleBin(hash, bin);
|
recycleBin(hash, bin);
|
||||||
|
|
||||||
|
@ -186,15 +175,15 @@ cpSpaceHashResize(cpSpaceHash *hash, cpFloat celldim, int numcells)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return true if the chain contains the handle.
|
// Return true if the chain contains the handle.
|
||||||
static inline int
|
static inline cpBool
|
||||||
containsHandle(cpSpaceHashBin *bin, cpHandle *hand)
|
containsHandle(cpSpaceHashBin *bin, cpHandle *hand)
|
||||||
{
|
{
|
||||||
while(bin){
|
while(bin){
|
||||||
if(bin->handle == hand) return 1;
|
if(bin->handle == hand) return cpTrue;
|
||||||
bin = bin->next;
|
bin = bin->next;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return cpFalse;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get a recycled or new bin.
|
// Get a recycled or new bin.
|
||||||
|
@ -266,36 +255,32 @@ hashHandle(cpSpaceHash *hash, cpHandle *hand, cpBB bb)
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
cpSpaceHashInsert(cpSpaceHash *hash, void *obj, cpHashValue hashid, cpBB bb)
|
cpSpaceHashInsert(cpSpaceHash *hash, void *obj, cpHashValue hashid, cpBB _deprecated_unused)
|
||||||
{
|
{
|
||||||
cpHandle *hand = (cpHandle *)cpHashSetInsert(hash->handleSet, hashid, obj, hash);
|
cpHandle *hand = (cpHandle *)cpHashSetInsert(hash->handleSet, hashid, obj, hash);
|
||||||
hashHandle(hash, hand, bb);
|
hashHandle(hash, hand, hash->bbfunc(obj));
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
cpSpaceHashRehashObject(cpSpaceHash *hash, void *obj, cpHashValue hashid)
|
cpSpaceHashRehashObject(cpSpaceHash *hash, void *obj, cpHashValue hashid)
|
||||||
{
|
{
|
||||||
cpHandle *hand = (cpHandle *)cpHashSetFind(hash->handleSet, hashid, obj);
|
cpHandle *hand = (cpHandle *)cpHashSetRemove(hash->handleSet, hashid, obj);
|
||||||
hashHandle(hash, hand, hash->bbfunc(obj));
|
|
||||||
|
if(hand){
|
||||||
|
hand->obj = NULL;
|
||||||
|
cpHandleRelease(hand, hash->pooledHandles);
|
||||||
|
|
||||||
|
cpSpaceHashInsert(hash, obj, hashid, cpBBNew(0.0f, 0.0f, 0.0f, 0.0f));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hashset iterator function for rehashing the spatial hash. (hash hash hash hash?)
|
static void handleRehashHelper(cpHandle *hand, cpSpaceHash *hash){hashHandle(hash, hand, hash->bbfunc(hand->obj));}
|
||||||
static void
|
|
||||||
handleRehashHelper(void *elt, void *data)
|
|
||||||
{
|
|
||||||
cpHandle *hand = (cpHandle *)elt;
|
|
||||||
cpSpaceHash *hash = (cpSpaceHash *)data;
|
|
||||||
|
|
||||||
hashHandle(hash, hand, hash->bbfunc(hand->obj));
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
cpSpaceHashRehash(cpSpaceHash *hash)
|
cpSpaceHashRehash(cpSpaceHash *hash)
|
||||||
{
|
{
|
||||||
clearHash(hash);
|
clearHash(hash);
|
||||||
|
cpHashSetEach(hash->handleSet, (cpHashSetIterFunc)handleRehashHelper, hash);
|
||||||
// Rehash all of the handles.
|
|
||||||
cpHashSetEach(hash->handleSet, &handleRehashHelper, hash);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -309,54 +294,63 @@ cpSpaceHashRemove(cpSpaceHash *hash, void *obj, cpHashValue hashid)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Used by the cpSpaceHashEach() iterator.
|
|
||||||
typedef struct eachPair {
|
typedef struct eachPair {
|
||||||
cpSpaceHashIterator func;
|
cpSpaceHashIterator func;
|
||||||
void *data;
|
void *data;
|
||||||
} eachPair;
|
} eachPair;
|
||||||
|
|
||||||
// Calls the user iterator function. (Gross I know.)
|
static void eachHelper(cpHandle *hand, eachPair *pair){pair->func(hand->obj, pair->data);}
|
||||||
static void
|
|
||||||
eachHelper(void *elt, void *data)
|
|
||||||
{
|
|
||||||
cpHandle *hand = (cpHandle *)elt;
|
|
||||||
eachPair *pair = (eachPair *)data;
|
|
||||||
|
|
||||||
pair->func(hand->obj, pair->data);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Iterate over the objects in the spatial hash.
|
// Iterate over the objects in the spatial hash.
|
||||||
void
|
void
|
||||||
cpSpaceHashEach(cpSpaceHash *hash, cpSpaceHashIterator func, void *data)
|
cpSpaceHashEach(cpSpaceHash *hash, cpSpaceHashIterator func, void *data)
|
||||||
{
|
{
|
||||||
// Bundle the callback up to send to the hashset iterator.
|
|
||||||
eachPair pair = {func, data};
|
eachPair pair = {func, data};
|
||||||
|
cpHashSetEach(hash->handleSet, (cpHashSetIterFunc)eachHelper, &pair);
|
||||||
cpHashSetEach(hash->handleSet, &eachHelper, &pair);
|
}
|
||||||
|
|
||||||
|
static inline void
|
||||||
|
removeOrphanedHandles(cpSpaceHash *hash, cpSpaceHashBin **bin_ptr)
|
||||||
|
{
|
||||||
|
cpSpaceHashBin *bin = *bin_ptr;
|
||||||
|
while(bin){
|
||||||
|
cpHandle *hand = bin->handle;
|
||||||
|
cpSpaceHashBin *next = bin->next;
|
||||||
|
|
||||||
|
if(!hand->obj){
|
||||||
|
// orphaned handle, unlink and recycle the bin
|
||||||
|
(*bin_ptr) = bin->next;
|
||||||
|
recycleBin(hash, bin);
|
||||||
|
|
||||||
|
cpHandleRelease(hand, hash->pooledHandles);
|
||||||
|
} else {
|
||||||
|
bin_ptr = &bin->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
bin = next;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calls the callback function for the objects in a given chain.
|
// Calls the callback function for the objects in a given chain.
|
||||||
static inline void
|
static inline void
|
||||||
query(cpSpaceHash *hash, cpSpaceHashBin *bin, void *obj, cpSpaceHashQueryFunc func, void *data)
|
query(cpSpaceHash *hash, cpSpaceHashBin **bin_ptr, void *obj, cpSpaceHashQueryFunc func, void *data)
|
||||||
{
|
{
|
||||||
for(; bin; bin = bin->next){
|
restart:
|
||||||
|
for(cpSpaceHashBin *bin = *bin_ptr; bin; bin = bin->next){
|
||||||
cpHandle *hand = bin->handle;
|
cpHandle *hand = bin->handle;
|
||||||
void *other = hand->obj;
|
void *other = hand->obj;
|
||||||
|
|
||||||
// Skip over certain conditions
|
if(hand->stamp == hash->stamp || obj == other){
|
||||||
if(
|
continue;
|
||||||
// Have we already tried this pair in this query?
|
} else if(other){
|
||||||
hand->stamp == hash->stamp
|
func(obj, other, data);
|
||||||
// Is obj the same as other?
|
hand->stamp = hash->stamp;
|
||||||
|| obj == other
|
} else {
|
||||||
// Has other been removed since the last rehash?
|
// The object for this handle has been removed
|
||||||
|| !other
|
// cleanup this cell and restart the query
|
||||||
) continue;
|
removeOrphanedHandles(hash, bin_ptr);
|
||||||
|
goto restart; // GCC not smart enough/able to tail call an inlined function.
|
||||||
func(obj, other, data);
|
}
|
||||||
|
|
||||||
// Stamp that the handle was checked already against this object.
|
|
||||||
hand->stamp = hash->stamp;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -366,10 +360,7 @@ cpSpaceHashPointQuery(cpSpaceHash *hash, cpVect point, cpSpaceHashQueryFunc func
|
||||||
cpFloat dim = hash->celldim;
|
cpFloat dim = hash->celldim;
|
||||||
int idx = hash_func(floor_int(point.x/dim), floor_int(point.y/dim), hash->numcells); // Fix by ShiftZ
|
int idx = hash_func(floor_int(point.x/dim), floor_int(point.y/dim), hash->numcells); // Fix by ShiftZ
|
||||||
|
|
||||||
query(hash, hash->table[idx], &point, func, data);
|
query(hash, &hash->table[idx], &point, func, data);
|
||||||
|
|
||||||
// Increment the stamp.
|
|
||||||
// Only one cell is checked, but query() requires it anyway.
|
|
||||||
hash->stamp++;
|
hash->stamp++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -384,16 +375,15 @@ cpSpaceHashQuery(cpSpaceHash *hash, void *obj, cpBB bb, cpSpaceHashQueryFunc fun
|
||||||
int t = floor_int(bb.t/dim);
|
int t = floor_int(bb.t/dim);
|
||||||
|
|
||||||
int n = hash->numcells;
|
int n = hash->numcells;
|
||||||
|
cpSpaceHashBin **table = hash->table;
|
||||||
|
|
||||||
// Iterate over the cells and query them.
|
// Iterate over the cells and query them.
|
||||||
for(int i=l; i<=r; i++){
|
for(int i=l; i<=r; i++){
|
||||||
for(int j=b; j<=t; j++){
|
for(int j=b; j<=t; j++){
|
||||||
int idx = hash_func(i,j,n);
|
query(hash, &table[hash_func(i,j,n)], obj, func, data);
|
||||||
query(hash, hash->table[idx], obj, func, data);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Increment the stamp.
|
|
||||||
hash->stamp++;
|
hash->stamp++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -425,29 +415,27 @@ handleQueryRehashHelper(void *elt, void *data)
|
||||||
int r = floor_int(bb.r/dim);
|
int r = floor_int(bb.r/dim);
|
||||||
int b = floor_int(bb.b/dim);
|
int b = floor_int(bb.b/dim);
|
||||||
int t = floor_int(bb.t/dim);
|
int t = floor_int(bb.t/dim);
|
||||||
|
|
||||||
|
cpSpaceHashBin **table = hash->table;
|
||||||
|
|
||||||
for(int i=l; i<=r; i++){
|
for(int i=l; i<=r; i++){
|
||||||
for(int j=b; j<=t; j++){
|
for(int j=b; j<=t; j++){
|
||||||
// // exit the loops if the object has been deleted in func().
|
|
||||||
// if(!hand->obj) goto break_out;
|
|
||||||
|
|
||||||
int idx = hash_func(i,j,n);
|
int idx = hash_func(i,j,n);
|
||||||
cpSpaceHashBin *bin = hash->table[idx];
|
cpSpaceHashBin *bin = table[idx];
|
||||||
|
|
||||||
if(containsHandle(bin, hand)) continue;
|
if(containsHandle(bin, hand)) continue;
|
||||||
|
|
||||||
cpHandleRetain(hand); // this MUST be done first in case the object is removed in func()
|
cpHandleRetain(hand); // this MUST be done first in case the object is removed in func()
|
||||||
query(hash, bin, obj, func, pair->data);
|
query(hash, &bin, obj, func, pair->data);
|
||||||
|
|
||||||
cpSpaceHashBin *newBin = getEmptyBin(hash);
|
cpSpaceHashBin *newBin = getEmptyBin(hash);
|
||||||
newBin->handle = hand;
|
newBin->handle = hand;
|
||||||
newBin->next = bin;
|
newBin->next = bin;
|
||||||
hash->table[idx] = newBin;
|
table[idx] = newBin;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// break_out:
|
// Increment the stamp for each object hashed.
|
||||||
// Increment the stamp for each object we hash.
|
|
||||||
hash->stamp++;
|
hash->stamp++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -461,26 +449,27 @@ cpSpaceHashQueryRehash(cpSpaceHash *hash, cpSpaceHashQueryFunc func, void *data)
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline cpFloat
|
static inline cpFloat
|
||||||
segmentQuery(cpSpaceHash *hash, cpSpaceHashBin *bin, void *obj, cpSpaceHashSegmentQueryFunc func, void *data)
|
segmentQuery(cpSpaceHash *hash, cpSpaceHashBin **bin_ptr, void *obj, cpSpaceHashSegmentQueryFunc func, void *data)
|
||||||
{
|
{
|
||||||
cpFloat t = 1.0f;
|
cpFloat t = 1.0f;
|
||||||
|
|
||||||
for(; bin; bin = bin->next){
|
restart:
|
||||||
|
for(cpSpaceHashBin *bin = *bin_ptr; bin; bin = bin->next){
|
||||||
cpHandle *hand = bin->handle;
|
cpHandle *hand = bin->handle;
|
||||||
void *other = hand->obj;
|
void *other = hand->obj;
|
||||||
|
|
||||||
// Skip over certain conditions
|
// Skip over certain conditions
|
||||||
if(
|
if(hand->stamp == hash->stamp){
|
||||||
// Have we already tried this pair in this query?
|
continue;
|
||||||
hand->stamp == hash->stamp
|
} else if(other){
|
||||||
// Has other been removed since the last rehash?
|
t = cpfmin(t, func(obj, other, data));
|
||||||
|| !other
|
hand->stamp = hash->stamp;
|
||||||
) continue;
|
} else {
|
||||||
|
// The object for this handle has been removed
|
||||||
// Stamp that the handle was checked already against this object.
|
// cleanup this cell and restart the query
|
||||||
hand->stamp = hash->stamp;
|
removeOrphanedHandles(hash, bin_ptr);
|
||||||
|
goto restart; // GCC not smart enough/able to tail call an inlined function.
|
||||||
t = cpfmin(t, func(obj, other, data));
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return t;
|
return t;
|
||||||
|
@ -492,8 +481,6 @@ void cpSpaceHashSegmentQuery(cpSpaceHash *hash, void *obj, cpVect a, cpVect b, c
|
||||||
a = cpvmult(a, 1.0f/hash->celldim);
|
a = cpvmult(a, 1.0f/hash->celldim);
|
||||||
b = cpvmult(b, 1.0f/hash->celldim);
|
b = cpvmult(b, 1.0f/hash->celldim);
|
||||||
|
|
||||||
cpFloat dt_dx = 1.0f/cpfabs(b.x - a.x), dt_dy = 1.0f/cpfabs(b.y - a.y);
|
|
||||||
|
|
||||||
int cell_x = floor_int(a.x), cell_y = floor_int(a.y);
|
int cell_x = floor_int(a.x), cell_y = floor_int(a.y);
|
||||||
|
|
||||||
cpFloat t = 0;
|
cpFloat t = 0;
|
||||||
|
@ -517,14 +504,20 @@ void cpSpaceHashSegmentQuery(cpSpaceHash *hash, void *obj, cpVect a, cpVect b, c
|
||||||
temp_v = (a.y - cpffloor(a.y));
|
temp_v = (a.y - cpffloor(a.y));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Division by zero is *very* slow on ARM
|
||||||
|
cpFloat dx = cpfabs(b.x - a.x), dy = cpfabs(b.y - a.y);
|
||||||
|
cpFloat dt_dx = (dx ? 1.0f/dx : INFINITY), dt_dy = (dy ? 1.0f/dy : INFINITY);
|
||||||
|
|
||||||
// fix NANs in horizontal directions
|
// fix NANs in horizontal directions
|
||||||
cpFloat next_h = (temp_h ? temp_h*dt_dx : dt_dx);
|
cpFloat next_h = (temp_h ? temp_h*dt_dx : dt_dx);
|
||||||
cpFloat next_v = (temp_v ? temp_v*dt_dy : dt_dy);
|
cpFloat next_v = (temp_v ? temp_v*dt_dy : dt_dy);
|
||||||
|
|
||||||
|
cpSpaceHashBin **table = hash->table;
|
||||||
|
|
||||||
int n = hash->numcells;
|
int n = hash->numcells;
|
||||||
while(t < t_exit){
|
while(t < t_exit){
|
||||||
int idx = hash_func(cell_x, cell_y, n);
|
int idx = hash_func(cell_x, cell_y, n);
|
||||||
t_exit = cpfmin(t_exit, segmentQuery(hash, hash->table[idx], obj, func, data));
|
t_exit = cpfmin(t_exit, segmentQuery(hash, &table[idx], obj, func, data));
|
||||||
|
|
||||||
if (next_v < next_h){
|
if (next_v < next_h){
|
||||||
cell_y += y_inc;
|
cell_y += y_inc;
|
||||||
|
|
Loading…
Reference in New Issue