moveit2
The MoveIt Motion Planning Framework for ROS 2.
Loading...
Searching...
No Matches
collision_env.cpp
Go to the documentation of this file.
1/*********************************************************************
2 * Software License Agreement (BSD License)
3 *
4 * Copyright (c) 2011, Willow Garage, Inc.
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 *
11 * * Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * * Redistributions in binary form must reproduce the above
14 * copyright notice, this list of conditions and the following
15 * disclaimer in the documentation and/or other materials provided
16 * with the distribution.
17 * * Neither the name of the copyright holder nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific prior written permission.
20 *
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
29 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
31 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32 * POSSIBILITY OF SUCH DAMAGE.
33 *********************************************************************/
34
35/* Author: Ioan Sucan, Jens Petit */
36
38#include <rclcpp/logger.hpp>
39#include <rclcpp/logging.hpp>
40#include <limits>
42
43namespace
44{
45rclcpp::Logger getLogger()
46{
47 return moveit::getLogger("moveit.core.collision_detection_env");
48}
49} // namespace
50
51static inline bool validateScale(const double scale)
52{
53 if (scale < std::numeric_limits<double>::epsilon())
54 {
55 RCLCPP_ERROR(getLogger(), "Scale must be positive");
56 return false;
57 }
58 if (scale > std::numeric_limits<double>::max())
59 {
60 RCLCPP_ERROR(getLogger(), "Scale must be finite");
61 return false;
62 }
63 return true;
64}
65
66static inline bool validatePadding(const double padding)
67{
68 if (padding < 0.0)
69 {
70 RCLCPP_ERROR(getLogger(), "Padding cannot be negative");
71 return false;
72 }
73 if (padding > std::numeric_limits<double>::max())
74 {
75 RCLCPP_ERROR(getLogger(), "Padding must be finite");
76 return false;
77 }
78 return true;
79}
80
81namespace collision_detection
82{
83CollisionEnv::CollisionEnv(const moveit::core::RobotModelConstPtr& model, double padding, double scale)
84 : robot_model_(model), world_(std::make_shared<World>()), world_const_(world_)
85{
86 if (!validateScale(scale))
87 scale = 1.0;
88 if (!validatePadding(padding))
89 padding = 0.0;
90
91 const std::vector<const moveit::core::LinkModel*>& links = robot_model_->getLinkModelsWithCollisionGeometry();
92 for (const auto link : links)
93 {
94 link_padding_[link->getName()] = padding;
95 link_scale_[link->getName()] = scale;
96 }
97}
98
99CollisionEnv::CollisionEnv(const moveit::core::RobotModelConstPtr& model, const WorldPtr& world, double padding,
100 double scale)
101 : robot_model_(model), world_(world), world_const_(world_)
102{
103 if (!validateScale(scale))
104 scale = 1.0;
105 if (!validatePadding(padding))
106 padding = 0.0;
107
108 const std::vector<const moveit::core::LinkModel*>& links = robot_model_->getLinkModelsWithCollisionGeometry();
109 for (const auto link : links)
110 {
111 link_padding_[link->getName()] = padding;
112 link_scale_[link->getName()] = scale;
113 }
114}
115
116CollisionEnv::CollisionEnv(const CollisionEnv& other, const WorldPtr& world)
117 : robot_model_(other.robot_model_), world_(world), world_const_(world)
118{
120 link_scale_ = other.link_scale_;
121}
122void CollisionEnv::setPadding(const double padding)
123{
124 if (!validatePadding(padding))
125 return;
126 std::vector<std::string> u;
127 const std::vector<const moveit::core::LinkModel*>& links = robot_model_->getLinkModelsWithCollisionGeometry();
128 for (const auto link : links)
129 {
130 if (getLinkPadding(link->getName()) != padding)
131 u.push_back(link->getName());
132 link_padding_[link->getName()] = padding;
133 }
134 if (!u.empty())
136}
137
138void CollisionEnv::setScale(const double scale)
139{
140 if (!validateScale(scale))
141 return;
142 std::vector<std::string> u;
143 const std::vector<const moveit::core::LinkModel*>& links = robot_model_->getLinkModelsWithCollisionGeometry();
144 for (const auto link : links)
145 {
146 if (getLinkScale(link->getName()) != scale)
147 u.push_back(link->getName());
148 link_scale_[link->getName()] = scale;
149 }
150 if (!u.empty())
152}
153
154void CollisionEnv::setLinkPadding(const std::string& link_name, const double padding)
155{
156 validatePadding(padding);
157 const bool update = getLinkPadding(link_name) != padding;
158 link_padding_[link_name] = padding;
159 if (update)
160 {
161 std::vector<std::string> u(1, link_name);
163 }
164}
165
166double CollisionEnv::getLinkPadding(const std::string& link_name) const
167{
168 auto it = link_padding_.find(link_name);
169 if (it != link_padding_.end())
170 {
171 return it->second;
172 }
173 else
174 {
175 return 0.0;
176 }
177}
178
179void CollisionEnv::setLinkPadding(const std::map<std::string, double>& padding)
180{
181 std::vector<std::string> u;
182 for (const auto& link_pad_pair : padding)
183 {
184 validatePadding(link_pad_pair.second);
185 const bool update = getLinkPadding(link_pad_pair.first) != link_pad_pair.second;
186 link_padding_[link_pad_pair.first] = link_pad_pair.second;
187 if (update)
188 u.push_back(link_pad_pair.first);
189 }
190 if (!u.empty())
192}
193
194const std::map<std::string, double>& CollisionEnv::getLinkPadding() const
195{
196 return link_padding_;
197}
198
199void CollisionEnv::setLinkScale(const std::string& link_name, const double scale)
200{
201 validateScale(scale);
202 const bool update = getLinkScale(link_name) != scale;
203 link_scale_[link_name] = scale;
204 if (update)
205 {
206 std::vector<std::string> u(1, link_name);
208 }
209}
210
211double CollisionEnv::getLinkScale(const std::string& link_name) const
212{
213 const auto it = link_scale_.find(link_name);
214 if (it != link_scale_.end())
215 {
216 return it->second;
217 }
218 else
219 {
220 return 1.0;
221 }
222}
223
224void CollisionEnv::setLinkScale(const std::map<std::string, double>& scale)
225{
226 std::vector<std::string> u;
227 for (const auto& link_scale_pair : scale)
228 {
229 const bool update = getLinkScale(link_scale_pair.first) != link_scale_pair.second;
230 link_scale_[link_scale_pair.first] = link_scale_pair.second;
231 if (update)
232 u.push_back(link_scale_pair.first);
233 }
234 if (!u.empty())
236}
237
238const std::map<std::string, double>& CollisionEnv::getLinkScale() const
239{
240 return link_scale_;
241}
242
243void CollisionEnv::setPadding(const std::vector<moveit_msgs::msg::LinkPadding>& padding)
244{
245 std::vector<std::string> u;
246 for (const auto& p : padding)
247 {
248 validatePadding(p.padding);
249 const bool update = getLinkPadding(p.link_name) != p.padding;
250 link_padding_[p.link_name] = p.padding;
251 if (update)
252 u.push_back(p.link_name);
253 }
254 if (!u.empty())
256}
257
258void CollisionEnv::setScale(const std::vector<moveit_msgs::msg::LinkScale>& scale)
259{
260 std::vector<std::string> u;
261 for (const auto& s : scale)
262 {
263 validateScale(s.scale);
264 const bool update = getLinkScale(s.link_name) != s.scale;
265 link_scale_[s.link_name] = s.scale;
266 if (update)
267 u.push_back(s.link_name);
268 }
269 if (!u.empty())
271}
272
273void CollisionEnv::getPadding(std::vector<moveit_msgs::msg::LinkPadding>& padding) const
274{
275 padding.clear();
276 for (const auto& lp : link_padding_)
277 {
278 moveit_msgs::msg::LinkPadding lp_msg;
279 lp_msg.link_name = lp.first;
280 lp_msg.padding = lp.second;
281 padding.push_back(lp_msg);
282 }
283}
284
285void CollisionEnv::getScale(std::vector<moveit_msgs::msg::LinkScale>& scale) const
286{
287 scale.clear();
288 for (const auto& ls : link_scale_)
289 {
290 moveit_msgs::msg::LinkScale ls_msg;
291 ls_msg.link_name = ls.first;
292 ls_msg.scale = ls.second;
293 scale.push_back(ls_msg);
294 }
295}
296
297void CollisionEnv::updatedPaddingOrScaling(const std::vector<std::string>& /*links*/)
298{
299}
300
301void CollisionEnv::setWorld(const WorldPtr& world)
302{
303 world_ = world;
304 if (!world_)
305 world_ = std::make_shared<World>();
306
307 world_const_ = world;
308}
309
311 const moveit::core::RobotState& state) const
312{
313 checkSelfCollision(req, res, state);
314 if (!res.collision || (req.contacts && res.contacts.size() < req.max_contacts))
315 checkRobotCollision(req, res, state);
316}
317
319 const moveit::core::RobotState& state, const AllowedCollisionMatrix& acm) const
320{
321 checkSelfCollision(req, res, state, acm);
322 if (!res.collision || (req.contacts && res.contacts.size() < req.max_contacts))
323 checkRobotCollision(req, res, state, acm);
324}
325} // end of namespace collision_detection
Definition of a structure for the allowed collision matrix. All elements in the collision world are r...
Provides the interface to the individual collision checking libraries.
virtual void setWorld(const WorldPtr &world)
void getPadding(std::vector< moveit_msgs::msg::LinkPadding > &padding) const
Get the link padding as a vector of messages.
void setLinkScale(const std::string &link_name, double scale)
Set the scaling for a particular link.
void setLinkPadding(const std::string &link_name, double padding)
Set the link padding for a particular link.
moveit::core::RobotModelConstPtr robot_model_
The kinematic model corresponding to this collision model.
virtual void checkSelfCollision(const CollisionRequest &req, CollisionResult &res, const moveit::core::RobotState &state) const =0
Check for robot self collision. Any collision between any pair of links is checked for,...
const std::map< std::string, double > & getLinkScale() const
Get the link scaling as a map (from link names to scale value)
void setScale(double scale)
Set the link scaling (for every link)
void setPadding(double padding)
Set the link padding (for every link)
virtual void checkCollision(const CollisionRequest &req, CollisionResult &res, const moveit::core::RobotState &state) const
Check whether the robot model is in collision with itself or the world at a particular state....
void getScale(std::vector< moveit_msgs::msg::LinkScale > &scale) const
Get the link scaling as a vector of messages.
std::map< std::string, double > link_padding_
The internally maintained map (from link names to padding)
virtual void checkRobotCollision(const CollisionRequest &req, CollisionResult &res, const moveit::core::RobotState &state) const =0
Check whether the robot model is in collision with the world. Any collisions between a robot link and...
const std::map< std::string, double > & getLinkPadding() const
Get the link paddings as a map (from link names to padding value)
virtual void updatedPaddingOrScaling(const std::vector< std::string > &links)
When the scale or padding is changed for a set of links by any of the functions in this class,...
std::map< std::string, double > link_scale_
The internally maintained map (from link names to scaling)
Maintain a representation of the environment.
Definition world.hpp:59
Representation of a robot's state. This includes position, velocity, acceleration and effort.
rclcpp::Logger getLogger(const std::string &name)
Creates a namespaced logger.
Definition logger.cpp:79
Representation of a collision checking request.
bool contacts
If true, compute contacts. Otherwise only a binary collision yes/no is reported.
std::size_t max_contacts
Overall maximum number of contacts to compute.
Representation of a collision checking result.
bool collision
True if collision was found, false otherwise.