Franky 0.12.0
A High-Level Motion API for Franka
Loading...
Searching...
No Matches
waypoint_motion.hpp
Go to the documentation of this file.
1#pragma once
2
3#include <optional>
4#include <ruckig/ruckig.hpp>
5
7
8namespace franky {
9
13struct MotionPlannerException : std::runtime_error {
14 using std::runtime_error::runtime_error;
15};
16
37template <typename TargetType>
38struct Waypoint {
40
42
43 std::optional<franka::Duration> minimum_time{std::nullopt};
44
45 franka::Duration hold_target_duration{0};
46
47 std::optional<franka::Duration> max_total_duration{std::nullopt};
48
50};
51
59template <typename ControlSignalType, typename WaypointType, typename TargetType>
60class WaypointMotion : public Motion<ControlSignalType> {
61 static_assert(
62 std::is_base_of_v<Waypoint<TargetType>, WaypointType>, "WaypointType must inherit from Waypoint<TargetType>");
63
64 public:
70 explicit WaypointMotion(std::vector<WaypointType> waypoints, bool return_when_finished = true)
71 : waypoints_(std::move(waypoints)), return_when_finished_(return_when_finished), prev_result_() {}
72
73 protected:
74 void initImpl(const RobotState &robot_state, const std::optional<ControlSignalType> &previous_command) override {
75 target_reached_time_ = std::nullopt;
76 waypoint_started_time_ = franka::Duration(0);
77
79
80 waypoint_iterator_ = waypoints_.begin();
81 if (waypoint_iterator_ != waypoints_.end()) {
82 checkWaypointPrivate(*waypoint_iterator_);
83 setNewWaypoint(robot_state, previous_command, *waypoint_iterator_, input_parameter_);
84 setInputLimits(*waypoint_iterator_, input_parameter_);
85 prev_result_ = ruckig::Result::Working;
86 } else {
87 prev_result_ = ruckig::Result::Finished;
88 }
89 }
90
92 const RobotState &robot_state, franka::Duration time_step, franka::Duration rel_time, franka::Duration abs_time,
93 const std::optional<ControlSignalType> &previous_command) override {
94 const auto expected_time_step = franka::Duration(1);
96 // In this case, we missed a couple of steps for some reason. Hence, extrapolate the way the robot does if it
97 // does not receive data (constant acceleration model).
98 // See https://frankaemika.github.io/docs/libfranka.html#under-the-hood
99 extrapolateMotion(time_step - expected_time_step, input_parameter_, output_parameter_);
100 output_parameter_.pass_to_input(input_parameter_);
101 }
102 auto max_time_reached = false;
103 if (waypoint_iterator_ != waypoints_.end() && waypoint_iterator_->max_total_duration.has_value()) {
104 max_time_reached = rel_time - waypoint_started_time_ >= waypoint_iterator_->max_total_duration.value();
105 }
106 if (prev_result_ == ruckig::Result::Finished || max_time_reached) {
107 if (!target_reached_time_.has_value()) {
108 target_reached_time_ = rel_time;
109 }
110 auto hold_target_duration = waypoint_iterator_->hold_target_duration;
111 if (waypoint_iterator_ + 1 == waypoints_.end()) {
112 // Allow cooldown of motion, so that the low-pass filter has time to adjust to target values
113 hold_target_duration = std::max(hold_target_duration, franka::Duration(5));
114 }
115 if (rel_time - target_reached_time_.value() >= hold_target_duration || max_time_reached) {
116 target_reached_time_ = std::nullopt;
117 if (waypoint_iterator_ != waypoints_.end()) {
118 ++waypoint_iterator_;
119 }
120 }
121 if (waypoint_iterator_ == waypoints_.end()) {
123 if (!return_when_finished_) return command;
124 return franka::MotionFinished(command);
125 } else {
126 checkWaypointPrivate(*waypoint_iterator_);
127 setNewWaypoint(robot_state, previous_command, *waypoint_iterator_, input_parameter_);
128 setInputLimits(*waypoint_iterator_, input_parameter_);
129 waypoint_started_time_ = rel_time;
130 }
131 }
132 if (waypoint_iterator_ != waypoints_.end()) {
133 auto blend = waypoint_iterator_->state_estimate_weight;
134
135 if (blend != 0) {
137 auto pos_t = toEigen(input_parameter_.current_position);
138 auto vel_t = toEigen(input_parameter_.current_velocity);
139 auto acc_t = toEigen(input_parameter_.current_acceleration);
140
141 auto pos_b = (1 - blend) * pos_t + blend * pos_s;
142 auto vel_b = (1 - blend) * vel_t + blend * vel_s;
143 auto acc_b = (1 - blend) * acc_t + blend * acc_s;
144
145 input_parameter_.current_position = toStdD<7>(pos_b);
146 input_parameter_.current_velocity = toStdD<7>(vel_b);
147 input_parameter_.current_acceleration = toStdD<7>(acc_b);
148 }
149
150 prev_result_ = trajectory_generator_.update(input_parameter_, output_parameter_);
151 if (prev_result_ == ruckig::Result::Working || prev_result_ == ruckig::Result::Finished) {
152 output_parameter_.pass_to_input(input_parameter_);
153 } else {
154 throw MotionPlannerException("Motion planner failed with error code " + std::to_string(prev_result_));
155 }
156 }
157
158 return getControlSignal(robot_state, time_step, previous_command, input_parameter_);
159 };
160
161 virtual void initWaypointMotion(
162 const RobotState &robot_state, const std::optional<ControlSignalType> &previous_command,
163 ruckig::InputParameter<7> &input_parameter) = 0;
164
165 virtual void setNewWaypoint(
166 const RobotState &robot_state, const std::optional<ControlSignalType> &previous_command,
167 const WaypointType &new_waypoint, ruckig::InputParameter<7> &input_parameter) = 0;
168
169 virtual void extrapolateMotion(
170 const franka::Duration &time_step, const ruckig::InputParameter<7> &input_parameter,
171 ruckig::OutputParameter<7> &output_parameter) const = 0;
172
173 virtual void checkWaypoint(const WaypointType &waypoint) const {}
174
175 [[nodiscard]] virtual std::tuple<Vector7d, Vector7d, Vector7d> getStateEstimate(
176 const RobotState &robot_state) const = 0;
177
178 [[nodiscard]] virtual std::tuple<Vector7d, Vector7d, Vector7d> getAbsoluteInputLimits() const = 0;
179
181 const RobotState &robot_state, const franka::Duration &time_step,
182 const std::optional<ControlSignalType> &previous_command, const ruckig::InputParameter<7> &input_parameter) = 0;
183
184 virtual void setInputLimits(const WaypointType &waypoint, ruckig::InputParameter<7> &input_parameter) const = 0;
185
186 private:
187 std::vector<WaypointType> waypoints_;
188 bool return_when_finished_{true};
189
190 ruckig::Ruckig<7> trajectory_generator_{Robot::control_rate};
191 ruckig::Result prev_result_;
192 ruckig::InputParameter<7> input_parameter_;
193 ruckig::OutputParameter<7> output_parameter_;
194
195 typename std::vector<WaypointType>::iterator waypoint_iterator_;
196
197 std::optional<franka::Duration> target_reached_time_;
198 franka::Duration waypoint_started_time_;
199
200 void checkWaypointPrivate(const WaypointType &waypoint) const {
201 if (waypoint.state_estimate_weight < 0 || waypoint.state_estimate_weight > 1)
202 throw std::runtime_error(
203 "state_estimate_weight must be between 0 and 1. Got " + std::to_string(waypoint.state_estimate_weight));
205 }
206};
207
208} // namespace franky
Base class for motions.
Definition motion.hpp:23
Relative dynamics factors.
Definition relative_dynamics_factor.hpp:13
static constexpr double control_rate
Definition robot.hpp:98
A motion following multiple waypoints in a time-optimal way. Works with arbitrary initial conditions.
Definition waypoint_motion.hpp:60
virtual std::tuple< Vector7d, Vector7d, Vector7d > getStateEstimate(const RobotState &robot_state) const =0
virtual void initWaypointMotion(const RobotState &robot_state, const std::optional< ControlSignalType > &previous_command, ruckig::InputParameter< 7 > &input_parameter)=0
virtual void setInputLimits(const WaypointType &waypoint, ruckig::InputParameter< 7 > &input_parameter) const =0
WaypointMotion(std::vector< WaypointType > waypoints, bool return_when_finished=true)
Definition waypoint_motion.hpp:70
virtual void checkWaypoint(const WaypointType &waypoint) const
Definition waypoint_motion.hpp:173
virtual ControlSignalType getControlSignal(const RobotState &robot_state, const franka::Duration &time_step, const std::optional< ControlSignalType > &previous_command, const ruckig::InputParameter< 7 > &input_parameter)=0
virtual void setNewWaypoint(const RobotState &robot_state, const std::optional< ControlSignalType > &previous_command, const WaypointType &new_waypoint, ruckig::InputParameter< 7 > &input_parameter)=0
void initImpl(const RobotState &robot_state, const std::optional< ControlSignalType > &previous_command) override
Definition waypoint_motion.hpp:74
ControlSignalType nextCommandImpl(const RobotState &robot_state, franka::Duration time_step, franka::Duration rel_time, franka::Duration abs_time, const std::optional< ControlSignalType > &previous_command) override
Definition waypoint_motion.hpp:91
virtual std::tuple< Vector7d, Vector7d, Vector7d > getAbsoluteInputLimits() const =0
virtual void extrapolateMotion(const franka::Duration &time_step, const ruckig::InputParameter< 7 > &input_parameter, ruckig::OutputParameter< 7 > &output_parameter) const =0
Definition dynamics_limit.cpp:8
std::array< double, dims > toStdD(const Eigen::Matrix< double, dims, 1 > &vector)
Definition util.hpp:18
Eigen::Matrix< T, dims, 1 > toEigen(const std::array< T, dims > &vector)
Definition util.hpp:23
ControlSignalType
Type of control signal.
Definition control_signal_type.hpp:8
Exception thrown if the motion planner fails.
Definition waypoint_motion.hpp:13
Full state of the robot.
Definition robot_state.hpp:20
A waypoint with a target and optional parameters.
Definition waypoint_motion.hpp:38
std::optional< franka::Duration > max_total_duration
Definition waypoint_motion.hpp:47
franka::Duration hold_target_duration
Definition waypoint_motion.hpp:45
double state_estimate_weight
Definition waypoint_motion.hpp:49
TargetType target
Definition waypoint_motion.hpp:39
RelativeDynamicsFactor relative_dynamics_factor
Definition waypoint_motion.hpp:41
std::optional< franka::Duration > minimum_time
Definition waypoint_motion.hpp:43