Loading [MathJax]/extensions/tex2jax.js
ros2_control - humble
All Classes Namespaces Functions Variables Typedefs Enumerations Pages
tricycle_controller.hpp
1// Copyright 2022 Pixel Robotics.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15/*
16 * Author: Tony Najjar
17 */
18
19#ifndef TRICYCLE_CONTROLLER__TRICYCLE_CONTROLLER_HPP_
20#define TRICYCLE_CONTROLLER__TRICYCLE_CONTROLLER_HPP_
21
22#include <chrono>
23#include <cmath>
24#include <memory>
25#include <queue>
26#include <string>
27#include <tuple>
28#include <vector>
29
30#include "ackermann_msgs/msg/ackermann_drive.hpp"
31#include "controller_interface/controller_interface.hpp"
32#include "geometry_msgs/msg/twist.hpp"
33#include "geometry_msgs/msg/twist_stamped.hpp"
34#include "hardware_interface/handle.hpp"
35#include "nav_msgs/msg/odometry.hpp"
36#include "rclcpp/rclcpp.hpp"
37#include "rclcpp_lifecycle/state.hpp"
38#include "realtime_tools/realtime_box.hpp"
39#include "realtime_tools/realtime_publisher.hpp"
40#include "std_srvs/srv/empty.hpp"
41#include "tf2_msgs/msg/tf_message.hpp"
42
43#include "tricycle_controller/odometry.hpp"
44#include "tricycle_controller/steering_limiter.hpp"
45#include "tricycle_controller/traction_limiter.hpp"
46#include "tricycle_controller/visibility_control.h"
47
48namespace tricycle_controller
49{
50using CallbackReturn = rclcpp_lifecycle::node_interfaces::LifecycleNodeInterface::CallbackReturn;
51
53{
54 using Twist = geometry_msgs::msg::Twist;
55 using TwistStamped = geometry_msgs::msg::TwistStamped;
56 using AckermannDrive = ackermann_msgs::msg::AckermannDrive;
57
58public:
59 TRICYCLE_CONTROLLER_PUBLIC
61
62 TRICYCLE_CONTROLLER_PUBLIC
64
65 TRICYCLE_CONTROLLER_PUBLIC
67
68 TRICYCLE_CONTROLLER_PUBLIC
69 controller_interface::return_type update(
70 const rclcpp::Time & time, const rclcpp::Duration & period) override;
71
72 TRICYCLE_CONTROLLER_PUBLIC
73 CallbackReturn on_init() override;
74
75 TRICYCLE_CONTROLLER_PUBLIC
76 CallbackReturn on_configure(const rclcpp_lifecycle::State & previous_state) override;
77
78 TRICYCLE_CONTROLLER_PUBLIC
79 CallbackReturn on_activate(const rclcpp_lifecycle::State & previous_state) override;
80
81 TRICYCLE_CONTROLLER_PUBLIC
82 CallbackReturn on_deactivate(const rclcpp_lifecycle::State & previous_state) override;
83
84 TRICYCLE_CONTROLLER_PUBLIC
85 CallbackReturn on_cleanup(const rclcpp_lifecycle::State & previous_state) override;
86
87 TRICYCLE_CONTROLLER_PUBLIC
88 CallbackReturn on_error(const rclcpp_lifecycle::State & previous_state) override;
89
90protected:
92 {
93 std::reference_wrapper<const hardware_interface::LoanedStateInterface> velocity_state;
94 std::reference_wrapper<hardware_interface::LoanedCommandInterface> velocity_command;
95 };
97 {
98 std::reference_wrapper<const hardware_interface::LoanedStateInterface> position_state;
99 std::reference_wrapper<hardware_interface::LoanedCommandInterface> position_command;
100 };
101
102 CallbackReturn get_traction(
103 const std::string & traction_joint_name, std::vector<TractionHandle> & joint);
104 CallbackReturn get_steering(
105 const std::string & steering_joint_name, std::vector<SteeringHandle> & joint);
106 double convert_trans_rot_vel_to_steering_angle(double v, double omega, double wheelbase);
107 std::tuple<double, double> twist_to_ackermann(double linear_command, double angular_command);
108
109 std::string traction_joint_name_;
110 std::string steering_joint_name_;
111
112 // HACK: put into vector to avoid initializing structs because they have no default constructors
113 std::vector<TractionHandle> traction_joint_;
114 std::vector<SteeringHandle> steering_joint_;
115
117 {
118 double wheelbase = 0.0;
119 double radius = 0.0;
120 } wheel_params_;
121
123 {
124 bool open_loop = false;
125 bool enable_odom_tf = false;
126 bool odom_only_twist = false; // for doing the pose integration in separate node
127 std::string base_frame_id = "base_link";
128 std::string odom_frame_id = "odom";
129 std::array<double, 6> pose_covariance_diagonal;
130 std::array<double, 6> twist_covariance_diagonal;
131 } odom_params_;
132
133 bool publish_ackermann_command_ = false;
134 std::shared_ptr<rclcpp::Publisher<AckermannDrive>> ackermann_command_publisher_ = nullptr;
135 std::shared_ptr<realtime_tools::RealtimePublisher<AckermannDrive>>
136 realtime_ackermann_command_publisher_ = nullptr;
137
138 Odometry odometry_;
139
140 std::shared_ptr<rclcpp::Publisher<nav_msgs::msg::Odometry>> odometry_publisher_ = nullptr;
141 std::shared_ptr<realtime_tools::RealtimePublisher<nav_msgs::msg::Odometry>>
142 realtime_odometry_publisher_ = nullptr;
143
144 std::shared_ptr<rclcpp::Publisher<tf2_msgs::msg::TFMessage>> odometry_transform_publisher_ =
145 nullptr;
146 std::shared_ptr<realtime_tools::RealtimePublisher<tf2_msgs::msg::TFMessage>>
147 realtime_odometry_transform_publisher_ = nullptr;
148
149 // Timeout to consider cmd_vel commands old
150 std::chrono::milliseconds cmd_vel_timeout_{500};
151
152 bool subscriber_is_active_ = false;
153 rclcpp::Subscription<TwistStamped>::SharedPtr velocity_command_subscriber_ = nullptr;
154 rclcpp::Subscription<geometry_msgs::msg::Twist>::SharedPtr
155 velocity_command_unstamped_subscriber_ = nullptr;
156
157 realtime_tools::RealtimeBox<std::shared_ptr<TwistStamped>> received_velocity_msg_ptr_{nullptr};
158
159 rclcpp::Service<std_srvs::srv::Empty>::SharedPtr reset_odom_service_;
160
161 std::queue<AckermannDrive> previous_commands_; // last two commands
162
163 // speed limiters
164 TractionLimiter limiter_traction_;
165 SteeringLimiter limiter_steering_;
166
167 bool is_halted = false;
168 bool use_stamped_vel_ = true;
169
170 void reset_odometry(
171 const std::shared_ptr<rmw_request_id_t> request_header,
172 const std::shared_ptr<std_srvs::srv::Empty::Request> req,
173 std::shared_ptr<std_srvs::srv::Empty::Response> res);
174 bool reset();
175 void halt();
176};
177} // namespace tricycle_controller
178#endif // TRICYCLE_CONTROLLER__TRICYCLE_CONTROLLER_HPP_
Definition controller_interface.hpp:29
Definition realtime_box.hpp:47
Definition odometry.hpp:30
Definition tricycle_controller.hpp:53
TRICYCLE_CONTROLLER_PUBLIC controller_interface::InterfaceConfiguration state_interface_configuration() const override
Get configuration for controller's required state interfaces.
Definition tricycle_controller.cpp:109
TRICYCLE_CONTROLLER_PUBLIC controller_interface::InterfaceConfiguration command_interface_configuration() const override
Get configuration for controller's required command interfaces.
Definition tricycle_controller.cpp:100
TRICYCLE_CONTROLLER_PUBLIC CallbackReturn on_init() override
Extending interface with initialization method which is individual for each controller.
Definition tricycle_controller.cpp:51
TRICYCLE_CONTROLLER_PUBLIC controller_interface::return_type update(const rclcpp::Time &time, const rclcpp::Duration &period) override
Definition tricycle_controller.cpp:118
Configuring what command/state interfaces to claim.
Definition controller_interface_base.hpp:56
Definition tricycle_controller.hpp:123
Definition tricycle_controller.hpp:97
Definition tricycle_controller.hpp:92
Definition tricycle_controller.hpp:117