Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/questsystem5/src/bullet/BulletDynamics/ConstraintSolver/btConeTwistConstraint.cpp @ 2916

Last change on this file since 2916 was 2908, checked in by dafrick, 16 years ago

Reverted to revision 2906 (because I'm too stupid to merge correctly, 2nd try will follow shortly. ;))

  • Property svn:eol-style set to native
File size: 9.3 KB
RevLine 
[1963]1/*
2Bullet Continuous Collision Detection and Physics Library
3btConeTwistConstraint is Copyright (c) 2007 Starbreeze Studios
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
15Written by: Marcus Hennix
16*/
17
18
19#include "btConeTwistConstraint.h"
20#include "BulletDynamics/Dynamics/btRigidBody.h"
21#include "LinearMath/btTransformUtil.h"
22#include "LinearMath/btMinMax.h"
23#include <new>
24
25btConeTwistConstraint::btConeTwistConstraint()
[2908]26:btTypedConstraint(CONETWIST_CONSTRAINT_TYPE)
[1963]27{
28}
29
30
31btConeTwistConstraint::btConeTwistConstraint(btRigidBody& rbA,btRigidBody& rbB, 
32                                                                                         const btTransform& rbAFrame,const btTransform& rbBFrame)
33                                                                                         :btTypedConstraint(CONETWIST_CONSTRAINT_TYPE, rbA,rbB),m_rbAFrame(rbAFrame),m_rbBFrame(rbBFrame),
[2908]34                                                                                         m_angularOnly(false)
[1963]35{
[2908]36        m_swingSpan1 = btScalar(1e30);
37        m_swingSpan2 = btScalar(1e30);
38        m_twistSpan  = btScalar(1e30);
39        m_biasFactor = 0.3f;
40        m_relaxationFactor = 1.0f;
41
42        m_solveTwistLimit = false;
43        m_solveSwingLimit = false;
44
[1963]45}
46
47btConeTwistConstraint::btConeTwistConstraint(btRigidBody& rbA,const btTransform& rbAFrame)
48                                                                                        :btTypedConstraint(CONETWIST_CONSTRAINT_TYPE,rbA),m_rbAFrame(rbAFrame),
[2908]49                                                                                         m_angularOnly(false)
[1963]50{
51        m_rbBFrame = m_rbAFrame;
[2908]52       
53        m_swingSpan1 = btScalar(1e30);
54        m_swingSpan2 = btScalar(1e30);
55        m_twistSpan  = btScalar(1e30);
56        m_biasFactor = 0.3f;
57        m_relaxationFactor = 1.0f;
[1963]58
[2908]59        m_solveTwistLimit = false;
60        m_solveSwingLimit = false;
61       
62}                       
[1963]63
[2908]64void    btConeTwistConstraint::buildJacobian()
[1963]65{
[2908]66        m_appliedImpulse = btScalar(0.);
67
68        //set bias, sign, clear accumulator
69        m_swingCorrection = btScalar(0.);
70        m_twistLimitSign = btScalar(0.);
[1963]71        m_solveTwistLimit = false;
72        m_solveSwingLimit = false;
[2908]73        m_accTwistLimitImpulse = btScalar(0.);
74        m_accSwingLimitImpulse = btScalar(0.);
[1963]75
[2908]76        if (!m_angularOnly)
[1963]77        {
[2908]78                btVector3 pivotAInW = m_rbA.getCenterOfMassTransform()*m_rbAFrame.getOrigin();
79                btVector3 pivotBInW = m_rbB.getCenterOfMassTransform()*m_rbBFrame.getOrigin();
80                btVector3 relPos = pivotBInW - pivotAInW;
[1963]81
[2908]82                btVector3 normal[3];
83                if (relPos.length2() > SIMD_EPSILON)
[1963]84                {
[2908]85                        normal[0] = relPos.normalized();
[1963]86                }
87                else
88                {
[2908]89                        normal[0].setValue(btScalar(1.0),0,0);
[1963]90                }
91
[2908]92                btPlaneSpace1(normal[0], normal[1], normal[2]);
[1963]93
[2908]94                for (int i=0;i<3;i++)
[1963]95                {
[2908]96                        new (&m_jac[i]) btJacobianEntry(
[1963]97                                m_rbA.getCenterOfMassTransform().getBasis().transpose(),
98                                m_rbB.getCenterOfMassTransform().getBasis().transpose(),
99                                pivotAInW - m_rbA.getCenterOfMassPosition(),
100                                pivotBInW - m_rbB.getCenterOfMassPosition(),
101                                normal[i],
102                                m_rbA.getInvInertiaDiagLocal(),
103                                m_rbA.getInvMass(),
104                                m_rbB.getInvInertiaDiagLocal(),
105                                m_rbB.getInvMass());
106                }
107        }
108
109        btVector3 b1Axis1,b1Axis2,b1Axis3;
110        btVector3 b2Axis1,b2Axis2;
111
112        b1Axis1 = getRigidBodyA().getCenterOfMassTransform().getBasis() * this->m_rbAFrame.getBasis().getColumn(0);
113        b2Axis1 = getRigidBodyB().getCenterOfMassTransform().getBasis() * this->m_rbBFrame.getBasis().getColumn(0);
114
115        btScalar swing1=btScalar(0.),swing2 = btScalar(0.);
116
117        btScalar swx=btScalar(0.),swy = btScalar(0.);
118        btScalar thresh = btScalar(10.);
119        btScalar fact;
120
121        // Get Frame into world space
122        if (m_swingSpan1 >= btScalar(0.05f))
123        {
124                b1Axis2 = getRigidBodyA().getCenterOfMassTransform().getBasis() * this->m_rbAFrame.getBasis().getColumn(1);
[2908]125//              swing1  = btAtan2Fast( b2Axis1.dot(b1Axis2),b2Axis1.dot(b1Axis1) );
[1963]126                swx = b2Axis1.dot(b1Axis1);
127                swy = b2Axis1.dot(b1Axis2);
128                swing1  = btAtan2Fast(swy, swx);
129                fact = (swy*swy + swx*swx) * thresh * thresh;
130                fact = fact / (fact + btScalar(1.0));
131                swing1 *= fact; 
[2908]132
[1963]133        }
134
135        if (m_swingSpan2 >= btScalar(0.05f))
136        {
137                b1Axis3 = getRigidBodyA().getCenterOfMassTransform().getBasis() * this->m_rbAFrame.getBasis().getColumn(2);                     
[2908]138//              swing2 = btAtan2Fast( b2Axis1.dot(b1Axis3),b2Axis1.dot(b1Axis1) );
[1963]139                swx = b2Axis1.dot(b1Axis1);
140                swy = b2Axis1.dot(b1Axis3);
141                swing2  = btAtan2Fast(swy, swx);
142                fact = (swy*swy + swx*swx) * thresh * thresh;
143                fact = fact / (fact + btScalar(1.0));
144                swing2 *= fact; 
145        }
146
147        btScalar RMaxAngle1Sq = 1.0f / (m_swingSpan1*m_swingSpan1);             
148        btScalar RMaxAngle2Sq = 1.0f / (m_swingSpan2*m_swingSpan2);     
149        btScalar EllipseAngle = btFabs(swing1*swing1)* RMaxAngle1Sq + btFabs(swing2*swing2) * RMaxAngle2Sq;
150
151        if (EllipseAngle > 1.0f)
152        {
153                m_swingCorrection = EllipseAngle-1.0f;
154                m_solveSwingLimit = true;
[2908]155               
[1963]156                // Calculate necessary axis & factors
157                m_swingAxis = b2Axis1.cross(b1Axis2* b2Axis1.dot(b1Axis2) + b1Axis3* b2Axis1.dot(b1Axis3));
158                m_swingAxis.normalize();
[2908]159
[1963]160                btScalar swingAxisSign = (b2Axis1.dot(b1Axis1) >= 0.0f) ? 1.0f : -1.0f;
161                m_swingAxis *= swingAxisSign;
[2908]162
163                m_kSwing =  btScalar(1.) / (getRigidBodyA().computeAngularImpulseDenominator(m_swingAxis) +
164                        getRigidBodyB().computeAngularImpulseDenominator(m_swingAxis));
165
[1963]166        }
167
168        // Twist limits
169        if (m_twistSpan >= btScalar(0.))
170        {
171                btVector3 b2Axis2 = getRigidBodyB().getCenterOfMassTransform().getBasis() * this->m_rbBFrame.getBasis().getColumn(1);
172                btQuaternion rotationArc = shortestArcQuat(b2Axis1,b1Axis1);
173                btVector3 TwistRef = quatRotate(rotationArc,b2Axis2); 
174                btScalar twist = btAtan2Fast( TwistRef.dot(b1Axis3), TwistRef.dot(b1Axis2) );
175
[2908]176                btScalar lockedFreeFactor = (m_twistSpan > btScalar(0.05f)) ? m_limitSoftness : btScalar(0.);
[1963]177                if (twist <= -m_twistSpan*lockedFreeFactor)
178                {
179                        m_twistCorrection = -(twist + m_twistSpan);
180                        m_solveTwistLimit = true;
[2908]181
[1963]182                        m_twistAxis = (b2Axis1 + b1Axis1) * 0.5f;
183                        m_twistAxis.normalize();
184                        m_twistAxis *= -1.0f;
185
[2908]186                        m_kTwist = btScalar(1.) / (getRigidBodyA().computeAngularImpulseDenominator(m_twistAxis) +
187                                getRigidBodyB().computeAngularImpulseDenominator(m_twistAxis));
[1963]188
[2908]189                }       else
190                        if (twist >  m_twistSpan*lockedFreeFactor)
[1963]191                        {
[2908]192                                m_twistCorrection = (twist - m_twistSpan);
193                                m_solveTwistLimit = true;
[1963]194
[2908]195                                m_twistAxis = (b2Axis1 + b1Axis1) * 0.5f;
196                                m_twistAxis.normalize();
[1963]197
[2908]198                                m_kTwist = btScalar(1.) / (getRigidBodyA().computeAngularImpulseDenominator(m_twistAxis) +
199                                        getRigidBodyB().computeAngularImpulseDenominator(m_twistAxis));
[1963]200
201                        }
[2908]202        }
203}
[1963]204
[2908]205void    btConeTwistConstraint::solveConstraint(btScalar timeStep)
206{
[1963]207
[2908]208        btVector3 pivotAInW = m_rbA.getCenterOfMassTransform()*m_rbAFrame.getOrigin();
209        btVector3 pivotBInW = m_rbB.getCenterOfMassTransform()*m_rbBFrame.getOrigin();
[1963]210
[2908]211        btScalar tau = btScalar(0.3);
[1963]212
[2908]213        //linear part
214        if (!m_angularOnly)
215        {
216                btVector3 rel_pos1 = pivotAInW - m_rbA.getCenterOfMassPosition(); 
217                btVector3 rel_pos2 = pivotBInW - m_rbB.getCenterOfMassPosition();
[1963]218
[2908]219                btVector3 vel1 = m_rbA.getVelocityInLocalPoint(rel_pos1);
220                btVector3 vel2 = m_rbB.getVelocityInLocalPoint(rel_pos2);
221                btVector3 vel = vel1 - vel2;
[1963]222
[2908]223                for (int i=0;i<3;i++)
224                {               
225                        const btVector3& normal = m_jac[i].m_linearJointAxis;
226                        btScalar jacDiagABInv = btScalar(1.) / m_jac[i].getDiagonal();
[1963]227
[2908]228                        btScalar rel_vel;
229                        rel_vel = normal.dot(vel);
230                        //positional error (zeroth order error)
231                        btScalar depth = -(pivotAInW - pivotBInW).dot(normal); //this is the error projected on the normal
232                        btScalar impulse = depth*tau/timeStep  * jacDiagABInv -  rel_vel * jacDiagABInv;
233                        m_appliedImpulse += impulse;
234                        btVector3 impulse_vector = normal * impulse;
235                        m_rbA.applyImpulse(impulse_vector, pivotAInW - m_rbA.getCenterOfMassPosition());
236                        m_rbB.applyImpulse(-impulse_vector, pivotBInW - m_rbB.getCenterOfMassPosition());
[1963]237                }
238        }
[2908]239       
240        {
241                ///solve angular part
242                const btVector3& angVelA = getRigidBodyA().getAngularVelocity();
243                const btVector3& angVelB = getRigidBodyB().getAngularVelocity();
[2907]244
[2908]245                // solve swing limit
246                if (m_solveSwingLimit)
[1963]247                {
[2908]248                        btScalar amplitude = ((angVelB - angVelA).dot( m_swingAxis )*m_relaxationFactor*m_relaxationFactor + m_swingCorrection*(btScalar(1.)/timeStep)*m_biasFactor);
249                        btScalar impulseMag = amplitude * m_kSwing;
[1963]250
[2908]251                        // Clamp the accumulated impulse
252                        btScalar temp = m_accSwingLimitImpulse;
253                        m_accSwingLimitImpulse = btMax(m_accSwingLimitImpulse + impulseMag, btScalar(0.0) );
254                        impulseMag = m_accSwingLimitImpulse - temp;
[1963]255
[2908]256                        btVector3 impulse = m_swingAxis * impulseMag;
[1963]257
[2908]258                        m_rbA.applyTorqueImpulse(impulse);
259                        m_rbB.applyTorqueImpulse(-impulse);
[1963]260
261                }
262
[2908]263                // solve twist limit
264                if (m_solveTwistLimit)
[1963]265                {
[2908]266                        btScalar amplitude = ((angVelB - angVelA).dot( m_twistAxis )*m_relaxationFactor*m_relaxationFactor + m_twistCorrection*(btScalar(1.)/timeStep)*m_biasFactor );
267                        btScalar impulseMag = amplitude * m_kTwist;
[1963]268
[2908]269                        // Clamp the accumulated impulse
270                        btScalar temp = m_accTwistLimitImpulse;
271                        m_accTwistLimitImpulse = btMax(m_accTwistLimitImpulse + impulseMag, btScalar(0.0) );
272                        impulseMag = m_accTwistLimitImpulse - temp;
[1963]273
[2908]274                        btVector3 impulse = m_twistAxis * impulseMag;
[1963]275
[2908]276                        m_rbA.applyTorqueImpulse(impulse);
277                        m_rbB.applyTorqueImpulse(-impulse);
[1963]278
[2908]279                }
280       
[1963]281        }
282
283}
284
[2908]285void    btConeTwistConstraint::updateRHS(btScalar       timeStep)
[1963]286{
[2908]287        (void)timeStep;
[1963]288
289}
Note: See TracBrowser for help on using the repository browser.