mirror of
https://github.com/bulletphysics/bullet3
synced 2025-01-05 23:31:06 +00:00
dba239fe8d
A fileIO plugin can override custom file IO operations. As a small test, load files from a zipfile in memory. Default fileIO implementation is in examples/Utils/b3BulletDefaultFileIO.h Affects URDF, SDF, MJCF, Wavefront OBJ, STL, DAE, images.
363 lines
12 KiB
C++
363 lines
12 KiB
C++
/*
|
|
Bullet Continuous Collision Detection and Physics Library
|
|
Copyright (c) 2015 Google Inc. http://bulletphysics.org
|
|
|
|
This software is provided 'as-is', without any express or implied warranty.
|
|
In no event will the authors be held liable for any damages arising from the use of this software.
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
including commercial applications, and to alter it and redistribute it freely,
|
|
subject to the following restrictions:
|
|
|
|
1. 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.
|
|
2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
*/
|
|
#include "InverseDynamicsExample.h"
|
|
|
|
#include "BulletDynamics/Featherstone/btMultiBodyLinkCollider.h"
|
|
#include "Bullet3Common/b3FileUtils.h"
|
|
#include "BulletDynamics/Featherstone/btMultiBodyJointMotor.h"
|
|
#include "BulletDynamics/Featherstone/btMultiBodyDynamicsWorld.h"
|
|
#include "../CommonInterfaces/CommonParameterInterface.h"
|
|
#include "../Utils/b3ResourcePath.h"
|
|
#include "../Importers/ImportURDFDemo/BulletUrdfImporter.h"
|
|
#include "../Importers/ImportURDFDemo/URDF2Bullet.h"
|
|
#include "../Importers/ImportURDFDemo/MyMultiBodyCreator.h"
|
|
|
|
#include "../CommonInterfaces/CommonMultiBodyBase.h"
|
|
|
|
#include "btBulletDynamicsCommon.h"
|
|
|
|
#include "LinearMath/btVector3.h"
|
|
#include "LinearMath/btAlignedObjectArray.h"
|
|
|
|
#include "../CommonInterfaces/CommonRigidBodyBase.h"
|
|
|
|
#include "BulletInverseDynamics/IDConfig.hpp"
|
|
#include "../Extras/InverseDynamics/btMultiBodyTreeCreator.hpp"
|
|
|
|
#include "../RenderingExamples/TimeSeriesCanvas.h"
|
|
|
|
#include <vector>
|
|
|
|
// the UI interface makes it easier to use static variables & free functions
|
|
// as parameters and callbacks
|
|
static btScalar kp = 10 * 10;
|
|
static btScalar kd = 2 * 10;
|
|
static bool useInverseModel = true;
|
|
static std::vector<btScalar> qd;
|
|
static std::vector<std::string> qd_name;
|
|
static std::vector<std::string> q_name;
|
|
|
|
static btVector4 sJointCurveColors[8] =
|
|
{
|
|
btVector4(1, 0.3, 0.3, 1),
|
|
btVector4(0.3, 1, 0.3, 1),
|
|
btVector4(0.3, 0.3, 1, 1),
|
|
btVector4(0.3, 1, 1, 1),
|
|
btVector4(1, 0.3, 1, 1),
|
|
btVector4(1, 1, 0.3, 1),
|
|
btVector4(1, 0.7, 0.7, 1),
|
|
btVector4(0.7, 1, 1, 1),
|
|
|
|
};
|
|
|
|
void toggleUseInverseModel(int buttonId, bool buttonState, void* userPointer)
|
|
{
|
|
useInverseModel = !useInverseModel;
|
|
// todo(thomas) is there a way to get a toggle button with changing text?
|
|
b3Printf("switched inverse model %s", useInverseModel ? "on" : "off");
|
|
}
|
|
|
|
class InverseDynamicsExample : public CommonMultiBodyBase
|
|
{
|
|
btInverseDynamicsExampleOptions m_option;
|
|
btMultiBody* m_multiBody;
|
|
btInverseDynamics::MultiBodyTree* m_inverseModel;
|
|
TimeSeriesCanvas* m_timeSeriesCanvas;
|
|
|
|
public:
|
|
InverseDynamicsExample(struct GUIHelperInterface* helper, btInverseDynamicsExampleOptions option);
|
|
virtual ~InverseDynamicsExample();
|
|
|
|
virtual void initPhysics();
|
|
virtual void stepSimulation(float deltaTime);
|
|
|
|
void setFileName(const char* urdfFileName);
|
|
|
|
virtual void resetCamera()
|
|
{
|
|
float dist = 1.5;
|
|
float pitch = -10;
|
|
float yaw = -80;
|
|
float targetPos[3] = {0, 0, 0};
|
|
m_guiHelper->resetCamera(dist, yaw, pitch, targetPos[0], targetPos[1], targetPos[2]);
|
|
}
|
|
};
|
|
|
|
InverseDynamicsExample::InverseDynamicsExample(struct GUIHelperInterface* helper, btInverseDynamicsExampleOptions option)
|
|
: CommonMultiBodyBase(helper),
|
|
m_option(option),
|
|
m_multiBody(0),
|
|
m_inverseModel(0),
|
|
m_timeSeriesCanvas(0)
|
|
{
|
|
}
|
|
|
|
InverseDynamicsExample::~InverseDynamicsExample()
|
|
{
|
|
delete m_inverseModel;
|
|
delete m_timeSeriesCanvas;
|
|
}
|
|
|
|
//todo(erwincoumans) Quick hack, reference to InvertedPendulumPDControl implementation. Will create a separate header/source file for this.
|
|
btMultiBody* createInvertedPendulumMultiBody(btMultiBodyDynamicsWorld* world, GUIHelperInterface* guiHelper, const btTransform& baseWorldTrans, bool fixedBase);
|
|
|
|
void InverseDynamicsExample::initPhysics()
|
|
{
|
|
//roboticists like Z up
|
|
int upAxis = 2;
|
|
m_guiHelper->setUpAxis(upAxis);
|
|
|
|
createEmptyDynamicsWorld();
|
|
btVector3 gravity(0, 0, 0);
|
|
// gravity[upAxis]=-9.8;
|
|
m_dynamicsWorld->setGravity(gravity);
|
|
|
|
m_guiHelper->createPhysicsDebugDrawer(m_dynamicsWorld);
|
|
|
|
{
|
|
SliderParams slider("Kp", &kp);
|
|
slider.m_minVal = 0;
|
|
slider.m_maxVal = 2000;
|
|
if (m_guiHelper->getParameterInterface())
|
|
m_guiHelper->getParameterInterface()->registerSliderFloatParameter(slider);
|
|
}
|
|
{
|
|
SliderParams slider("Kd", &kd);
|
|
slider.m_minVal = 0;
|
|
slider.m_maxVal = 50;
|
|
if (m_guiHelper->getParameterInterface())
|
|
m_guiHelper->getParameterInterface()->registerSliderFloatParameter(slider);
|
|
}
|
|
|
|
if (m_option == BT_ID_PROGRAMMATICALLY)
|
|
{
|
|
ButtonParams button("toggle inverse model", 0, true);
|
|
button.m_callback = toggleUseInverseModel;
|
|
m_guiHelper->getParameterInterface()->registerButtonParameter(button);
|
|
}
|
|
|
|
switch (m_option)
|
|
{
|
|
case BT_ID_LOAD_URDF:
|
|
{
|
|
BulletURDFImporter u2b(m_guiHelper, 0, 0, 1, 0);
|
|
bool loadOk = u2b.loadURDF("kuka_iiwa/model.urdf"); // lwr / kuka.urdf");
|
|
if (loadOk)
|
|
{
|
|
int rootLinkIndex = u2b.getRootLinkIndex();
|
|
b3Printf("urdf root link index = %d\n", rootLinkIndex);
|
|
MyMultiBodyCreator creation(m_guiHelper);
|
|
btTransform identityTrans;
|
|
identityTrans.setIdentity();
|
|
ConvertURDF2Bullet(u2b, creation, identityTrans, m_dynamicsWorld, true, u2b.getPathPrefix());
|
|
for (int i = 0; i < u2b.getNumAllocatedCollisionShapes(); i++)
|
|
{
|
|
m_collisionShapes.push_back(u2b.getAllocatedCollisionShape(i));
|
|
}
|
|
m_multiBody = creation.getBulletMultiBody();
|
|
if (m_multiBody)
|
|
{
|
|
//kuka without joint control/constraints will gain energy explode soon due to timestep/integrator
|
|
//temporarily set some extreme damping factors until we have some joint control or constraints
|
|
m_multiBody->setAngularDamping(0 * 0.99);
|
|
m_multiBody->setLinearDamping(0 * 0.99);
|
|
b3Printf("Root link name = %s", u2b.getLinkName(u2b.getRootLinkIndex()).c_str());
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case BT_ID_PROGRAMMATICALLY:
|
|
{
|
|
btTransform baseWorldTrans;
|
|
baseWorldTrans.setIdentity();
|
|
m_multiBody = createInvertedPendulumMultiBody(m_dynamicsWorld, m_guiHelper, baseWorldTrans, false);
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
b3Error("Unknown option in InverseDynamicsExample::initPhysics");
|
|
b3Assert(0);
|
|
}
|
|
};
|
|
|
|
if (m_multiBody)
|
|
{
|
|
{
|
|
if (m_guiHelper->getAppInterface() && m_guiHelper->getParameterInterface())
|
|
{
|
|
m_timeSeriesCanvas = new TimeSeriesCanvas(m_guiHelper->getAppInterface()->m_2dCanvasInterface, 512, 230, "Joint Space Trajectory");
|
|
m_timeSeriesCanvas->setupTimeSeries(3, 100, 0);
|
|
}
|
|
}
|
|
|
|
// construct inverse model
|
|
btInverseDynamics::btMultiBodyTreeCreator id_creator;
|
|
if (-1 == id_creator.createFromBtMultiBody(m_multiBody, false))
|
|
{
|
|
b3Error("error creating tree\n");
|
|
}
|
|
else
|
|
{
|
|
m_inverseModel = btInverseDynamics::CreateMultiBodyTree(id_creator);
|
|
}
|
|
// add joint target controls
|
|
qd.resize(m_multiBody->getNumDofs());
|
|
|
|
qd_name.resize(m_multiBody->getNumDofs());
|
|
q_name.resize(m_multiBody->getNumDofs());
|
|
|
|
if (m_timeSeriesCanvas && m_guiHelper->getParameterInterface())
|
|
{
|
|
for (std::size_t dof = 0; dof < qd.size(); dof++)
|
|
{
|
|
qd[dof] = 0;
|
|
char tmp[25];
|
|
sprintf(tmp, "q_desired[%lu]", dof);
|
|
qd_name[dof] = tmp;
|
|
SliderParams slider(qd_name[dof].c_str(), &qd[dof]);
|
|
slider.m_minVal = -3.14;
|
|
slider.m_maxVal = 3.14;
|
|
|
|
sprintf(tmp, "q[%lu]", dof);
|
|
q_name[dof] = tmp;
|
|
m_guiHelper->getParameterInterface()->registerSliderFloatParameter(slider);
|
|
btVector4 color = sJointCurveColors[dof & 7];
|
|
m_timeSeriesCanvas->addDataSource(q_name[dof].c_str(), color[0] * 255, color[1] * 255, color[2] * 255);
|
|
}
|
|
}
|
|
}
|
|
|
|
m_guiHelper->autogenerateGraphicsObjects(m_dynamicsWorld);
|
|
}
|
|
|
|
void InverseDynamicsExample::stepSimulation(float deltaTime)
|
|
{
|
|
if (m_multiBody)
|
|
{
|
|
const int num_dofs = m_multiBody->getNumDofs();
|
|
btInverseDynamics::vecx nu(num_dofs), qdot(num_dofs), q(num_dofs), joint_force(num_dofs);
|
|
btInverseDynamics::vecx pd_control(num_dofs);
|
|
|
|
// compute joint forces from one of two control laws:
|
|
// 1) "computed torque" control, which gives perfect, decoupled,
|
|
// linear second order error dynamics per dof in case of a
|
|
// perfect model and (and negligible time discretization effects)
|
|
// 2) decoupled PD control per joint, without a model
|
|
for (int dof = 0; dof < num_dofs; dof++)
|
|
{
|
|
q(dof) = m_multiBody->getJointPos(dof);
|
|
qdot(dof) = m_multiBody->getJointVel(dof);
|
|
|
|
const btScalar qd_dot = 0;
|
|
const btScalar qd_ddot = 0;
|
|
if (m_timeSeriesCanvas)
|
|
m_timeSeriesCanvas->insertDataAtCurrentTime(q[dof], dof, true);
|
|
|
|
// pd_control is either desired joint torque for pd control,
|
|
// or the feedback contribution to nu
|
|
pd_control(dof) = kd * (qd_dot - qdot(dof)) + kp * (qd[dof] - q(dof));
|
|
// nu is the desired joint acceleration for computed torque control
|
|
nu(dof) = qd_ddot + pd_control(dof);
|
|
}
|
|
if (useInverseModel)
|
|
{
|
|
// calculate joint forces corresponding to desired accelerations nu
|
|
if (m_multiBody->hasFixedBase())
|
|
{
|
|
if (-1 != m_inverseModel->calculateInverseDynamics(q, qdot, nu, &joint_force))
|
|
{
|
|
//joint_force(dof) += damping*dot_q(dof);
|
|
// use inverse model: apply joint force corresponding to
|
|
// desired acceleration nu
|
|
|
|
for (int dof = 0; dof < num_dofs; dof++)
|
|
{
|
|
m_multiBody->addJointTorque(dof, joint_force(dof));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//the inverse dynamics model represents the 6 DOFs of the base, unlike btMultiBody.
|
|
//append some dummy values to represent the 6 DOFs of the base
|
|
btInverseDynamics::vecx nu6(num_dofs + 6), qdot6(num_dofs + 6), q6(num_dofs + 6), joint_force6(num_dofs + 6);
|
|
for (int i = 0; i < num_dofs; i++)
|
|
{
|
|
nu6[6 + i] = nu[i];
|
|
qdot6[6 + i] = qdot[i];
|
|
q6[6 + i] = q[i];
|
|
joint_force6[6 + i] = joint_force[i];
|
|
}
|
|
if (-1 != m_inverseModel->calculateInverseDynamics(q6, qdot6, nu6, &joint_force6))
|
|
{
|
|
//joint_force(dof) += damping*dot_q(dof);
|
|
// use inverse model: apply joint force corresponding to
|
|
// desired acceleration nu
|
|
|
|
for (int dof = 0; dof < num_dofs; dof++)
|
|
{
|
|
m_multiBody->addJointTorque(dof, joint_force6(dof + 6));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (int dof = 0; dof < num_dofs; dof++)
|
|
{
|
|
// no model: just apply PD control law
|
|
m_multiBody->addJointTorque(dof, pd_control(dof));
|
|
}
|
|
}
|
|
}
|
|
|
|
if (m_timeSeriesCanvas)
|
|
m_timeSeriesCanvas->nextTick();
|
|
|
|
//todo: joint damping for btMultiBody, tune parameters
|
|
|
|
// step the simulation
|
|
if (m_dynamicsWorld)
|
|
{
|
|
// todo(thomas) check that this is correct:
|
|
// want to advance by 10ms, with 1ms timesteps
|
|
m_dynamicsWorld->stepSimulation(1e-3, 0); //,1e-3);
|
|
btAlignedObjectArray<btQuaternion> scratch_q;
|
|
btAlignedObjectArray<btVector3> scratch_m;
|
|
m_multiBody->forwardKinematics(scratch_q, scratch_m);
|
|
#if 0
|
|
for (int i = 0; i < m_multiBody->getNumLinks(); i++)
|
|
{
|
|
//btVector3 pos = m_multiBody->getLink(i).m_cachedWorldTransform.getOrigin();
|
|
btTransform tr = m_multiBody->getLink(i).m_cachedWorldTransform;
|
|
btVector3 pos = tr.getOrigin() - quatRotate(tr.getRotation(), m_multiBody->getLink(i).m_dVector);
|
|
btVector3 localAxis = m_multiBody->getLink(i).m_axes[0].m_topVec;
|
|
//printf("link %d: %f,%f,%f, local axis:%f,%f,%f\n", i, pos.x(), pos.y(), pos.z(), localAxis.x(), localAxis.y(), localAxis.z());
|
|
|
|
|
|
|
|
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
CommonExampleInterface* InverseDynamicsExampleCreateFunc(CommonExampleOptions& options)
|
|
{
|
|
return new InverseDynamicsExample(options.m_guiHelper, btInverseDynamicsExampleOptions(options.m_option));
|
|
}
|
|
|
|
B3_STANDALONE_EXAMPLE(InverseDynamicsExampleCreateFunc)
|