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

Add Saturation Joint Limiter that uses clamping method #971

Merged
merged 75 commits into from
Jul 4, 2024
Merged
Show file tree
Hide file tree
Changes from 71 commits
Commits
Show all changes
75 commits
Select commit Hold shift + click to select a range
5d81a98
Added controller parameters structures and movement interfaces.
destogl Sep 17, 2021
aa004a3
Added joint limiter plugins.
destogl Sep 18, 2021
bbd9d59
Debug Ruckig JointLimiter.
destogl Sep 24, 2021
ee7faf2
Modify simple joint limiting plugin (same as changes to moveit2 filte…
destogl Feb 9, 2023
a07c7ac
Remove all Ruckig-limiter files.
destogl Mar 16, 2023
5260724
Merge branch 'master' into add-simple-joint-limiter
destogl Mar 16, 2023
6675e7d
Merge branch 'master' into add-simple-joint-limiter
destogl Mar 23, 2023
ceb3beb
Merge branch 'master' into add-simple-joint-limiter
destogl Apr 13, 2023
f939193
Apply suggestions from code review
destogl Apr 13, 2023
d7db070
Remove definition of movement interface because the concept is not us…
destogl Apr 13, 2023
4054b04
Polish out some stuff using "GH Programming"
destogl Apr 13, 2023
456c715
Polish out some stuff using "GH Programming"
destogl Apr 13, 2023
8b33153
Add SimpleJointLimiter as plugin into controllers.
destogl Apr 24, 2023
cbfc06a
Version with deceleration for simple joint limiter.
destogl May 4, 2023
2777c61
Merge branch 'master' into add-simple-joint-limiter
destogl May 9, 2023
b401169
Formatting and comment to check.
destogl May 12, 2023
e6f52d7
Added test of joint_limiter
gwalck Jun 7, 2023
7bfc94e
Fixed deceleration limit application
gwalck Jun 7, 2023
c338b8c
Updated authorship
gwalck Jun 9, 2023
6e0e6c5
Split test, apply review changes, add future tests
gwalck Jun 9, 2023
cfc8fe5
Applied review comment, added 2 tests & fixed typo
gwalck Jun 12, 2023
186c66e
Improved close to limit trigger
gwalck Jun 12, 2023
d6db2a1
Merge branch 'master' into add-simple-joint-limiter
destogl Jun 13, 2023
1e393f3
Merge branch 'master' into add-simple-joint-limiter
destogl Jun 27, 2023
1749f6d
Update joint_limits/src/simple_joint_limiter.cpp
destogl Jun 27, 2023
e254294
Fix formatting.
destogl Jun 29, 2023
21944b3
Added temporary pseudo-code
gwalck Jul 7, 2023
7530ba8
Updated pseudo-code
gwalck Jul 7, 2023
27e3304
Refactored to match pseudo-code
gwalck Jul 7, 2023
d5892e8
Adapted tests
gwalck Jul 7, 2023
d848a7d
lint
gwalck Jul 7, 2023
8c771cf
Cleanup debug helpers
gwalck Jul 7, 2023
7e8e4cd
Removed pseudo-code
gwalck Jul 7, 2023
c7fdd46
Merge branch 'master' into add-simple-joint-limiter
destogl Jul 13, 2023
098a431
Merge branch 'master' into add-simple-joint-limiter
destogl Aug 3, 2023
c184da8
Merge branch 'master' into add-simple-joint-limiter
destogl Aug 30, 2023
345370a
Apply suggestions from code review
destogl Sep 4, 2023
c99eb86
Apply suggestions from code review
destogl Sep 4, 2023
5a9985c
Unify how joints were limits are reached are stored.
destogl Sep 4, 2023
9c2bba7
Fix formatting.
destogl Sep 4, 2023
6ca6594
Fix unstable limiter because velocity is caculated from position when…
destogl Sep 12, 2023
40228d9
fix formatting.
destogl Sep 12, 2023
d6b0628
Merge branch 'master' into add-simple-joint-limiter
destogl Nov 1, 2023
5d164b1
Fix unstable limiter because velocity is caculated from position when…
destogl Sep 12, 2023
e99fca2
Make acceleration more stable in limiter.
destogl Sep 14, 2023
96bed7c
Enable dynamic update of limits from parameters.
destogl Nov 1, 2023
47389fd
Fix formatting.
destogl Nov 1, 2023
af5e004
Debug position limiting in the simiple joint limiter.
destogl Nov 16, 2023
4f868b1
Update tests after changes.
destogl Nov 21, 2023
72b1ea7
Merge branch 'master' into add-simple-joint-limiter
destogl Nov 21, 2023
47d073d
Merge branch 'master' into add-simple-joint-limiter
destogl Nov 30, 2023
94da997
Fixup shadowing variable.
destogl Dec 7, 2023
9ed9b01
Merge branch 'master' into add-simple-joint-limiter
destogl Dec 7, 2023
bcf42aa
rename to JointSaturationLimiter
mamueluth Dec 8, 2023
fde4814
rename files as well
mamueluth Dec 11, 2023
15b9255
add effort limits
mamueluth Dec 13, 2023
8e032f4
if move limit efforts to seperate function
mamueluth Dec 14, 2023
5d4e7e4
Merge branch 'master' into add-simple-joint-limiter
destogl Dec 18, 2023
0c75f46
Merge branch 'master' into add-simple-joint-limiter
destogl Dec 18, 2023
9a40a75
add generic enforce method
mamueluth Jan 8, 2024
92d0468
Merge branch 'master' into add-simple-joint-limiter
destogl Jan 13, 2024
18f0ec0
Merge branch 'master' into add-simple-joint-limiter
destogl Jan 17, 2024
7e27edc
Interface cleanup and small optimizations from reviewers' comments.
destogl Jan 18, 2024
bd25f6e
Merge branch 'master' into add-simple-joint-limiter
destogl Jan 18, 2024
56dd00f
Added package to CI
destogl Jan 22, 2024
5c50bd6
Merge branch 'master' into add-simple-joint-limiter
destogl Jan 22, 2024
ed2f374
Merge branch 'master' into add-simple-joint-limiter
destogl Jan 25, 2024
fdc4e05
Merge branch 'master' into add-simple-joint-limiter
destogl Jan 25, 2024
f96f8da
Merge branch 'master' into add-simple-joint-limiter
destogl Feb 8, 2024
284437a
Correct sorting of packages.
destogl Feb 8, 2024
d3838fd
Small fixes of calculation when differnet set of inputs are provided.
destogl Feb 8, 2024
f097085
Merge remote-tracking branch 'origin/master' into add-simple-joint-li…
destogl Jun 11, 2024
271b6ba
Merge branch 'master' into add-simple-joint-limiter
bmagyar Jun 17, 2024
e18fb05
Merge branch 'master' into add-simple-joint-limiter
destogl Jul 3, 2024
33860f9
Update the year in license.
destogl Jul 3, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .github/workflows/ci-coverage-build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ jobs:
controller_manager
hardware_interface
hardware_interface_testing
joint_limits
transmission_interface

vcs-repo-file-url: |
Expand Down
1 change: 1 addition & 0 deletions .github/workflows/ci-ros-lint.yml
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ env:
controller_manager_msgs
hardware_interface
hardware_interface_testing
joint_limits
ros2controlcli
ros2_control
ros2_control_test_assets
Expand Down
55 changes: 53 additions & 2 deletions joint_limits/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,16 @@ if(CMAKE_CXX_COMPILER_ID MATCHES "(GNU|Clang)")
endif()

set(THIS_PACKAGE_INCLUDE_DEPENDS
pluginlib
realtime_tools
rclcpp
rclcpp_lifecycle
trajectory_msgs
urdf
)

find_package(ament_cmake REQUIRED)
find_package(backward_ros REQUIRED)
foreach(Dependency IN ITEMS ${THIS_PACKAGE_INCLUDE_DEPENDS})
find_package(${Dependency} REQUIRED)
endforeach()
Expand All @@ -24,10 +28,42 @@ target_include_directories(joint_limits INTERFACE
)
ament_target_dependencies(joint_limits INTERFACE ${THIS_PACKAGE_INCLUDE_DEPENDS})

add_library(joint_limiter_interface SHARED
src/joint_limiter_interface.cpp
)
target_compile_features(joint_limiter_interface PUBLIC cxx_std_17)
target_include_directories(joint_limiter_interface PUBLIC
$<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}/include>
$<INSTALL_INTERFACE:include/joint_limits>
)
ament_target_dependencies(joint_limiter_interface PUBLIC ${THIS_PACKAGE_INCLUDE_DEPENDS})
# Causes the visibility macros to use dllexport rather than dllimport,
# which is appropriate when building the dll but not consuming it.
target_compile_definitions(joint_limiter_interface PRIVATE "JOINT_LIMITS_BUILDING_DLL")


add_library(joint_saturation_limiter SHARED
src/joint_saturation_limiter.cpp
)
target_compile_features(joint_saturation_limiter PUBLIC cxx_std_17)
target_include_directories(joint_saturation_limiter PUBLIC
$<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}/include>
$<INSTALL_INTERFACE:include/joint_limits>
)
ament_target_dependencies(joint_saturation_limiter PUBLIC ${THIS_PACKAGE_INCLUDE_DEPENDS})
# Causes the visibility macros to use dllexport rather than dllimport,
# which is appropriate when building the dll but not consuming it.
target_compile_definitions(joint_saturation_limiter PRIVATE "JOINT_LIMITS_BUILDING_DLL")

pluginlib_export_plugin_description_file(joint_limits joint_limiters.xml)

if(BUILD_TESTING)
find_package(ament_cmake_gtest REQUIRED)
find_package(ament_cmake_gmock REQUIRED)
find_package(generate_parameter_library REQUIRED)
find_package(launch_testing_ament_cmake REQUIRED)
find_package(pluginlib REQUIRED)
find_package(rclcpp REQUIRED)

ament_add_gtest_executable(joint_limits_rosparam_test test/joint_limits_rosparam_test.cpp)
target_link_libraries(joint_limits_rosparam_test joint_limits)
Expand All @@ -41,16 +77,31 @@ if(BUILD_TESTING)
DESTINATION lib/joint_limits
)
install(
FILES test/joint_limits_rosparam.yaml
FILES test/joint_limits_rosparam.yaml test/joint_saturation_limiter_param.yaml
DESTINATION share/joint_limits/test
)

add_rostest_with_parameters_gmock(test_joint_saturation_limiter test/test_joint_saturation_limiter.cpp
${CMAKE_CURRENT_SOURCE_DIR}/test/joint_saturation_limiter_param.yaml
)
target_include_directories(test_joint_saturation_limiter PRIVATE include)
target_link_libraries(test_joint_saturation_limiter joint_limiter_interface)
ament_target_dependencies(
test_joint_saturation_limiter
pluginlib
rclcpp
)

endif()

install(
DIRECTORY include/
DESTINATION include/joint_limits
)
install(TARGETS joint_limits
install(TARGETS
joint_limits
joint_limiter_interface
joint_saturation_limiter
EXPORT export_joint_limits
ARCHIVE DESTINATION lib
LIBRARY DESTINATION lib
Expand Down
250 changes: 250 additions & 0 deletions joint_limits/include/joint_limits/joint_limiter_interface.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,250 @@
// Copyright (c) 2023, Stogl Robotics Consulting UG (haftungsbeschränkt)
destogl marked this conversation as resolved.
Show resolved Hide resolved
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

/// \author Denis Stogl

#ifndef JOINT_LIMITS__JOINT_LIMITER_INTERFACE_HPP_
#define JOINT_LIMITS__JOINT_LIMITER_INTERFACE_HPP_

#include <string>
#include <vector>

#include "joint_limits/joint_limits.hpp"
#include "joint_limits/joint_limits_rosparam.hpp"
#include "joint_limits/visibility_control.h"
#include "rclcpp/node.hpp"
#include "rclcpp_lifecycle/lifecycle_node.hpp"
#include "realtime_tools/realtime_buffer.h"
#include "trajectory_msgs/msg/joint_trajectory_point.hpp"

namespace joint_limits
{
using JointLimitsStateDataType = trajectory_msgs::msg::JointTrajectoryPoint;

template <typename LimitsType>
class JointLimiterInterface
{
public:
JOINT_LIMITS_PUBLIC JointLimiterInterface() = default;

JOINT_LIMITS_PUBLIC virtual ~JointLimiterInterface() = default;

/// Initialization of every JointLimiter.
/**
* Initialization of JointLimiter for defined joints with their names.
* Robot description topic provides a topic name where URDF of the robot can be found.
* This is needed to use joint limits from URDF (not implemented yet!).
* Override this method only if initialization and reading joint limits should be adapted.
* Otherwise, initialize your custom limiter in `on_limit` method.
*
* \param[in] joint_names names of joints where limits should be applied.
* \param[in] param_itf node parameters interface object to access parameters.
* \param[in] logging_itf node logging interface to log if error happens.
* \param[in] robot_description_topic string of a topic where robot description is accessible.
*/
JOINT_LIMITS_PUBLIC virtual bool init(
const std::vector<std::string> & joint_names,
const rclcpp::node_interfaces::NodeParametersInterface::SharedPtr & param_itf,
const rclcpp::node_interfaces::NodeLoggingInterface::SharedPtr & logging_itf,
const std::string & robot_description_topic = "/robot_description")
{
number_of_joints_ = joint_names.size();
joint_names_ = joint_names;
joint_limits_.resize(number_of_joints_);
node_param_itf_ = param_itf;
node_logging_itf_ = logging_itf;

bool result = true;

// TODO(destogl): get limits from URDF

// Initialize and get joint limits from parameter server
for (size_t i = 0; i < number_of_joints_; ++i)
{
if (!declare_parameters(joint_names[i], node_param_itf_, node_logging_itf_))
{
RCLCPP_ERROR(
node_logging_itf_->get_logger(),
"JointLimiter: Joint '%s': parameter declaration has failed", joint_names[i].c_str());
result = false;
break;
}
if (!get_joint_limits(joint_names[i], node_param_itf_, node_logging_itf_, joint_limits_[i]))
{
RCLCPP_ERROR(
node_logging_itf_->get_logger(),
"JointLimiter: Joint '%s': getting parameters has failed", joint_names[i].c_str());
result = false;
break;
}
RCLCPP_INFO(
node_logging_itf_->get_logger(), "Limits for joint %zu (%s) are:\n%s", i,
joint_names[i].c_str(), joint_limits_[i].to_string().c_str());
}
updated_limits_.writeFromNonRT(joint_limits_);

auto on_parameter_event_callback = [this](const std::vector<rclcpp::Parameter> & parameters)
{
rcl_interfaces::msg::SetParametersResult set_parameters_result;
set_parameters_result.successful = true;

std::vector<LimitsType> updated_joint_limits = joint_limits_;
bool changed = false;

for (size_t i = 0; i < number_of_joints_; ++i)
{
changed |= joint_limits::check_for_limits_update(
joint_names_[i], parameters, node_logging_itf_, updated_joint_limits[i]);
}

if (changed)
{
updated_limits_.writeFromNonRT(updated_joint_limits);
RCLCPP_INFO(node_logging_itf_->get_logger(), "Limits are dynamically updated!");
}

return set_parameters_result;
};

parameter_callback_ =
node_param_itf_->add_on_set_parameters_callback(on_parameter_event_callback);

if (result)
{
result = on_init();
}

(void)robot_description_topic; // avoid linters output

return result;
}

/**
* Wrapper init method that accepts pointer to the Node.
* For details see other init method.
*/
JOINT_LIMITS_PUBLIC virtual bool init(
const std::vector<std::string> & joint_names, const rclcpp::Node::SharedPtr & node,
const std::string & robot_description_topic = "/robot_description")
{
return init(
joint_names, node->get_node_parameters_interface(), node->get_node_logging_interface(),
robot_description_topic);
}

/**
* Wrapper init method that accepts pointer to the LifecycleNode.
* For details see other init method.
*/
JOINT_LIMITS_PUBLIC virtual bool init(
const std::vector<std::string> & joint_names,
const rclcpp_lifecycle::LifecycleNode::SharedPtr & lifecycle_node,
const std::string & robot_description_topic = "/robot_description")
{
return init(
joint_names, lifecycle_node->get_node_parameters_interface(),
lifecycle_node->get_node_logging_interface(), robot_description_topic);
}

JOINT_LIMITS_PUBLIC virtual bool configure(const JointLimitsStateDataType & current_joint_states)
{
return on_configure(current_joint_states);
}

/** \brief Enforce joint limits to desired joint state for multiple physical quantities.
*
* Generic enforce method that calls implementation-specific `on_enforce` method.
*
* \param[in] current_joint_states current joint states a robot is in.
* \param[in,out] desired_joint_states joint state that should be adjusted to obey the limits.
* \param[in] dt time delta to calculate missing integrals and derivation in joint limits.
* \returns true if limits are enforced, otherwise false.
*/
JOINT_LIMITS_PUBLIC virtual bool enforce(
JointLimitsStateDataType & current_joint_states,
JointLimitsStateDataType & desired_joint_states, const rclcpp::Duration & dt)
{
joint_limits_ = *(updated_limits_.readFromRT());
return on_enforce(current_joint_states, desired_joint_states, dt);
}

/** \brief Enforce joint limits to desired joint state for single physical quantity.
*
* Generic enforce method that calls implementation-specific `on_enforce` method.
*
* \param[in,out] desired_joint_states joint state that should be adjusted to obey the limits.
* \returns true if limits are enforced, otherwise false.
*/
JOINT_LIMITS_PUBLIC virtual bool enforce(std::vector<double> & desired_joint_states)
{
joint_limits_ = *(updated_limits_.readFromRT());
return on_enforce(desired_joint_states);
}

protected:
/** \brief Method is realized by an implementation.
*
* Implementation-specific initialization of limiter's internal states and libraries.
* \returns true if initialization was successful, otherwise false.
*/
JOINT_LIMITS_PUBLIC virtual bool on_init() = 0;

/** \brief Method is realized by an implementation.
*
* Implementation-specific configuration of limiter's internal states and libraries.
* \returns true if initialization was successful, otherwise false.
*/
JOINT_LIMITS_PUBLIC virtual bool on_configure(
const JointLimitsStateDataType & current_joint_states) = 0;

/** \brief Method is realized by an implementation.
*
* Filter-specific implementation of the joint limits enforce algorithm for multiple dependent
* physical quantities.
*
* \param[in] current_joint_states current joint states a robot is in.
* \param[in,out] desired_joint_states joint state that should be adjusted to obey the limits.
* \param[in] dt time delta to calculate missing integrals and derivation in joint limits.
* \returns true if limits are enforced, otherwise false.
*/
JOINT_LIMITS_PUBLIC virtual bool on_enforce(
JointLimitsStateDataType & current_joint_states,
JointLimitsStateDataType & desired_joint_states, const rclcpp::Duration & dt) = 0;

/** \brief Method is realized by an implementation.
*
* Filter-specific implementation of the joint limits enforce algorithm for single physical
* quantity.
* This method might use "effort" limits since they are often used as wild-card.
* Check the documentation of the exact implementation for more details.
*
* \param[in,out] desired_joint_states joint state that should be adjusted to obey the limits.
* \returns true if limits are enforced, otherwise false.
*/
JOINT_LIMITS_PUBLIC virtual bool on_enforce(std::vector<double> & desired_joint_states) = 0;

size_t number_of_joints_;
std::vector<std::string> joint_names_;
std::vector<LimitsType> joint_limits_;
rclcpp::node_interfaces::NodeParametersInterface::SharedPtr node_param_itf_;
rclcpp::node_interfaces::NodeLoggingInterface::SharedPtr node_logging_itf_;

private:
rclcpp::node_interfaces::OnSetParametersCallbackHandle::SharedPtr parameter_callback_;
realtime_tools::RealtimeBuffer<std::vector<LimitsType>> updated_limits_;
};

} // namespace joint_limits

#endif // JOINT_LIMITS__JOINT_LIMITER_INTERFACE_HPP_
Loading
Loading