AStarPlanner.cpp
Go to the documentation of this file.
1 #include "AStarPlanner.h"
2 
3 #include <algorithm>
4 #include <cmath>
5 #include <cstddef>
6 #include <map>
7 #include <memory>
8 #include <vector>
9 
10 #include <boost/geometry.hpp> // IWYU pragma: keep
11 #include <boost/geometry/algorithms/detail/intersects/interface.hpp>
12 #include <boost/geometry/core/cs.hpp>
13 #include <boost/geometry/geometries/box.hpp>
14 #include <boost/geometry/geometries/point.hpp>
15 
16 #include <Eigen/Geometry>
17 
18 #include <VirtualRobot/BoundingBox.h>
19 
23 
26 
27 #include <omp.h>
28 #include <range/v3/algorithm/reverse.hpp>
29 
30 namespace bg = boost::geometry;
31 
33 {
34 
35  bool
36  intersects(const VirtualRobot::BoundingBox& bb1, const VirtualRobot::BoundingBox& bb2)
37  {
38  using point_t = bg::model::point<double, 3, bg::cs::cartesian>;
39  using box_t = bg::model::box<point_t>;
40 
41  const auto toPoint = [](const Eigen::Vector3f& pt)
42  { return point_t(pt.x(), pt.y(), pt.z()); };
43 
44  const box_t box1(toPoint(bb1.getMin()), toPoint(bb1.getMax()));
45  const box_t box2(toPoint(bb2.getMin()), toPoint(bb2.getMax()));
46 
47  return bg::intersects(box1, box2);
48  }
49 
50  AStarPlanner::AStarPlanner(const algorithms::Costmap& costmap) : costmap(costmap)
51  {
52  }
53 
54  float
55  AStarPlanner::heuristic(const NodePtr& n1, const NodePtr& n2)
56  {
57  return (1.F - obstacleDistanceWeightFactor) * (n1->position - n2->position).norm();
58  }
59 
60  void
61  AStarPlanner::createUniformGrid()
62  {
64 
65  const size_t rows = costmap.getGrid().rows();
66  const size_t cols = costmap.getGrid().cols();
67 
68  // create the grid with the correct size
69  ARMARX_VERBOSE << "Creating grid";
70  for (size_t r = 0; r < rows; r++)
71  {
72  grid.push_back(std::vector<NodePtr>(cols));
73  }
74 
75  // intitializing grid
76  for (size_t r = 0; r < rows; r++)
77  {
78  for (size_t c = 0; c < cols; c++)
79  {
80  const auto pos = costmap.toPositionGlobal({r, c});
81  const float obstacleDistance = costmap.getGrid()(r, c);
82 
83  grid[r][c] = std::make_shared<Node>(pos, obstacleDistance);
84  }
85  }
86 
87  ARMARX_VERBOSE << "Creating graph";
88 
89  // Init successors
90  for (size_t r = 0; r < rows; r++)
91  {
92  for (size_t c = 0; c < cols; c++)
93  {
94  NodePtr candidate = grid[r][c];
95  if (!fulfillsConstraints(candidate))
96  {
97  continue;
98  }
99 
100  // Add the valid node as successor to all its neighbors
101  for (int nR = -1; nR <= 1; nR++)
102  {
103  for (int nC = -1; nC <= 1; nC++)
104  {
105  const int neighborIndexC = c + nC;
106  const int neighborIndexR = r + nR;
107  if (neighborIndexC < 0 || neighborIndexR < 0 || (nR == 0 && nC == 0))
108  {
109  continue;
110  }
111 
112  if (neighborIndexR >= static_cast<int>(rows) ||
113  neighborIndexC >= static_cast<int>(cols))
114  {
115  continue;
116  }
117 
118  grid[neighborIndexR][neighborIndexC]->successors.push_back(candidate);
119  }
120  }
121  }
122  }
123 
124  ARMARX_VERBOSE << "Done.";
125  }
126 
127  bool
128  AStarPlanner::fulfillsConstraints(const NodePtr& n)
129  {
130  ARMARX_TRACE;
132 
133  return not costmap.isInCollision(n->position);
134  }
135 
136  NodePtr
137  AStarPlanner::closestNode(const Eigen::Vector2f& v)
138  {
139  const auto vertex = costmap.toVertex(v);
140 
141  const int r = vertex.index.x();
142  const int c = vertex.index.y();
143 
146 
147  const int rows = static_cast<int>(costmap.getGrid().rows());
148  const int cols = static_cast<int>(costmap.getGrid().cols());
149 
150  ARMARX_CHECK_LESS_EQUAL(r, rows - 1);
151  ARMARX_CHECK_LESS_EQUAL(c, cols - 1);
152 
153  return grid[static_cast<int>(r)][static_cast<int>(c)];
154  }
155 
156  std::vector<Eigen::Vector2f>
157  AStarPlanner::plan(const Eigen::Vector2f& start, const Eigen::Vector2f& goal)
158  {
159  ARMARX_TRACE;
160  grid.clear();
161  std::vector<Eigen::Vector2f> result;
162 
163  createUniformGrid();
164  ARMARX_DEBUG << "Setting start node for position " << start;
165  NodePtr nodeStart = closestNode(start);
166  // FIXME activate ARMARX_CHECK(fulfillsConstraints(nodeStart)) << "Start node in collision!";
167 
168  ARMARX_DEBUG << "Setting goal node for position " << goal;
169  NodePtr nodeGoal = closestNode(goal);
170  ARMARX_CHECK(fulfillsConstraints(nodeGoal)) << "Goal node in collision!";
171 
172  std::vector<NodePtr> closedSet;
173 
174  std::vector<NodePtr> openSet;
175  openSet.push_back(nodeStart);
176 
177  std::map<NodePtr, float> gScore;
178  gScore[nodeStart] = 0;
179 
180  std::map<NodePtr, float> fScore;
181  fScore[nodeStart] = gScore[nodeStart] + heuristic(nodeStart, nodeGoal);
182 
183  while (!openSet.empty())
184  {
185  float lowestScore = fScore[openSet[0]];
186  auto currentIT = openSet.begin();
187  for (auto it = openSet.begin() + 1; it != openSet.end(); it++)
188  {
189  if (fScore[*it] < lowestScore)
190  {
191  lowestScore = fScore[*it];
192  currentIT = it;
193  }
194  }
195  NodePtr currentBest = *currentIT;
196  if (currentBest == nodeGoal)
197  {
198  break;
199  }
200 
201  openSet.erase(currentIT);
202  closedSet.push_back(currentBest);
203  for (size_t i = 0; i < currentBest->successors.size(); i++)
204  {
205  NodePtr neighbor = currentBest->successors[i];
206  if (std::find(closedSet.begin(), closedSet.end(), neighbor) != closedSet.end())
207  {
208  continue;
209  }
210 
211  float tentativeGScore = gScore[currentBest] + costs(currentBest, neighbor);
212  bool notInOS = std::find(openSet.begin(), openSet.end(), neighbor) == openSet.end();
213  if (notInOS || tentativeGScore < gScore[neighbor])
214  {
215  neighbor->predecessor = currentBest;
216  gScore[neighbor] = tentativeGScore;
217  fScore[neighbor] = tentativeGScore + heuristic(neighbor, nodeGoal);
218  if (notInOS)
219  {
220  openSet.push_back(neighbor);
221  }
222  }
223  }
224  }
225 
226  // Found solution, now retrieve path from goal to start
227  if (nodeGoal)
228  {
229  result = nodeGoal->traversePredecessors();
230  }
231 
232  // Since the graph was traversed from goal to start, we have to reverse the order
233  ranges::reverse(result);
234 
235  return result;
236  }
237 
238  float
239  AStarPlanner::costs(const NodePtr& n1, const NodePtr& n2) const
240  {
241  const float euclideanDistance = (n1->position - n2->position).norm();
242 
243  // additional costs if
244  const float obstacleDistanceN1 = std::min(n1->obstacleDistance, maxObstacleDistance);
245  const float obstacleDistanceN2 = std::min(n2->obstacleDistance, maxObstacleDistance);
246 
247  // const float obstacleProximityChangeCosts = std::max(obstacleDistanceN1 - obstacleDistanceN2, 0.F);
248  const float obstacleProximityChangeCosts = obstacleDistanceN1 - obstacleDistanceN2;
249 
250  const float costs =
251  euclideanDistance + obstacleProximityChangeCosts * obstacleDistanceWeightFactor;
252  return std::max(costs, 0.F);
253  }
254 
255 
256 } // namespace armarx::navigation::algorithm::astar
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
armarx::navigation::algorithm::astar::intersects
bool intersects(const VirtualRobot::BoundingBox &bb1, const VirtualRobot::BoundingBox &bb2)
Definition: AStarPlanner.cpp:36
ARMARX_CHECK_NOT_NULL
#define ARMARX_CHECK_NOT_NULL(ptr)
This macro evaluates whether ptr is not null and if it turns out to be false it will throw an Express...
Definition: ExpressionException.h:206
trace.h
armarx::navigation::algorithms::Costmap::Vertex::index
Index index
Definition: Costmap.h:69
c
constexpr T c
Definition: UnscentedKalmanFilterTest.cpp:46
armarx::navigation::algorithms::Costmap::toPositionGlobal
Position toPositionGlobal(const Index &index) const
Definition: Costmap.cpp:69
armarx::navigation::algorithm::astar::AStarPlanner::plan
std::vector< Eigen::Vector2f > plan(const Eigen::Vector2f &start, const Eigen::Vector2f &goal)
Definition: AStarPlanner.cpp:157
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
Costmap.h
armarx::reverse
T reverse(const T &o)
Definition: container.h:33
armarx::navigation::algorithm::astar::NodePtr
std::shared_ptr< Node > NodePtr
Definition: Node.h:11
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
armarx::navigation::algorithm::astar::AStarPlanner::AStarPlanner
EIGEN_MAKE_ALIGNED_OPERATOR_NEW AStarPlanner(const algorithms::Costmap &costmap)
Definition: AStarPlanner.cpp:50
AStarPlanner.h
ARMARX_DEBUG
#define ARMARX_DEBUG
Definition: Logging.h:184
armarx::navigation::algorithm::astar
Definition: AStarPlanner.cpp:32
Node.h
armarx::navigation::algorithms::Costmap::toVertex
Vertex toVertex(const Position &globalPosition) const
Definition: Costmap.cpp:103
armarx::navigation::algorithms::Costmap::getGrid
const Grid & getGrid() const
Definition: Costmap.cpp:243
max
T max(T t1, T t2)
Definition: gdiam.h:51
ARMARX_CHECK_GREATER_EQUAL
#define ARMARX_CHECK_GREATER_EQUAL(lhs, rhs)
This macro evaluates whether lhs is greater or equal (>=) rhs and if it turns out to be false it will...
Definition: ExpressionException.h:123
ARMARX_CHECK_LESS_EQUAL
#define ARMARX_CHECK_LESS_EQUAL(lhs, rhs)
This macro evaluates whether lhs is less or equal (<=) rhs and if it turns out to be false it will th...
Definition: ExpressionException.h:109
ExpressionException.h
armarx::euclideanDistance
float euclideanDistance(IteratorType1 first1, IteratorType1 last1, IteratorType2 first2)
Returns the euclidean distance.
Definition: Metrics.h:104
armarx::ctrlutil::v
double v(double t, double v0, double a0, double j)
Definition: CtrlUtil.h:39
F
Definition: ExportDialogControllerTest.cpp:18
Logging.h
min
T min(T t1, T t2)
Definition: gdiam.h:44
armarx::navigation::util::geometry::toPoint
point_type toPoint(const Eigen::Vector2f &pt)
Definition: geometry.cpp:43
armarx::navigation::algorithms::Costmap::isInCollision
bool isInCollision(const Position &p) const
Definition: Costmap.cpp:249
armarx::navigation::algorithms::Costmap
Definition: Costmap.h:16
norm
double norm(const Point &a)
Definition: point.hpp:102