Skip to content

Commit

Permalink
fix the soft limiting logic and add respective tests
Browse files Browse the repository at this point in the history
  • Loading branch information
saikishor committed Oct 11, 2024
1 parent 2a346b8 commit 2f91440
Show file tree
Hide file tree
Showing 2 changed files with 47 additions and 16 deletions.
14 changes: 8 additions & 6 deletions joint_limits/src/joint_soft_limiter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -96,13 +96,13 @@ bool JointSoftLimiter::on_enforce(
double soft_max_vel = std::numeric_limits<double>::infinity();
double position = std::numeric_limits<double>::infinity();

if (prev_command_.has_position() && std::isfinite(prev_command_.position.value()))
if (actual.has_position())
{
position = prev_command_.position.value();
position = actual.position.value();
}
else if (actual.has_position())
else if (prev_command_.has_position() && std::isfinite(prev_command_.position.value()))
{
position = actual.position.value();
position = prev_command_.position.value();
}

if (hard_limits.has_velocity_limits)
Expand All @@ -122,15 +122,17 @@ bool JointSoftLimiter::on_enforce(
-soft_joint_limits.k_position * (position - soft_joint_limits.max_position),
-hard_limits.max_velocity, hard_limits.max_velocity);

if ((position < hard_limits.min_position) || (position > hard_limits.max_position))
if (
(position < (hard_limits.min_position - internal::POSITION_BOUNDS_TOLERANCE)) ||
(position > (hard_limits.max_position + internal::POSITION_BOUNDS_TOLERANCE)))
{
soft_min_vel = 0.0;
soft_max_vel = 0.0;
}
else if (
(position < soft_joint_limits.min_position) || (position > soft_joint_limits.max_position))
{
constexpr double soft_limit_reach_velocity = 1.0 * (M_PI / 180.0);
const double soft_limit_reach_velocity = 1.0 * (M_PI / 180.0);
soft_min_vel = std::copysign(soft_limit_reach_velocity, soft_min_vel);
soft_max_vel = std::copysign(soft_limit_reach_velocity, soft_max_vel);
}
Expand Down
49 changes: 39 additions & 10 deletions joint_limits/test/test_joint_soft_limiter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -214,6 +214,11 @@ TEST_F(JointSoftLimiterTest, check_desired_position_only_cases)
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.0, COMMON_THRESHOLD);
actual_state_.position = 0.95;
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.95, COMMON_THRESHOLD);
actual_state_.position = 1.5;
desired_state_.position = 2.0;
ASSERT_FALSE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 2.0, COMMON_THRESHOLD);
Expand All @@ -228,34 +233,58 @@ TEST_F(JointSoftLimiterTest, check_desired_position_only_cases)
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.0, COMMON_THRESHOLD);
actual_state_.position = 0.45;
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.45, COMMON_THRESHOLD);
actual_state_.position = 0.95;
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.5, COMMON_THRESHOLD);

soft_limits.k_position = 2.0;
// The case of no actual position feedback
soft_limits.k_position = 0.5;
soft_limits.max_position = 1.5;
soft_limits.min_position = -1.5;
ASSERT_TRUE(Init(limits, soft_limits));
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.0, COMMON_THRESHOLD);
actual_state_.position = std::nullopt;
desired_state_.position = 0.2;
ASSERT_FALSE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 0.2, COMMON_THRESHOLD);
auto prev_command_state = desired_state_;
while (desired_state_.position.value() < (soft_limits.max_position - COMMON_THRESHOLD))
{
desired_state_.position = 2.0;
double expected_pos =
prev_command_state.position.value() +
(soft_limits.max_position - prev_command_state.position.value()) * soft_limits.k_position;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), expected_pos, COMMON_THRESHOLD);
prev_command_state = desired_state_;
}
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.5, COMMON_THRESHOLD);

// More generic test case
soft_limits.k_position = 0.5;
soft_limits.max_position = 2.0;
soft_limits.min_position = -2.0;
ASSERT_TRUE(Init(limits, soft_limits));
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.0, COMMON_THRESHOLD);
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.5, COMMON_THRESHOLD);
desired_state_.position = 2.0;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), 1.75, COMMON_THRESHOLD);
actual_state_.position = 0.2;
while (actual_state_.position.value() < (desired_state_.position.value() - COMMON_THRESHOLD))
{
desired_state_.position = 2.0;
double expected_pos =
actual_state_.position.value() +
(soft_limits.max_position - actual_state_.position.value()) * soft_limits.k_position;
ASSERT_TRUE(joint_limiter_->enforce(actual_state_, desired_state_, period));
EXPECT_NEAR(desired_state_.position.value(), expected_pos, COMMON_THRESHOLD);
actual_state_.position = expected_pos;
}

// Now test when there are no position limits and soft limits, then the desired position is not
// saturated
Expand Down

0 comments on commit 2f91440

Please sign in to comment.