-
Notifications
You must be signed in to change notification settings - Fork 342
Link::AddForceInInertialFrame APIs added #2816
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Changes from 24 commits
024e0d6
94ad125
f2cd174
9869266
9660d0c
a4d3ee3
2aa9794
d0d498b
f7c09b7
1282177
03cbfba
334aea2
ee1a690
b107a65
85bdea9
c5ceda6
e9ba7cc
53b8b28
e7eb0ed
f77af75
fcbe12d
6d5067f
54e4c5d
06f6ee8
3371d9f
431d151
eb0233d
10e1b9d
352cffc
3b4180c
6c228b0
cb4eea4
0a9e696
9aa3d10
cb4cf99
d5e24a1
a38852c
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -486,6 +486,69 @@ void Link::AddWorldForce(EntityComponentManager &_ecm, | |
this->AddWorldWrench(_ecm, _force, torque); | ||
} | ||
|
||
////////////////////////////////////////////////// | ||
void Link::AddForceInInertialFrame(EntityComponentManager &_ecm, | ||
const math::Vector3d &_force) const | ||
{ | ||
auto inertial = _ecm.Component<components::Inertial>(this->dataPtr->id); | ||
auto worldPose = _ecm.ComponentData<components::WorldPose>(this->dataPtr->id) | ||
.value_or(sim::worldPose(this->dataPtr->id, _ecm)); | ||
|
||
// Can't apply force if the inertial's pose is not found | ||
if (!inertial) | ||
return; | ||
|
||
// The force is expressed in terms of the link's inertial coordinate frame, | ||
// We'll first convert this to force expressed | ||
//in terms of link's coordinate frame | ||
math::Vector3d linkForce = inertial->Data().Pose().Rot( | ||
).RotateVector(_force); | ||
|
||
// ExternalWorldForcecmd applies the force expressed in world coordinates | ||
Vedant87 marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
// so we need to compute the force expressed in world coordinates | ||
math::Vector3d worldForce = worldPose.Rot().RotateVector(linkForce); | ||
|
||
// Apply Force using AddWorldForce method | ||
this->AddWorldForce(_ecm, worldForce); | ||
} | ||
|
||
////////////////////////////////////////////////// | ||
void Link::AddForceInInertialFrame(EntityComponentManager &_ecm, | ||
const math::Vector3d &_force, | ||
const math::Vector3d &_position) const | ||
{ | ||
auto inertial = _ecm.Component<components::Inertial>(this->dataPtr->id); | ||
auto worldPose = _ecm.ComponentData<components::WorldPose>(this->dataPtr->id) | ||
.value_or(sim::worldPose(this->dataPtr->id, _ecm)); | ||
|
||
// Can't apply force if the inertial's pose is not found | ||
if (!inertial) | ||
return; | ||
|
||
// The force is expressed in terms of the link's inertial coordinate frame, | ||
// We'll first convert this to force expressed | ||
//in terms of link's coordinate frame | ||
Vedant87 marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
math::Vector3d linkForce = | ||
inertial->Data().Pose().Rot().RotateVector(_force); | ||
|
||
// ExternalWorldForcecmd applies the force expressed in world coordinates | ||
Vedant87 marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
// so we need to compute the force expressed in world coordinates | ||
math::Vector3d worldForce = worldPose.Rot().RotateVector(linkForce); | ||
|
||
// ExternalWorldForcecmd applies the force at a position expressed in terms | ||
// of link's coordinate frame.So we compute the position expressed in terms | ||
// of link's coordinate frame. | ||
Vedant87 marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
math::Pose3d forceApplicationRelativeToInertialFrame( | ||
_position, | ||
math::Quaterniond::Identity); | ||
math::Vector3d positionInLinkFrame = | ||
(inertial->Data().Pose() * | ||
forceApplicationRelativeToInertialFrame).Pos(); | ||
|
||
|
||
Vedant87 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
// Apply Force using AddWorldForce method | ||
this->AddWorldForce(_ecm, worldForce, positionInLinkFrame); | ||
} | ||
|
||
////////////////////////////////////////////////// | ||
void Link::AddWorldWrench(EntityComponentManager &_ecm, | ||
const math::Vector3d &_force, | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -698,3 +698,240 @@ TEST_F(LinkIntegrationTest, LinkAddWorldForce) | |
EXPECT_EQ(math::Vector3d::Zero, math::Vector3d( | ||
wrenchMsg.torque().x(), wrenchMsg.torque().y(), wrenchMsg.torque().z())); | ||
} | ||
|
||
TEST_F(LinkIntegrationTest, LinkAddForceInInertialFrame) | ||
{ | ||
EntityComponentManager ecm; | ||
EventManager eventMgr; | ||
SdfEntityCreator creator(ecm, eventMgr); | ||
|
||
auto eLink = ecm.CreateEntity(); | ||
ecm.CreateComponent(eLink, components::Link()); | ||
|
||
Link link(eLink); | ||
EXPECT_EQ(eLink, link.Entity()); | ||
|
||
ASSERT_TRUE(link.Valid(ecm)); | ||
|
||
// No ExternalWorldWrenchCmd should exist by default | ||
EXPECT_EQ(nullptr, ecm.Component<components::ExternalWorldWrenchCmd>(eLink)); | ||
|
||
// Add force in Inertial Frame | ||
math::Vector3d force(0, 0, 1.0); | ||
link.AddForceInInertialFrame(ecm, force); | ||
|
||
// No WorldPose or Inertial component exists so command should not work | ||
EXPECT_EQ(nullptr, ecm.Component<components::ExternalWorldWrenchCmd>(eLink)); | ||
|
||
// create WorldPose and Inertial component and try adding force again | ||
math::Pose3d linkWorldPose; | ||
linkWorldPose.Set(1.0, 0.0, 0.0, 0, 0, GZ_PI_4); | ||
math::Pose3d inertiaPose; | ||
inertiaPose.Set(1.0, 2.0, 3.0, 0, GZ_PI_2, 0); | ||
math::Inertiald linkInertial; | ||
linkInertial.SetPose(inertiaPose); | ||
ecm.CreateComponent(eLink, components::WorldPose(linkWorldPose)); | ||
ecm.CreateComponent(eLink, components::Inertial(linkInertial)); | ||
link.AddForceInInertialFrame(ecm, force); | ||
|
||
// ExternalWorldWrenchCmd component should now be created | ||
auto wrenchComp = ecm.Component<components::ExternalWorldWrenchCmd>(eLink); | ||
EXPECT_NE(nullptr, wrenchComp); | ||
|
||
// Summary of dynamic state at this time in the test: | ||
// - The link has a world pose with position (1, 0, 0) | ||
// and orientation (0, 0, π/4). | ||
// - The link has an inertial pose with position (1, 2, 3) | ||
// and orientation (0, π/2, 0). | ||
// - A force of (0, 0, 1) N is applied in the inertial frame. | ||
// Original force vector in inertial frame | ||
// force = [0, 0, 1] | ||
// Rotation matrix for inertiaPose (rotation about Y-axis by π/2 radians) | ||
// Ry = [ cos(π/2) 0 sin(π/2) | ||
// 0 1 0 | ||
// -sin(π/2) 0 cos(π/2) ] | ||
// Ry = [ 0 0 1 | ||
// 0 1 0 | ||
// -1 0 0 ] | ||
// | ||
// Calculate linkForce = Ry * force: | ||
// linkForce_x = (0 * 0) + (0 * 0) + (1 * 1) = 1 | ||
// linkForce_y = (0 * 0) + (1 * 0) + (0 * 1) = 0 | ||
// linkForce_z = (-1 * 0) + (0 * 0) + (0 * 1) = 0 | ||
// | ||
// Thus, linkForce = [1, 0, 0] | ||
// | ||
// Rotation matrix for linkWorldPose (rotation about Z-axis by π/4 radians) | ||
// Rz = [ cos(π/4) -sin(π/4) 0 | ||
// sin(π/4) cos(π/4) 0 | ||
// 0 0 1 ] | ||
// | ||
// Rz ≈ [ √2/2 -√2/2 0 | ||
// √2/2 √2/2 0 | ||
// 0 0 1 ] | ||
Vedant87 marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
// | ||
// Calculate worldForce = Rz * linkForce: | ||
// worldForce_x = (√2/2)*1 + (-√2/2)*0 + (0)*0 = √2/2 ≈ 0.7071 | ||
// worldForce_y = (√2/2)*1 + (√2/2)*0 + (0)*0 = √2/2 ≈ 0.7071 | ||
// worldForce_z = (0)*1 + (0)*0 + (1)*0 = 0 | ||
// | ||
// Thus, worldForce ≈ [√2/2, √2/2, 0] ≈ [0.7071, 0.7071, 0] | ||
// | ||
// Expected torque in world frame | ||
// First, rotate the inertia pose position to the world frame | ||
// using the link's world pose rotation. | ||
// Inertia pose position: (1, 2, 3) | ||
// Link world pose rotation matrix (around Z-axis by π/4 radians): | ||
// R_world = [ √2/2 -√2/2 0 | ||
// √2/2 √2/2 0 | ||
// 0 0 1] | ||
Vedant87 marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
// | ||
// Rotate inertia pose position to world frame: | ||
// r_world = R_world * [1, 2, 3] = [-√2/2, 3√2/2, 3] | ||
// | ||
// Then, calculate the cross product with the world force | ||
// to get the expected torque. | ||
// World force: (√2/2, √2/2, 0) | ||
// Cross product: | ||
// expectedTorque = r_world × F_world | ||
// expectedTorque = (-3√2/2, 3√2/2, -2) | ||
// | ||
// After the calculations the expected wrench is | ||
const double expectedForceX = std::sqrt(2) / 2; | ||
const double expectedForceY = std::sqrt(2) / 2; | ||
const double expectedForceZ = 0; | ||
const double expectedTorqueX = -3 * std::sqrt(2)/2; | ||
const double expectedTorqueY = 3 * std::sqrt(2)/2; | ||
const double expectedTorqueZ = -2; | ||
|
||
// verify Wrench values | ||
auto wrenchMsg = wrenchComp->Data(); | ||
|
||
// Looser tolerances are needed for the nonzero terms | ||
EXPECT_NEAR(expectedForceX, wrenchMsg.force().x(), 1e-2); | ||
EXPECT_NEAR(expectedForceY, wrenchMsg.force().y(), 1e-2); | ||
EXPECT_NEAR(expectedForceZ, wrenchMsg.force().z(), 1e-6); | ||
EXPECT_NEAR(expectedTorqueX, wrenchMsg.torque().x(), 1e-2); | ||
EXPECT_NEAR(expectedTorqueY, wrenchMsg.torque().y(), 1e-2); | ||
EXPECT_NEAR(expectedTorqueZ, wrenchMsg.torque().z(), 1e-2); | ||
|
||
// apply opposite force. Since the cmd is not processed yet, this should | ||
// cancel out the existing wrench cmd | ||
link.AddForceInInertialFrame(ecm, -force); | ||
wrenchComp = ecm.Component<components::ExternalWorldWrenchCmd>(eLink); | ||
EXPECT_NE(nullptr, wrenchComp); | ||
wrenchMsg = wrenchComp->Data(); | ||
|
||
EXPECT_EQ(math::Vector3d::Zero, math::Vector3d( | ||
wrenchMsg.force().x(), wrenchMsg.force().y(), wrenchMsg.force().z())); | ||
EXPECT_EQ(math::Vector3d::Zero, math::Vector3d( | ||
wrenchMsg.torque().x(), wrenchMsg.torque().y(), wrenchMsg.torque().z())); | ||
|
||
// Add force in Inertial Frame at an offset | ||
math::Vector3d offset{0.0, 1.0, 0.0}; | ||
|
||
link.AddForceInInertialFrame(ecm, force, offset); | ||
|
||
wrenchComp = ecm.Component<components::ExternalWorldWrenchCmd>(eLink); | ||
EXPECT_NE(nullptr, wrenchComp); | ||
|
||
// Summary of dynamic state at this time in the test: | ||
// - The link has a world pose with position (1, 0, 0) | ||
// and orientation (0, 0, π/4). | ||
// - The link has an inertial pose with position (1, 2, 3) | ||
// and orientation (0, π/2, 0). | ||
// - A force of (0, 0, 1) N is appled at an offset (0, 1, 0), | ||
// both force and offset are expressed in terms of the inertial frame. | ||
// Calculate the offset expressed in terms of link's coordinate frame | ||
// First, create a pose for the offset in the inertial frame. | ||
// offset = (0, 1, 0) | ||
// forceApplicationRelativeToInertialFrame = | ||
// Pose3d(offset, Quaterniond::Identity) | ||
// This pose represents a translation by the | ||
// offset vector without any rotation. | ||
// Transform this pose by the inertial pose to get the | ||
// offset in the link's frame. | ||
// Inertial pose: (1, 2, 3) with orientation (0, π/2, 0) | ||
// Inertial rotation matrix (around Y-axis by π/2 radians): | ||
// Ry = [ 0 0 1 | ||
// 0 1 0 | ||
// -1 0 0 ] | ||
// | ||
// Inertial translation matrix: | ||
// T = [1 0n, 0 1] | ||
Vedant87 marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
// [0 1 0 2] | ||
// [0 0 1 3] | ||
// [0 0 0 1] | ||
// | ||
// Combine rotation and translation into a homogeneous transformation matrix: | ||
// H_inertial = [Ry | T] = [0 0 1 | 1] | ||
// [0 1 0 | 2] | ||
// [-1 0 0 | 3] | ||
// [0 0 0 | 1] | ||
Vedant87 marked this conversation as resolved.
Outdated
Show resolved
Hide resolved
|
||
// | ||
// Apply this transformation to the offset pose: | ||
// offsetInLinkFrame = H_inertial * [0, 1, 0, 1]^T | ||
// = [0 0 1 | 1] * [0] | ||
// [0 1 0 | 2] [1] | ||
// [-1 0 0 | 3] [0] | ||
// [0 0 0 | 1] [1] | ||
// = [1, 3, 3] | ||
// offsetInLInkFrame = [1, 3, 3] | ||
// | ||
// Calculate the force expressed in link's coordinate frame with offset | ||
// First, rotate the force by the inertia pose's rotation. | ||
// Inertia pose rotation matrix (around Y-axis by π/2 radians): | ||
// Ry = [ 0 0 1 | ||
// 0 1 0 | ||
// -1 0 0 ] | ||
// Force in inertial frame: (0, 0, 1) | ||
// linkForceWithOffset = Ry * force = [1, 0, 0] | ||
// Then, rotate this force by the link's world pose | ||
// rotation to get the world force. | ||
// Link world pose rotation matrix (around Z-axis by π/4 radians): | ||
// Rz = [ √2/2 -√2/2 0 | ||
// √2/2 √2/2 0 | ||
// 0 0 1 ] | ||
// worldForceWithOffset = Rz * linkForceWithOffset = [√2/2, √2/2, 0] | ||
scpeters marked this conversation as resolved.
Show resolved
Hide resolved
|
||
// | ||
// Calculate the expected torque with offset | ||
// First, calculate the effective position vector in the world frame. | ||
// offsetInLinkFrame = [1, 3, 3] | ||
// inertiaPose.Pos() = [1, 2, 3] | ||
// effectivePosition = offsetInLinkFrame + inertiaPose.Pos() = [2, 5, 6] | ||
// Rotate this effective position by the link's world pose rotation: | ||
// effectivePositionWorld = Rz * effectivePosition | ||
// = [-3√2/2, 7√2/2, 6] | ||
// Calculate the cross product with the world force to get the torque. | ||
// worldForceWithOffset = [√2/2, √2/2, 0] | ||
// expectedTorqueWithOffset = effectivePositionWorld × worldForceWithOffset | ||
// expectedTorqueWithOffset = (-3√2, 3√2, -5) | ||
// After the calculations the expected wrench is | ||
const double expectedForceWithOffsetX = std::sqrt(2) / 2; | ||
const double expectedForceWithOffsetY = std::sqrt(2) / 2; | ||
const double expectedForceWithOffsetZ = 0; | ||
const double expectedTorqueWithOffsetX = -3 * std::sqrt(2); | ||
const double expectedTorqueWithOffsetY = 3 * std::sqrt(2); | ||
const double expectedTorqueWithOffsetZ = -5; | ||
|
||
wrenchMsg = wrenchComp->Data(); | ||
|
||
// Looser tolerances are needed for the nonzero terms | ||
EXPECT_NEAR(expectedForceWithOffsetX, wrenchMsg.force().x(), 1e-2); | ||
EXPECT_NEAR(expectedForceWithOffsetY, wrenchMsg.force().y(), 1e-2); | ||
EXPECT_NEAR(expectedForceWithOffsetZ, wrenchMsg.force().z(), 1e-6); | ||
EXPECT_NEAR(expectedTorqueWithOffsetX, wrenchMsg.torque().x(), 1e-2); | ||
EXPECT_NEAR(expectedTorqueWithOffsetY, wrenchMsg.torque().y(), 1e-2); | ||
EXPECT_NEAR(expectedTorqueWithOffsetZ, wrenchMsg.torque().z(), 1e-2); | ||
|
||
// apply opposite force again and verify the resulting wrench values are zero | ||
link.AddForceInInertialFrame(ecm, -force, offset); | ||
wrenchComp = ecm.Component<components::ExternalWorldWrenchCmd>(eLink); | ||
EXPECT_NE(nullptr, wrenchComp); | ||
wrenchMsg = wrenchComp->Data(); | ||
|
||
EXPECT_EQ(math::Vector3d::Zero, math::Vector3d( | ||
wrenchMsg.force().x(), wrenchMsg.force().y(), wrenchMsg.force().z())); | ||
EXPECT_EQ(math::Vector3d::Zero, math::Vector3d( | ||
wrenchMsg.torque().x(), wrenchMsg.torque().y(), wrenchMsg.torque().z())); | ||
|
||
} |
Uh oh!
There was an error while loading. Please reload this page.