Bullet Collision Detection & Physics Library
btHingeConstraint.h
Go to the documentation of this file.
1/*
2Bullet Continuous Collision Detection and Physics Library
3Copyright (c) 2003-2006 Erwin Coumans http://continuousphysics.com/Bullet/
4
5This software is provided 'as-is', without any express or implied warranty.
6In no event will the authors be held liable for any damages arising from the use of this software.
7Permission is granted to anyone to use this software for any purpose,
8including commercial applications, and to alter it and redistribute it freely,
9subject to the following restrictions:
10
111. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required.
122. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
133. This notice may not be removed or altered from any source distribution.
14*/
15
16/* Hinge Constraint by Dirk Gregorius. Limits added by Marcus Hennix at Starbreeze Studios */
17
18#ifndef BT_HINGECONSTRAINT_H
19#define BT_HINGECONSTRAINT_H
20
21#define _BT_USE_CENTER_LIMIT_ 1
22
23
25#include "btJacobianEntry.h"
26#include "btTypedConstraint.h"
27
28class btRigidBody;
29
30#ifdef BT_USE_DOUBLE_PRECISION
31#define btHingeConstraintData btHingeConstraintDoubleData2 //rename to 2 for backwards compatibility, so we can still load the 'btHingeConstraintDoubleData' version
32#define btHingeConstraintDataName "btHingeConstraintDoubleData2"
33#else
34#define btHingeConstraintData btHingeConstraintFloatData
35#define btHingeConstraintDataName "btHingeConstraintFloatData"
36#endif //BT_USE_DOUBLE_PRECISION
37
38
39
41{
46};
47
48
52{
53#ifdef IN_PARALLELL_SOLVER
54public:
55#endif
56 btJacobianEntry m_jac[3]; //3 orthogonal linear constraints
57 btJacobianEntry m_jacAng[3]; //2 orthogonal angular constraints+ 1 for limit/motor
58
59 btTransform m_rbAFrame; // constraint axii. Assumes z is hinge axis.
61
64
65
66#ifdef _BT_USE_CENTER_LIMIT_
68#else
69 btScalar m_lowerLimit;
70 btScalar m_upperLimit;
72 btScalar m_correction;
73
74 btScalar m_limitSoftness;
75 btScalar m_biasFactor;
76 btScalar m_relaxationFactor;
77
78 bool m_solveLimit;
79#endif
80
82
83
87
93
95
101
102
103public:
104
106
108
110
112
114
115
116 virtual void buildJacobian();
117
118 virtual void getInfo1 (btConstraintInfo1* info);
119
120 void getInfo1NonVirtual(btConstraintInfo1* info);
121
122 virtual void getInfo2 (btConstraintInfo2* info);
123
124 void getInfo2NonVirtual(btConstraintInfo2* info,const btTransform& transA,const btTransform& transB,const btVector3& angVelA,const btVector3& angVelB);
125
126 void getInfo2Internal(btConstraintInfo2* info,const btTransform& transA,const btTransform& transB,const btVector3& angVelA,const btVector3& angVelB);
127 void getInfo2InternalUsingFrameOffset(btConstraintInfo2* info,const btTransform& transA,const btTransform& transB,const btVector3& angVelA,const btVector3& angVelB);
128
129
130 void updateRHS(btScalar timeStep);
131
133 {
134 return m_rbA;
135 }
137 {
138 return m_rbB;
139 }
140
142 {
143 return m_rbA;
144 }
145
147 {
148 return m_rbB;
149 }
150
152 {
153 return m_rbAFrame;
154 }
155
157 {
158 return m_rbBFrame;
159 }
160
161 void setFrames(const btTransform& frameA, const btTransform& frameB);
162
164 {
165 m_angularOnly = angularOnly;
166 }
167
169 {
170 m_enableAngularMotor = enableMotor;
171 m_motorTargetVelocity = targetVelocity;
172 m_maxMotorImpulse = maxMotorImpulse;
173 }
174
175 // extra motor API, including ability to set a target rotation (as opposed to angular velocity)
176 // note: setMotorTarget sets angular velocity under the hood, so you must call it every tick to
177 // maintain a given angular target.
178 void enableMotor(bool enableMotor) { m_enableAngularMotor = enableMotor; }
181 void setMotorTarget(const btQuaternion& qAinB, btScalar dt); // qAinB is rotation of body A wrt body B.
182 void setMotorTarget(btScalar targetAngle, btScalar dt);
183
184
186 {
187#ifdef _BT_USE_CENTER_LIMIT_
188 m_limit.set(low, high, _softness, _biasFactor, _relaxationFactor);
189#else
190 m_lowerLimit = btNormalizeAngle(low);
191 m_upperLimit = btNormalizeAngle(high);
192 m_limitSoftness = _softness;
193 m_biasFactor = _biasFactor;
194 m_relaxationFactor = _relaxationFactor;
195#endif
196 }
197
199 {
200#ifdef _BT_USE_CENTER_LIMIT_
201 return m_limit.getSoftness();
202#else
203 return m_limitSoftness;
204#endif
205 }
206
208 {
209#ifdef _BT_USE_CENTER_LIMIT_
210 return m_limit.getBiasFactor();
211#else
212 return m_biasFactor;
213#endif
214 }
215
217 {
218#ifdef _BT_USE_CENTER_LIMIT_
219 return m_limit.getRelaxationFactor();
220#else
221 return m_relaxationFactor;
222#endif
223 }
224
226 {
229 btVector3 pivotInA = m_rbAFrame.getOrigin();
230// m_rbAFrame.getOrigin() = pivotInA;
231 m_rbAFrame.getBasis().setValue( rbAxisA1.getX(),rbAxisA2.getX(),axisInA.getX(),
232 rbAxisA1.getY(),rbAxisA2.getY(),axisInA.getY(),
233 rbAxisA1.getZ(),rbAxisA2.getZ(),axisInA.getZ() );
234
235 btVector3 axisInB = m_rbA.getCenterOfMassTransform().getBasis() * axisInA;
236
240
241 m_rbBFrame.getOrigin() = m_rbB.getCenterOfMassTransform().inverse()(m_rbA.getCenterOfMassTransform()(pivotInA));
242
243 m_rbBFrame.getBasis().setValue( rbAxisB1.getX(),rbAxisB2.getX(),axisInB.getX(),
244 rbAxisB1.getY(),rbAxisB2.getY(),axisInB.getY(),
245 rbAxisB1.getZ(),rbAxisB2.getZ(),axisInB.getZ() );
246 m_rbBFrame.getBasis() = m_rbB.getCenterOfMassTransform().getBasis().inverse() * m_rbBFrame.getBasis();
247
248 }
249
250 bool hasLimit() const {
251#ifdef _BT_USE_CENTER_LIMIT_
252 return m_limit.getHalfRange() > 0;
253#else
254 return m_lowerLimit <= m_upperLimit;
255#endif
256 }
257
259 {
260#ifdef _BT_USE_CENTER_LIMIT_
261 return m_limit.getLow();
262#else
263 return m_lowerLimit;
264#endif
265 }
266
268 {
269#ifdef _BT_USE_CENTER_LIMIT_
270 return m_limit.getHigh();
271#else
272 return m_upperLimit;
273#endif
274 }
275
276
278 btScalar getHingeAngle();
279
280 btScalar getHingeAngle(const btTransform& transA,const btTransform& transB);
281
282 void testLimit(const btTransform& transA,const btTransform& transB);
283
284
285 const btTransform& getAFrame() const { return m_rbAFrame; };
286 const btTransform& getBFrame() const { return m_rbBFrame; };
287
288 btTransform& getAFrame() { return m_rbAFrame; };
289 btTransform& getBFrame() { return m_rbBFrame; };
290
291 inline int getSolveLimit()
292 {
293#ifdef _BT_USE_CENTER_LIMIT_
294 return m_limit.isLimit();
295#else
296 return m_solveLimit;
297#endif
298 }
299
301 {
302#ifdef _BT_USE_CENTER_LIMIT_
303 return m_limit.getSign();
304#else
305 return m_limitSign;
306#endif
307 }
308
309 inline bool getAngularOnly()
310 {
311 return m_angularOnly;
312 }
314 {
315 return m_enableAngularMotor;
316 }
318 {
319 return m_motorTargetVelocity;
320 }
322 {
323 return m_maxMotorImpulse;
324 }
325 // access for UseFrameOffset
326 bool getUseFrameOffset() { return m_useOffsetForConstraintFrame; }
327 void setUseFrameOffset(bool frameOffsetOnOff) { m_useOffsetForConstraintFrame = frameOffsetOnOff; }
328 // access for UseReferenceFrameA
329 bool getUseReferenceFrameA() const { return m_useReferenceFrameA; }
330 void setUseReferenceFrameA(bool useReferenceFrameA) { m_useReferenceFrameA = useReferenceFrameA; }
331
334 virtual void setParam(int num, btScalar value, int axis = -1);
336 virtual btScalar getParam(int num, int axis = -1) const;
337
338 virtual int getFlags() const
339 {
340 return m_flags;
341 }
342
343 virtual int calculateSerializeBufferSize() const;
344
346 virtual const char* serialize(void* dataBuffer, btSerializer* serializer) const;
347
348
349};
350
351
352//only for backward compatibility
353#ifdef BT_BACKWARDS_COMPATIBLE_SERIALIZATION
356{
358 btTransformDoubleData m_rbAFrame; // constraint axii. Assumes z is hinge axis.
365
371
372};
373#endif //BT_BACKWARDS_COMPATIBLE_SERIALIZATION
374
377{
378protected:
380public:
381
383
386 {
387 m_accumulatedAngle=getHingeAngle();
388 }
389
392 {
393 m_accumulatedAngle=getHingeAngle();
394 }
395
398 {
399 m_accumulatedAngle=getHingeAngle();
400 }
401
404 {
405 m_accumulatedAngle=getHingeAngle();
406 }
407 btScalar getAccumulatedHingeAngle();
408 void setAccumulatedHingeAngle(btScalar accAngle);
409 virtual void getInfo1 (btConstraintInfo1* info);
410
411};
412
414{
416 btTransformFloatData m_rbAFrame; // constraint axii. Assumes z is hinge axis.
420
424
430
431};
432
433
434
437{
439 btTransformDoubleData m_rbAFrame; // constraint axii. Assumes z is hinge axis.
446
452 char m_padding1[4];
453
454};
455
456
457
458
460{
461 return sizeof(btHingeConstraintData);
462}
463
466{
468 btTypedConstraint::serialize(&hingeData->m_typeConstraintData,serializer);
469
470 m_rbAFrame.serialize(hingeData->m_rbAFrame);
471 m_rbBFrame.serialize(hingeData->m_rbBFrame);
472
473 hingeData->m_angularOnly = m_angularOnly;
474 hingeData->m_enableAngularMotor = m_enableAngularMotor;
475 hingeData->m_maxMotorImpulse = float(m_maxMotorImpulse);
476 hingeData->m_motorTargetVelocity = float(m_motorTargetVelocity);
477 hingeData->m_useReferenceFrameA = m_useReferenceFrameA;
478#ifdef _BT_USE_CENTER_LIMIT_
479 hingeData->m_lowerLimit = float(m_limit.getLow());
480 hingeData->m_upperLimit = float(m_limit.getHigh());
481 hingeData->m_limitSoftness = float(m_limit.getSoftness());
482 hingeData->m_biasFactor = float(m_limit.getBiasFactor());
483 hingeData->m_relaxationFactor = float(m_limit.getRelaxationFactor());
484#else
485 hingeData->m_lowerLimit = float(m_lowerLimit);
486 hingeData->m_upperLimit = float(m_upperLimit);
487 hingeData->m_limitSoftness = float(m_limitSoftness);
488 hingeData->m_biasFactor = float(m_biasFactor);
489 hingeData->m_relaxationFactor = float(m_relaxationFactor);
490#endif
491
492 // Fill padding with zeros to appease msan.
493#ifdef BT_USE_DOUBLE_PRECISION
494 hingeData->m_padding1[0] = 0;
495 hingeData->m_padding1[1] = 0;
496 hingeData->m_padding1[2] = 0;
497 hingeData->m_padding1[3] = 0;
498#endif
499
501}
502
503#endif //BT_HINGECONSTRAINT_H
btHingeFlags
@ BT_HINGE_FLAGS_CFM_STOP
@ BT_HINGE_FLAGS_CFM_NORM
@ BT_HINGE_FLAGS_ERP_NORM
@ BT_HINGE_FLAGS_ERP_STOP
#define btHingeConstraintData
#define btHingeConstraintDataName
const T & btMax(const T &a, const T &b)
Definition btMinMax.h:29
btQuaternion shortestArcQuat(const btVector3 &v0, const btVector3 &v1)
btVector3 quatRotate(const btQuaternion &rotation, const btVector3 &v)
btScalar btNormalizeAngle(btScalar angleInRadians)
Definition btScalar.h:759
float btScalar
The btScalar type abstracts floating point numbers, to easily switch between double and single floati...
Definition btScalar.h:292
#define ATTRIBUTE_ALIGNED16(a)
Definition btScalar.h:82
#define SIMD_FORCE_INLINE
Definition btScalar.h:81
void btPlaneSpace1(const T &n, T &p, T &q)
Definition btVector3.h:1283
bool isLimit() const
Returns true when the last test() invocation recognized limit violation.
btScalar getBiasFactor() const
Returns limit's bias factor.
btScalar getLow() const
btScalar getHigh() const
btScalar getSoftness() const
Returns limit's softness.
btScalar getSign() const
Returns sign value evaluated when test() was invoked.
void set(btScalar low, btScalar high, btScalar _softness=0.9f, btScalar _biasFactor=0.3f, btScalar _relaxationFactor=1.0f)
Sets all limit's parameters.
btScalar getHalfRange() const
Gives half of the distance between min and max limit angle.
btScalar getRelaxationFactor() const
Returns limit's relaxation factor.
The getAccumulatedHingeAngle returns the accumulated hinge angle, taking rotation across the -PI/PI b...
btHingeAccumulatedAngleConstraint(btRigidBody &rbA, const btVector3 &pivotInA, const btVector3 &axisInA, bool useReferenceFrameA=false)
btHingeAccumulatedAngleConstraint(btRigidBody &rbA, btRigidBody &rbB, const btVector3 &pivotInA, const btVector3 &pivotInB, const btVector3 &axisInA, const btVector3 &axisInB, bool useReferenceFrameA=false)
btHingeAccumulatedAngleConstraint(btRigidBody &rbA, btRigidBody &rbB, const btTransform &rbAFrame, const btTransform &rbBFrame, bool useReferenceFrameA=false)
btHingeAccumulatedAngleConstraint(btRigidBody &rbA, const btTransform &rbAFrame, bool useReferenceFrameA=false)
hinge constraint between two rigidbodies each with a pivotpoint that descibes the axis location in lo...
const btRigidBody & getRigidBodyB() const
virtual int getFlags() const
void setAngularOnly(bool angularOnly)
const btTransform & getAFrame() const
btScalar getUpperLimit() const
btScalar getLimitRelaxationFactor() const
void setUseReferenceFrameA(bool useReferenceFrameA)
const btTransform & getBFrame() const
btScalar getLowerLimit() const
btTransform & getFrameOffsetA()
btScalar getLimitSoftness() const
void setAxis(btVector3 &axisInA)
void setMotorTargetVelocity(btScalar motorTargetVelocity)
virtual int calculateSerializeBufferSize() const
btScalar getLimitBiasFactor() const
void setMaxMotorImpulse(btScalar maxMotorImpulse)
btRigidBody & getRigidBodyA()
void setLimit(btScalar low, btScalar high, btScalar _softness=0.9f, btScalar _biasFactor=0.3f, btScalar _relaxationFactor=1.0f)
void setUseFrameOffset(bool frameOffsetOnOff)
void enableAngularMotor(bool enableMotor, btScalar targetVelocity, btScalar maxMotorImpulse)
btAngularLimit m_limit
btTransform & getFrameOffsetB()
bool getUseReferenceFrameA() const
void enableMotor(bool enableMotor)
btTransform & getBFrame()
btScalar getMaxMotorImpulse()
btScalar getMotorTargetVelocity()
btTransform & getAFrame()
const btRigidBody & getRigidBodyA() const
virtual const char * serialize(void *dataBuffer, btSerializer *serializer) const
fills the dataBuffer and returns the struct name (and 0 on failure)
btRigidBody & getRigidBodyB()
Jacobian entry is an abstraction that allows to describe constraints it can be used in combination wi...
btMatrix3x3 inverse() const
Return the inverse of the matrix.
void setValue(const btScalar &xx, const btScalar &xy, const btScalar &xz, const btScalar &yx, const btScalar &yy, const btScalar &yz, const btScalar &zx, const btScalar &zy, const btScalar &zz)
Set the values of the matrix explicitly (row major)
The btQuaternion implements quaternion to perform linear algebra rotations in combination with btMatr...
The btRigidBody is the main class for rigid body objects.
Definition btRigidBody.h:63
virtual const char * serialize(void *dataBuffer, class btSerializer *serializer) const
fills the dataBuffer and returns the struct name (and 0 on failure)
virtual int calculateSerializeBufferSize() const
The btTransform class supports rigid transforms with only translation and rotation and no scaling/she...
Definition btTransform.h:34
void serialize(struct btTransformData &dataOut) const
btMatrix3x3 & getBasis()
Return the basis matrix for the rotation.
btVector3 & getOrigin()
Return the origin vector translation.
TypedConstraint is the baseclass for Bullet constraints and vehicles.
virtual const char * serialize(void *dataBuffer, btSerializer *serializer) const
fills the dataBuffer and returns the struct name (and 0 on failure)
btVector3 can be used to represent 3D points and vectors.
Definition btVector3.h:84
btVector3 cross(const btVector3 &v) const
Return the cross product between this and another vector.
Definition btVector3.h:389
do not change those serialization structures, it requires an updated sBulletDNAstr/sBulletDNAstr64
btTypedConstraintDoubleData m_typeConstraintData
btTransformDoubleData m_rbBFrame
btTransformDoubleData m_rbAFrame
this structure is not used, except for loading pre-2.82 .bullet files
btTransformDoubleData m_rbAFrame
btTransformDoubleData m_rbBFrame
btTypedConstraintData m_typeConstraintData
btTypedConstraintData m_typeConstraintData
btTransformFloatData m_rbAFrame
btTransformFloatData m_rbBFrame
for serialization
this structure is not used, except for loading pre-2.82 .bullet files