Bullet Collision Detection & Physics Library
btRaycastVehicle.cpp
Go to the documentation of this file.
1/*
2 * Copyright (c) 2005 Erwin Coumans https://bulletphysics.org
3 *
4 * Permission to use, copy, modify, distribute and sell this software
5 * and its documentation for any purpose is hereby granted without fee,
6 * provided that the above copyright notice appear in all copies.
7 * Erwin Coumans makes no representations about the suitability
8 * of this software for any purpose.
9 * It is provided "as is" without express or implied warranty.
10*/
11
13#include "btRaycastVehicle.h"
14
19#include "btVehicleRaycaster.h"
20#include "btWheelInfo.h"
21#include "LinearMath/btMinMax.h"
24
25#define ROLLING_INFLUENCE_FIX
26
28{
29 static btRigidBody s_fixed(0, 0, 0);
30 s_fixed.setMassProps(btScalar(0.), btVector3(btScalar(0.), btScalar(0.), btScalar(0.)));
31 return s_fixed;
32}
33
35 : m_vehicleRaycaster(raycaster),
36 m_pitchControl(btScalar(0.))
37{
40 m_indexUpAxis = 2;
43}
44
46{
47 (void)tuning;
50}
51
53{
54}
55
56//
57// basically most of the code is general for 2 or 4 wheel vehicles, but some of it needs to be reviewed
58//
60{
62
64 ci.m_wheelDirectionCS = wheelDirectionCS0;
65 ci.m_wheelAxleCS = wheelAxleCS;
66 ci.m_suspensionRestLength = suspensionRestLength;
67 ci.m_wheelRadius = wheelRadius;
68 ci.m_suspensionStiffness = tuning.m_suspensionStiffness;
69 ci.m_wheelsDampingCompression = tuning.m_suspensionCompression;
70 ci.m_wheelsDampingRelaxation = tuning.m_suspensionDamping;
71 ci.m_frictionSlip = tuning.m_frictionSlip;
72 ci.m_bIsFrontWheel = isFrontWheel;
73 ci.m_maxSuspensionTravelCm = tuning.m_maxSuspensionTravelCm;
74 ci.m_maxSuspensionForce = tuning.m_maxSuspensionForce;
75
77
79
82 return wheel;
83}
84
86{
89 return wheel.m_worldTransform;
90}
91
93{
96 btVector3 up = -wheel.m_raycastInfo.m_wheelDirectionWS;
97 const btVector3& right = wheel.m_raycastInfo.m_wheelAxleWS;
99 fwd = fwd.normalize();
100 // up = right.cross(fwd);
101 // up.normalize();
102
103 //rotate around steering over de wheelAxleWS
104 btScalar steering = wheel.m_steering;
105
106 btQuaternion steeringOrn(up, steering); //wheel.m_steering);
108
109 btQuaternion rotatingOrn(right, -wheel.m_rotation);
111
113 basis2[0][m_indexRightAxis] = -right[0];
114 basis2[1][m_indexRightAxis] = -right[1];
115 basis2[2][m_indexRightAxis] = -right[2];
116
117 basis2[0][m_indexUpAxis] = up[0];
118 basis2[1][m_indexUpAxis] = up[1];
119 basis2[2][m_indexUpAxis] = up[2];
120
124
125 wheel.m_worldTransform.setBasis(steeringMat * rotatingMat * basis2);
126 wheel.m_worldTransform.setOrigin(
127 wheel.m_raycastInfo.m_hardPointWS + wheel.m_raycastInfo.m_wheelDirectionWS * wheel.m_raycastInfo.m_suspensionLength);
128}
129
131{
132 int i;
133 for (i = 0; i < m_wheelInfo.size(); i++)
134 {
136 wheel.m_raycastInfo.m_suspensionLength = wheel.getSuspensionRestLength();
137 wheel.m_suspensionRelativeVelocity = btScalar(0.0);
138
139 wheel.m_raycastInfo.m_contactNormalWS = -wheel.m_raycastInfo.m_wheelDirectionWS;
140 //wheel_info.setContactFriction(btScalar(0.0));
141 wheel.m_clippedInvContactDotSuspension = btScalar(1.0);
142 }
143}
144
146{
147 wheel.m_raycastInfo.m_isInContact = false;
148
150 if (interpolatedTransform && (getRigidBody()->getMotionState()))
151 {
153 }
154
155 wheel.m_raycastInfo.m_hardPointWS = chassisTrans(wheel.m_chassisConnectionPointCS);
156 wheel.m_raycastInfo.m_wheelDirectionWS = chassisTrans.getBasis() * wheel.m_wheelDirectionCS;
157 wheel.m_raycastInfo.m_wheelAxleWS = chassisTrans.getBasis() * wheel.m_wheelAxleCS;
158}
159
161{
163
164 btScalar depth = -1;
165
166 btScalar raylen = wheel.getSuspensionRestLength() + wheel.m_wheelsRadius;
167
168 btVector3 rayvector = wheel.m_raycastInfo.m_wheelDirectionWS * (raylen);
169 const btVector3& source = wheel.m_raycastInfo.m_hardPointWS;
170 wheel.m_raycastInfo.m_contactPointWS = source + rayvector;
171 const btVector3& target = wheel.m_raycastInfo.m_contactPointWS;
172
173 btScalar param = btScalar(0.);
174
176
178
179 void* object = m_vehicleRaycaster->castRay(source, target, rayResults);
180
181 wheel.m_raycastInfo.m_groundObject = 0;
182
183 if (object)
184 {
185 param = rayResults.m_distFraction;
186 depth = raylen * rayResults.m_distFraction;
187 wheel.m_raycastInfo.m_contactNormalWS = rayResults.m_hitNormalInWorld;
188 wheel.m_raycastInfo.m_isInContact = true;
189
190 wheel.m_raycastInfo.m_groundObject = &getFixedBody();
191 //wheel.m_raycastInfo.m_groundObject = object;
192
193 btScalar hitDistance = param * raylen;
194 wheel.m_raycastInfo.m_suspensionLength = hitDistance - wheel.m_wheelsRadius;
195 //clamp on max suspension travel
196
197 btScalar minSuspensionLength = wheel.getSuspensionRestLength() - wheel.m_maxSuspensionTravelCm * btScalar(0.01);
198 btScalar maxSuspensionLength = wheel.getSuspensionRestLength() + wheel.m_maxSuspensionTravelCm * btScalar(0.01);
199 if (wheel.m_raycastInfo.m_suspensionLength < minSuspensionLength)
200 {
201 wheel.m_raycastInfo.m_suspensionLength = minSuspensionLength;
202 }
203 if (wheel.m_raycastInfo.m_suspensionLength > maxSuspensionLength)
204 {
205 wheel.m_raycastInfo.m_suspensionLength = maxSuspensionLength;
206 }
207
208 wheel.m_raycastInfo.m_contactPointWS = rayResults.m_hitPointInWorld;
209
210 btScalar denominator = wheel.m_raycastInfo.m_contactNormalWS.dot(wheel.m_raycastInfo.m_wheelDirectionWS);
211
213 btVector3 relpos = wheel.m_raycastInfo.m_contactPointWS - getRigidBody()->getCenterOfMassPosition();
214
216
217 btScalar projVel = wheel.m_raycastInfo.m_contactNormalWS.dot(chassis_velocity_at_contactPoint);
218
219 if (denominator >= btScalar(-0.1))
220 {
221 wheel.m_suspensionRelativeVelocity = btScalar(0.0);
222 wheel.m_clippedInvContactDotSuspension = btScalar(1.0) / btScalar(0.1);
223 }
224 else
225 {
226 btScalar inv = btScalar(-1.) / denominator;
227 wheel.m_suspensionRelativeVelocity = projVel * inv;
228 wheel.m_clippedInvContactDotSuspension = inv;
229 }
230 }
231 else
232 {
233 //put wheel info as in rest position
234 wheel.m_raycastInfo.m_suspensionLength = wheel.getSuspensionRestLength();
235 wheel.m_suspensionRelativeVelocity = btScalar(0.0);
236 wheel.m_raycastInfo.m_contactNormalWS = -wheel.m_raycastInfo.m_wheelDirectionWS;
237 wheel.m_clippedInvContactDotSuspension = btScalar(1.0);
238 }
239
240 return depth;
241}
242
244{
245 /*if (getRigidBody()->getMotionState())
246 {
247 btTransform chassisWorldTrans;
248 getRigidBody()->getMotionState()->getWorldTransform(chassisWorldTrans);
249 return chassisWorldTrans;
250 }
251 */
252
254}
255
257{
258 {
259 for (int i = 0; i < getNumWheels(); i++)
260 {
261 updateWheelTransform(i, false);
262 }
263 }
264
266
268
270 chassisTrans.getBasis()[0][m_indexForwardAxis],
271 chassisTrans.getBasis()[1][m_indexForwardAxis],
272 chassisTrans.getBasis()[2][m_indexForwardAxis]);
273
274 if (forwardW.dot(getRigidBody()->getLinearVelocity()) < btScalar(0.))
275 {
277 }
278
279 //
280 // simulate suspension
281 //
282
283 int i = 0;
284 for (i = 0; i < m_wheelInfo.size(); i++)
285 {
286 //btScalar depth;
287 //depth =
289 }
290
292
293 for (i = 0; i < m_wheelInfo.size(); i++)
294 {
295 //apply suspension force
297
298 btScalar suspensionForce = wheel.m_wheelsSuspensionForce;
299
300 if (suspensionForce > wheel.m_maxSuspensionForce)
301 {
302 suspensionForce = wheel.m_maxSuspensionForce;
303 }
304 btVector3 impulse = wheel.m_raycastInfo.m_contactNormalWS * suspensionForce * step;
305 btVector3 relpos = wheel.m_raycastInfo.m_contactPointWS - getRigidBody()->getCenterOfMassPosition();
306
308 }
309
311
312 for (i = 0; i < m_wheelInfo.size(); i++)
313 {
315 btVector3 relpos = wheel.m_raycastInfo.m_hardPointWS - getRigidBody()->getCenterOfMassPosition();
317
318 if (wheel.m_raycastInfo.m_isInContact)
319 {
321
326
327 btScalar proj = fwd.dot(wheel.m_raycastInfo.m_contactNormalWS);
328 fwd -= wheel.m_raycastInfo.m_contactNormalWS * proj;
329
330 btScalar proj2 = fwd.dot(vel);
331
332 wheel.m_deltaRotation = (proj2 * step) / (wheel.m_wheelsRadius);
333 wheel.m_rotation += wheel.m_deltaRotation;
334 }
335 else
336 {
337 wheel.m_rotation += wheel.m_deltaRotation;
338 }
339
340 wheel.m_deltaRotation *= btScalar(0.99); //damping of rotation when not in contact
341 }
342}
343
345{
346 btAssert(wheel >= 0 && wheel < getNumWheels());
347
349 wheelInfo.m_steering = steering;
350}
351
353{
355}
356
358{
359 btAssert(wheel >= 0 && wheel < getNumWheels());
361 wheelInfo.m_engineForce = force;
362}
363
365{
366 btAssert((index >= 0) && (index < getNumWheels()));
367
368 return m_wheelInfo[index];
369}
370
372{
373 btAssert((index >= 0) && (index < getNumWheels()));
374
375 return m_wheelInfo[index];
376}
377
379{
382}
383
385{
387
389
390 for (int w_it = 0; w_it < getNumWheels(); w_it++)
391 {
393
394 if (wheel_info.m_raycastInfo.m_isInContact)
395 {
397 // Spring
398 {
399 btScalar susp_length = wheel_info.getSuspensionRestLength();
400 btScalar current_length = wheel_info.m_raycastInfo.m_suspensionLength;
401
403
404 force = wheel_info.m_suspensionStiffness * length_diff * wheel_info.m_clippedInvContactDotSuspension;
405 }
406
407 // Damper
408 {
409 btScalar projected_rel_vel = wheel_info.m_suspensionRelativeVelocity;
410 {
412 if (projected_rel_vel < btScalar(0.0))
413 {
414 susp_damping = wheel_info.m_wheelsDampingCompression;
415 }
416 else
417 {
418 susp_damping = wheel_info.m_wheelsDampingRelaxation;
419 }
421 }
422 }
423
424 // RESULT
425 wheel_info.m_wheelsSuspensionForce = force * chassisMass;
426 if (wheel_info.m_wheelsSuspensionForce < btScalar(0.))
427 {
428 wheel_info.m_wheelsSuspensionForce = btScalar(0.);
429 }
430 }
431 else
432 {
433 wheel_info.m_wheelsSuspensionForce = btScalar(0.0);
434 }
435 }
436}
437
439{
446
448 : m_body0(body0),
449 m_body1(body1),
453 {
454 btScalar denom0 = body0->computeImpulseDenominator(frictionPosWorld, frictionDirectionWorld);
455 btScalar denom1 = body1->computeImpulseDenominator(frictionPosWorld, frictionDirectionWorld);
456 btScalar relaxation = 1.f;
458 }
459};
460
463{
464 btScalar j1 = 0.f;
465
466 const btVector3& contactPosWorld = contactPoint.m_frictionPositionWorld;
467
468 btVector3 rel_pos1 = contactPosWorld - contactPoint.m_body0->getCenterOfMassPosition();
469 btVector3 rel_pos2 = contactPosWorld - contactPoint.m_body1->getCenterOfMassPosition();
470
471 btScalar maxImpulse = contactPoint.m_maxImpulse;
472
473 btVector3 vel1 = contactPoint.m_body0->getVelocityInLocalPoint(rel_pos1);
474 btVector3 vel2 = contactPoint.m_body1->getVelocityInLocalPoint(rel_pos2);
476
477 btScalar vrel = contactPoint.m_frictionDirectionWorld.dot(vel);
478
479 // calculate j that moves us to zero relative velocity
480 j1 = -vrel * contactPoint.m_jacDiagABInv / btScalar(numWheelsOnGround);
483
484 return j1;
485}
486
489{
490 //calculate the impulse, so that the wheels don't move sidewards
491 int numWheel = getNumWheels();
492 if (!numWheel)
493 return;
494
499
500 int numWheelsOnGround = 0;
501
502 //collapse all those loops into one!
503 for (int i = 0; i < getNumWheels(); i++)
504 {
506 class btRigidBody* groundObject = (class btRigidBody*)wheelInfo.m_raycastInfo.m_groundObject;
507 if (groundObject)
509 m_sideImpulse[i] = btScalar(0.);
510 m_forwardImpulse[i] = btScalar(0.);
511 }
512
513 {
514 for (int i = 0; i < getNumWheels(); i++)
515 {
517
518 class btRigidBody* groundObject = (class btRigidBody*)wheelInfo.m_raycastInfo.m_groundObject;
519
520 if (groundObject)
521 {
523
525 m_axle[i] = -btVector3(
529
530 const btVector3& surfNormalWS = wheelInfo.m_raycastInfo.m_contactNormalWS;
531 btScalar proj = m_axle[i].dot(surfNormalWS);
532 m_axle[i] -= surfNormalWS * proj;
533 m_axle[i] = m_axle[i].normalize();
534
535 m_forwardWS[i] = surfNormalWS.cross(m_axle[i]);
536 m_forwardWS[i].normalize();
537
538 resolveSingleBilateral(*m_chassisBody, wheelInfo.m_raycastInfo.m_contactPointWS,
539 *groundObject, wheelInfo.m_raycastInfo.m_contactPointWS,
540 btScalar(0.), m_axle[i], m_sideImpulse[i], timeStep);
541
543 }
544 }
545 }
546
548 btScalar fwdFactor = 0.5;
549
550 bool sliding = false;
551 {
552 for (int wheel = 0; wheel < getNumWheels(); wheel++)
553 {
555 class btRigidBody* groundObject = (class btRigidBody*)wheelInfo.m_raycastInfo.m_groundObject;
556
558
559 if (groundObject)
560 {
561 if (wheelInfo.m_engineForce != 0.f)
562 {
563 rollingFriction = wheelInfo.m_engineForce * timeStep;
564 }
565 else
566 {
572 }
573 }
574
575 //switch between active rolling (throttle), braking and non-active rolling friction (no throttle/break)
576
578 m_wheelInfo[wheel].m_skidInfo = btScalar(1.);
579
580 if (groundObject)
581 {
582 m_wheelInfo[wheel].m_skidInfo = btScalar(1.);
583
584 btScalar maximp = wheelInfo.m_wheelsSuspensionForce * timeStep * wheelInfo.m_frictionSlip;
586
588
589 m_forwardImpulse[wheel] = rollingFriction; //wheelInfo.m_engineForce* timeStep;
590
593
594 btScalar impulseSquared = (x * x + y * y);
595
597 {
598 sliding = true;
599
601
602 m_wheelInfo[wheel].m_skidInfo *= factor;
603 }
604 }
605 }
606 }
607
608 if (sliding)
609 {
610 for (int wheel = 0; wheel < getNumWheels(); wheel++)
611 {
612 if (m_sideImpulse[wheel] != btScalar(0.))
613 {
614 if (m_wheelInfo[wheel].m_skidInfo < btScalar(1.))
615 {
616 m_forwardImpulse[wheel] *= m_wheelInfo[wheel].m_skidInfo;
617 m_sideImpulse[wheel] *= m_wheelInfo[wheel].m_skidInfo;
618 }
619 }
620 }
621 }
622
623 // apply the impulses
624 {
625 for (int wheel = 0; wheel < getNumWheels(); wheel++)
626 {
628
629 btVector3 rel_pos = wheelInfo.m_raycastInfo.m_contactPointWS -
631
632 if (m_forwardImpulse[wheel] != btScalar(0.))
633 {
635 }
636 if (m_sideImpulse[wheel] != btScalar(0.))
637 {
638 class btRigidBody* groundObject = (class btRigidBody*)m_wheelInfo[wheel].m_raycastInfo.m_groundObject;
639
640 btVector3 rel_pos2 = wheelInfo.m_raycastInfo.m_contactPointWS -
641 groundObject->getCenterOfMassPosition();
642
644
645#if defined ROLLING_INFLUENCE_FIX // fix. It only worked if car's up was along Y - VT.
647 rel_pos -= vChassisWorldUp * (vChassisWorldUp.dot(rel_pos) * (1.f - wheelInfo.m_rollInfluence));
648#else
649 rel_pos[m_indexUpAxis] *= wheelInfo.m_rollInfluence;
650#endif
652
653 //apply friction impulse on the ground
654 groundObject->applyImpulse(-sideImp, rel_pos2);
655 }
656 }
657 }
658}
659
661{
662 for (int v = 0; v < this->getNumWheels(); v++)
663 {
664 btVector3 wheelColor(0, 1, 1);
665 if (getWheelInfo(v).m_raycastInfo.m_isInContact)
666 {
667 wheelColor.setValue(0, 0, 1);
668 }
669 else
670 {
671 wheelColor.setValue(1, 0, 1);
672 }
673
675
677 getWheelInfo(v).m_worldTransform.getBasis()[0][getRightAxis()],
678 getWheelInfo(v).m_worldTransform.getBasis()[1][getRightAxis()],
679 getWheelInfo(v).m_worldTransform.getBasis()[2][getRightAxis()]);
680
681 //debug wheels (cylinders)
683 debugDrawer->drawLine(wheelPosWS, getWheelInfo(v).m_raycastInfo.m_contactPointWS, wheelColor);
684 }
685}
686
688{
689 // RayResultCallback& resultCallback;
690
692
694
695 if (rayCallback.hasHit())
696 {
697 const btRigidBody* body = btRigidBody::upcast(rayCallback.m_collisionObject);
698 if (body && body->hasContactResponse())
699 {
700 result.m_hitPointInWorld = rayCallback.m_hitPointWorld;
701 result.m_hitNormalInWorld = rayCallback.m_hitNormalWorld;
702 result.m_hitNormalInWorld.normalize();
703 result.m_distFraction = rayCallback.m_closestHitFraction;
704 return (void*)body;
705 }
706 }
707 return 0;
708}
void resolveSingleBilateral(btRigidBody &body1, const btVector3 &pos1, btRigidBody &body2, const btVector3 &pos2, btScalar distance, const btVector3 &normal, btScalar &impulse, btScalar timeStep)
resolveSingleBilateral is an obsolete methods used for vehicle friction between two dynamic objects
void btSetMin(T &a, const T &b)
Definition btMinMax.h:39
const T & btMax(const T &a, const T &b)
Definition btMinMax.h:27
void btSetMax(T &a, const T &b)
Definition btMinMax.h:48
btScalar calcRollingFriction(btWheelContactPoint &contactPoint, int numWheelsOnGround)
btScalar sideFrictionStiffness2
float btScalar
The btScalar type abstracts floating point numbers, to easily switch between double and single floati...
Definition btScalar.h:314
btScalar btSqrt(btScalar y)
Definition btScalar.h:466
#define btAssert(x)
Definition btScalar.h:153
static btRigidBody & getFixedBody()
int size() const
return the number of elements in the array
void resize(int newsize, const T &fillData=T())
void push_back(const T &_Val)
bool hasContactResponse() const
virtual void rayTest(const btVector3 &rayFromWorld, const btVector3 &rayToWorld, RayResultCallback &resultCallback) const
rayTest performs a raycast on all objects in the btCollisionWorld, and calls the resultCallback This ...
virtual void * castRay(const btVector3 &from, const btVector3 &to, btVehicleRaycasterResult &result)
btDynamicsWorld * m_dynamicsWorld
The btIDebugDraw interface class allows hooking up a debug renderer to visually debug simulations.
The btMatrix3x3 class implements a 3x3 rotation matrix, to perform linear algebra in combination with...
Definition btMatrix3x3.h:50
btVector3 getColumn(int i) const
Get a column of the matrix as a vector.
virtual void getWorldTransform(btTransform &worldTrans) const =0
The btQuaternion implements quaternion to perform linear algebra rotations in combination with btMatr...
btAlignedObjectArray< btVector3 > m_forwardWS
virtual void updateFriction(btScalar timeStep)
void defaultInit(const btVehicleTuning &tuning)
void updateWheelTransformsWS(btWheelInfo &wheel, bool interpolatedTransform=true)
btRigidBody * getRigidBody()
btAlignedObjectArray< btScalar > m_sideImpulse
const btTransform & getWheelTransformWS(int wheelIndex) const
int getNumWheels() const
btScalar m_currentVehicleSpeedKmHour
btScalar rayCast(btWheelInfo &wheel)
btScalar getSteeringValue(int wheel) const
void setBrake(btScalar brake, int wheelIndex)
void updateSuspension(btScalar deltaTime)
btAlignedObjectArray< btVector3 > m_axle
btRaycastVehicle(const btVehicleTuning &tuning, btRigidBody *chassis, btVehicleRaycaster *raycaster)
int getRightAxis() const
btVehicleRaycaster * m_vehicleRaycaster
virtual void updateVehicle(btScalar step)
btAlignedObjectArray< btScalar > m_forwardImpulse
btAlignedObjectArray< btWheelInfo > m_wheelInfo
btRigidBody * m_chassisBody
void applyEngineForce(btScalar force, int wheel)
btWheelInfo & addWheel(const btVector3 &connectionPointCS0, const btVector3 &wheelDirectionCS0, const btVector3 &wheelAxleCS, btScalar suspensionRestLength, btScalar wheelRadius, const btVehicleTuning &tuning, bool isFrontWheel)
void debugDraw(btIDebugDraw *debugDrawer)
btActionInterface interface
const btTransform & getChassisWorldTransform() const
void updateWheelTransform(int wheelIndex, bool interpolatedTransform=true)
const btWheelInfo & getWheelInfo(int index) const
void setSteeringValue(btScalar steering, int wheel)
The btRigidBody is the main class for rigid body objects.
Definition btRigidBody.h:60
btVector3 getVelocityInLocalPoint(const btVector3 &rel_pos) const
btScalar getInvMass() const
const btTransform & getCenterOfMassTransform() const
void applyImpulse(const btVector3 &impulse, const btVector3 &rel_pos)
btMotionState * getMotionState()
const btVector3 & getCenterOfMassPosition() const
static const btRigidBody * upcast(const btCollisionObject *colObj)
to keep collision detection and dynamics separate we don't store a rigidbody pointer but a rigidbody ...
const btVector3 & getLinearVelocity() const
The btTransform class supports rigid transforms with only translation and rotation and no scaling/she...
Definition btTransform.h:30
btMatrix3x3 & getBasis()
Return the basis matrix for the rotation.
btVector3 & getOrigin()
Return the origin vector translation.
btVector3 can be used to represent 3D points and vectors.
Definition btVector3.h:82
btScalar length() const
Return the length of the vector.
Definition btVector3.h:257
btVector3 cross(const btVector3 &v) const
Return the cross product between this and another vector.
Definition btVector3.h:380
btScalar dot(const btVector3 &v) const
Return the dot product.
Definition btVector3.h:229
btVector3 & normalize()
Normalize this vector x^2 + y^2 + z^2 = 1.
Definition btVector3.h:303
btVehicleRaycaster is provides interface for between vehicle simulation and raycasting
virtual void * castRay(const btVector3 &from, const btVector3 &to, btVehicleRaycasterResult &result)=0
btWheelContactPoint(btRigidBody *body0, btRigidBody *body1, const btVector3 &frictionPosWorld, const btVector3 &frictionDirectionWorld, btScalar maxImpulse)
btWheelInfo contains information per wheel about friction and suspension.
Definition btWheelInfo.h:38
btScalar m_steering
Definition btWheelInfo.h:67
btScalar m_brake
Definition btWheelInfo.h:75
btTransform m_worldTransform
Definition btWheelInfo.h:54