Nav2 Navigation Stack - rolling  main
ROS 2 Navigation Stack
costmap_scorer.cpp
1 // Copyright (c) 2025 Open Navigation LLC
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 #include <memory>
16 #include <string>
17 
18 #include "nav2_route/plugins/edge_cost_functions/costmap_scorer.hpp"
19 
20 namespace nav2_route
21 {
22 
24  const nav2::LifecycleNode::SharedPtr node,
25  const std::shared_ptr<tf2_ros::Buffer>/* tf_buffer */,
26  std::shared_ptr<nav2_costmap_2d::CostmapSubscriber> costmap_subscriber,
27  const std::string & name)
28 {
29  RCLCPP_INFO(node->get_logger(), "Configuring costmap scorer.");
30  name_ = name;
31  logger_ = node->get_logger();
32  clock_ = node->get_clock();
33 
34  // Find whether to use average or maximum cost values
35  nav2::declare_parameter_if_not_declared(
36  node, getName() + ".use_maximum", rclcpp::ParameterValue(true));
37  use_max_ = static_cast<float>(node->get_parameter(getName() + ".use_maximum").as_bool());
38 
39  // Edge is invalid if its in collision
40  nav2::declare_parameter_if_not_declared(
41  node, getName() + ".invalid_on_collision", rclcpp::ParameterValue(true));
42  invalid_on_collision_ =
43  static_cast<float>(node->get_parameter(getName() + ".invalid_on_collision").as_bool());
44 
45  // Edge is invalid if edge is off the costmap
46  nav2::declare_parameter_if_not_declared(
47  node, getName() + ".invalid_off_map", rclcpp::ParameterValue(true));
48  invalid_off_map_ =
49  static_cast<float>(node->get_parameter(getName() + ".invalid_off_map").as_bool());
50 
51  // Max cost to be considered valid
52  nav2::declare_parameter_if_not_declared(
53  node, getName() + ".max_cost", rclcpp::ParameterValue(253.0));
54  max_cost_ = static_cast<float>(node->get_parameter(getName() + ".max_cost").as_double());
55 
56  // Resolution to check the costmap over (1=every cell, 2=every other cell, etc.)
57  nav2::declare_parameter_if_not_declared(
58  node, getName() + ".check_resolution", rclcpp::ParameterValue(2));
59  check_resolution_ = static_cast<unsigned int>(
60  node->get_parameter(getName() + ".check_resolution").as_int());
61 
62  // Create costmap subscriber if not the same as the server costmap
63  std::string server_costmap_topic = node->get_parameter("costmap_topic").as_string();
64  nav2::declare_parameter_if_not_declared(
65  node, getName() + ".costmap_topic",
66  rclcpp::ParameterValue(std::string("global_costmap/costmap_raw")));
67  std::string costmap_topic =
68  node->get_parameter(getName() + ".costmap_topic").as_string();
69  if (costmap_topic != server_costmap_topic) {
70  costmap_subscriber_ = std::make_shared<nav2_costmap_2d::CostmapSubscriber>(
71  node, costmap_topic);
72  RCLCPP_INFO(node->get_logger(),
73  "Using costmap topic: %s instead of server costmap topic: %s for CostmapScorer.",
74  costmap_topic.c_str(), server_costmap_topic.c_str());
75  } else {
76  costmap_subscriber_ = costmap_subscriber;
77  }
78 
79  // Find the proportional weight to apply, if multiple cost functions
80  nav2::declare_parameter_if_not_declared(
81  node, getName() + ".weight", rclcpp::ParameterValue(1.0));
82  weight_ = static_cast<float>(node->get_parameter(getName() + ".weight").as_double());
83 }
84 
86 {
87  try {
88  costmap_ = costmap_subscriber_->getCostmap();
89  } catch (...) {
90  costmap_.reset();
91  }
92 }
93 
95  const EdgePtr edge,
96  const RouteRequest & /* route_request */,
97  const EdgeType & /* edge_type */, float & cost)
98 {
99  if (!costmap_) {
100  RCLCPP_WARN_THROTTLE(logger_, *clock_, 1000, "No costmap yet received!");
101  return false;
102  }
103 
104  float largest_cost = 0.0, running_cost = 0.0, point_cost = 0.0;
105  unsigned int x0, y0, x1, y1, idx = 0;
106  if (!costmap_->worldToMap(edge->start->coords.x, edge->start->coords.y, x0, y0) ||
107  !costmap_->worldToMap(edge->end->coords.x, edge->end->coords.y, x1, y1))
108  {
109  if (invalid_off_map_) {
110  // Edge is invalid if it is off the costmap
111  return false;
112  }
113  return true;
114  }
115 
116  for (nav2_util::LineIterator iter(x0, y0, x1, y1); iter.isValid(); ) {
117  point_cost = static_cast<float>(costmap_->getCost(iter.getX(), iter.getY()));
118  if (point_cost >= max_cost_ && max_cost_ != 255.0f /*Unknown*/ && invalid_on_collision_) {
119  // Edge is invalid if it is in collision or higher than max allowed cost
120  return false;
121  }
122 
123  idx++;
124  running_cost += point_cost;
125  if (largest_cost < point_cost && point_cost != 255.0) {
126  largest_cost = point_cost;
127  }
128 
129  // Advance the iterator by the check resolution on the edge, pruning to a coarse resolution
130  for (unsigned int i = 0; i < check_resolution_; i++) {
131  iter.advance();
132  }
133  }
134 
135  if (use_max_) {
136  cost = weight_ * largest_cost / max_cost_;
137  } else {
138  cost = weight_ * running_cost / (static_cast<float>(idx) * max_cost_);
139  }
140 
141  return true;
142 }
143 
145 {
146  return name_;
147 }
148 
149 } // namespace nav2_route
150 
151 #include "pluginlib/class_list_macros.hpp"
Scores edges by the average or maximum cost found while iterating over the edge's line segment in the...
std::string getName() override
Get name of the plugin for parameter scope mapping.
void prepare() override
Prepare for a new cycle, by resetting state, grabbing data to use for all immediate requests,...
bool score(const EdgePtr edge, const RouteRequest &route_request, const EdgeType &edge_type, float &cost) override
Main scoring plugin API.
void configure(const nav2::LifecycleNode::SharedPtr node, const std::shared_ptr< tf2_ros::Buffer > tf_buffer, std::shared_ptr< nav2_costmap_2d::CostmapSubscriber > costmap_subscriber, const std::string &name) override
Configure.
A plugin interface to score edges during graph search to modify the lowest cost path (e....
An iterator implementing Bresenham Ray-Tracing.
bool isValid() const
If the iterator is valid.
An object representing edges between nodes.
Definition: types.hpp:134
An object to store salient features of the route request including its start and goal node ids,...
Definition: types.hpp:224