Skip to content
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

Normalize joint axis xyz vector when parsing from SDFormat #312

Merged
merged 10 commits into from
Aug 7, 2020
3 changes: 3 additions & 0 deletions Changelog.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,9 @@

### libsdformat 10.0.0 (202X-XX-XX)

1. Normalize joint axis xyz vector when parsing from SDFormat
* [Pull request 312](https://github.com/osrf/sdformat/pull/312)

1. Enforce minimum/maximum values specified in SDFormat description files.
* [Pull request 303](https://github.com/osrf/sdformat/pull/303)

Expand Down
4 changes: 4 additions & 0 deletions Migration.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,10 @@ but with improved human-readability..

### Modifications

1. Axis vectors specified in <joint><axis><xyz> are normalized if their norm is
greater than 0. A vector with 0 norm generates an error
* [Pull request 312](https://github.com/osrf/sdformat/pull/312)

1. + Minimum/maximum values specified in SDFormat description files are now enforced
+ [Pull request 303](https://github.com/osrf/sdformat/pull/303)

Expand Down
2 changes: 2 additions & 0 deletions include/sdf/JointAxis.hh
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
#include <string>
#include <ignition/math/Vector3.hh>
#include "sdf/Element.hh"
#include "sdf/Exception.hh"
#include "sdf/Types.hh"
#include "sdf/sdf_config.h"
#include "sdf/system_util.hh"
Expand Down Expand Up @@ -93,6 +94,7 @@ namespace sdf

/// \brief Set the x,y,z components of the axis unit vector.
/// \param[in] _xyz The x,y,z components of the axis unit vector.
/// \throws sdf::AssertionInternalError if the norm of the xyz vector is 0.
azeey marked this conversation as resolved.
Show resolved Hide resolved
/// \sa ignition::math::Vector3d Xyz() const
public: void SetXyz(const ignition::math::Vector3d &_xyz);

Expand Down
16 changes: 14 additions & 2 deletions src/JointAxis.cc
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,8 @@
*/
#include <ignition/math/Pose3.hh>
#include <ignition/math/Vector3.hh>

#include "sdf/Assert.hh"
#include "sdf/Error.hh"
#include "sdf/JointAxis.hh"
#include "FrameSemantics.hh"
Expand Down Expand Up @@ -136,8 +138,15 @@ Errors JointAxis::Load(ElementPtr _sdf)
// Read the xyz values.
if (_sdf->HasElement("xyz"))
{
this->dataPtr->xyz = _sdf->Get<ignition::math::Vector3d>("xyz",
ignition::math::Vector3d::UnitZ).first;
try
{
using ignition::math::Vector3d;
this->SetXyz(_sdf->Get<Vector3d>("xyz", Vector3d::UnitZ).first);
}
catch(const sdf::AssertionInternalError &_e)
{
errors.push_back({ErrorCode::ELEMENT_INVALID, _e.GetErrorStr()});
}
auto e = _sdf->GetElement("xyz");
if (e->HasAttribute("expressed_in"))
{
Expand Down Expand Up @@ -211,7 +220,10 @@ ignition::math::Vector3d JointAxis::Xyz() const
/////////////////////////////////////////////////
void JointAxis::SetXyz(const ignition::math::Vector3d &_xyz)
{
SDF_ASSERT(!sdf::equal(_xyz.Length(), 0.0),
"The norm of the xyz vector cannot be zero");
this->dataPtr->xyz = _xyz;
this->dataPtr->xyz.Normalize();
}

/////////////////////////////////////////////////
Expand Down
9 changes: 9 additions & 0 deletions src/JointAxis_TEST.cc
Original file line number Diff line number Diff line change
Expand Up @@ -145,3 +145,12 @@ TEST(DOMJointAxis, CopyAssignmentAfterMove)
EXPECT_EQ(axis2, jointAxis1.Xyz());
EXPECT_EQ(axis1, jointAxis2.Xyz());
}

/////////////////////////////////////////////////
TEST(DOMJointAxis, ZeroNormVectorThrows)
{
sdf::JointAxis axis;
EXPECT_NO_THROW(axis.SetXyz({1.0, 0, 0}));
EXPECT_THROW(axis.SetXyz(ignition::math::Vector3d::Zero),
sdf::AssertionInternalError);
}
53 changes: 53 additions & 0 deletions test/integration/joint_axis_dom.cc
Original file line number Diff line number Diff line change
Expand Up @@ -219,3 +219,56 @@ TEST(DOMJointAxis, XyzExpressedIn)
EXPECT_EQ(0u, model->FrameCount());
EXPECT_EQ(nullptr, model->FrameByIndex(0));
}

//////////////////////////////////////////////////
TEST(DOMJointAxis, XyzNormalization)
{
const std::string testFile =
sdf::filesystem::append(PROJECT_SOURCE_PATH, "test", "sdf",
"joint_axis_xyz_normalization.sdf");

// Load the SDF file
sdf::Root root;
sdf::Errors errors = root.Load(testFile);

ASSERT_EQ(1u, errors.size());
EXPECT_TRUE(
errors[0].Message().find("The norm of the xyz vector cannot be zero") !=
std::string::npos);

using ignition::math::Vector3d;

// Get the first model
const sdf::Model *model = root.ModelByIndex(0);
ASSERT_NE(nullptr, model);

{
auto joint1 = model->JointByName("joint1");
ASSERT_FALSE(nullptr == joint1);
ASSERT_FALSE(nullptr == joint1->Axis(0));
EXPECT_EQ(Vector3d::UnitZ, joint1->Axis(0)->Xyz());
}

{
auto joint2 = model->JointByName("joint2");
ASSERT_FALSE(nullptr == joint2);
ASSERT_FALSE(nullptr == joint2->Axis(0));
EXPECT_EQ(Vector3d::UnitX, joint2->Axis(0)->Xyz());
}

{
auto joint3 = model->JointByName("joint3");
ASSERT_FALSE(nullptr == joint3);
ASSERT_FALSE(nullptr == joint3->Axis(0));
EXPECT_EQ(-Vector3d::UnitX, joint3->Axis(0)->Xyz());
ASSERT_FALSE(nullptr == joint3->Axis(1));
EXPECT_EQ(Vector3d::UnitY, joint3->Axis(1)->Xyz());
}

{
auto joint4 = model->JointByName("joint4");
ASSERT_FALSE(nullptr == joint4);
ASSERT_FALSE(nullptr == joint4->Axis(0));
EXPECT_EQ(Vector3d::UnitZ, joint4->Axis(0)->Xyz());
}
}
43 changes: 43 additions & 0 deletions test/sdf/joint_axis_xyz_normalization.sdf
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
<?xml version="1.0" ?>
<sdf version="1.6">
<model name="test">
<link name="link1"/>
<link name="link2"/>
<link name="link3"/>
<link name="link4"/>
<link name="link5"/>

<joint name="joint1" type="revolute">
<child>link1</child>
<parent>link2</parent>
<axis>
<xyz>0 0 1</xyz>
</axis>
</joint>
<joint name="joint2" type="revolute">
<child>link3</child>
<parent>link4</parent>
<axis>
<xyz>10 0 0</xyz>
</axis>
</joint>
<joint name="joint3" type="universal">
<child>link4</child>
<parent>link5</parent>
<axis>
<xyz>-10 0 0</xyz>
</axis>
<axis2>
<xyz>0 10 0</xyz>
</axis2>
</joint>
<joint name="joint4" type="revolute">
<child>link5</child>
<parent>link6</parent>
<axis>
<xyz>0 0 0</xyz>
</axis>
</joint>
</model>
</sdf>