util.cpp
Go to the documentation of this file.
1 /**
2  * This file is part of ArmarX.
3  *
4  * ArmarX is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License version 2 as
6  * published by the Free Software Foundation.
7  *
8  * ArmarX is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11  * GNU General Public License for more details.
12  *
13  * You should have received a copy of the GNU General Public License
14  * along with this program. If not, see <http://www.gnu.org/licenses/>.
15  *
16  * @author Fabian Reister ( fabian dot reister at kit dot edu )
17  * @date 2022
18  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
19  * GNU General Public License
20  */
21 
22 #include "util.h"
23 
24 #include <algorithm>
25 #include <cstddef>
26 #include <cstdint>
27 #include <optional>
28 #include <vector>
29 
30 #include <opencv2/core/eigen.hpp>
31 #include <opencv2/core/mat.hpp>
32 #include <opencv2/core/types.hpp>
33 #include <opencv2/imgproc.hpp>
34 
35 #include <VirtualRobot/BoundingBox.h>
36 #include <VirtualRobot/CollisionDetection/CollisionModel.h> // IWYU pragma: keep
37 #include <VirtualRobot/Random.h>
38 #include <VirtualRobot/Robot.h> // IWYU pragma: keep
39 #include <VirtualRobot/SceneObjectSet.h> // IWYU pragma: keep
40 #include <VirtualRobot/VirtualRobot.h>
41 #include <VirtualRobot/Workspace/WorkspaceGrid.h>
42 
47 
53 
54 #include <range/v3/algorithm/for_each.hpp>
55 #include <range/v3/view/zip.hpp>
56 
58 {
59 
60 
61  SceneBounds
62  computeSceneBounds(const VirtualRobot::SceneObjectSetPtr& obstacles,
63  const std::vector<VirtualRobot::RobotPtr>& articulatedObjects,
64  const SceneBounds& init,
65  const std::vector<Room>& rooms,
66  const float margin,
67  const bool restrictToRooms)
68  {
69  SceneBounds bounds = init;
70 
71  if (restrictToRooms)
72  {
73  ARMARX_INFO << "Computing scene bounds from rooms";
74 
76 
77  for (const auto& room : rooms)
78  {
79  for (const auto& point : room.polygon)
80  {
81  bounds.min.x() = std::min(point.x(), bounds.min.x());
82  bounds.min.y() = std::min(point.y(), bounds.min.y());
83  bounds.max.x() = std::max(point.x(), bounds.max.x());
84  bounds.max.y() = std::max(point.y(), bounds.max.y());
85  }
86  }
87 
88  return bounds;
89  }
90 
91  // if no rooms are given, use the obstacles and articulated objects
92  {
93 
94  ARMARX_CHECK_NOT_NULL(obstacles);
95 
96  ARMARX_CHECK((not obstacles->getCollisionModels().empty()) or
97  (not articulatedObjects.empty()));
98 
99  const auto expandBounds = [&bounds](const VirtualRobot::BoundingBox&& bb)
100  {
101  bounds.min.x() = std::min(bb.getMin().x(), bounds.min.x());
102  bounds.min.y() = std::min(bb.getMin().y(), bounds.min.y());
103  bounds.max.x() = std::max(bb.getMax().x(), bounds.max.x());
104  bounds.max.y() = std::max(bb.getMax().y(), bounds.max.y());
105  };
106 
107  // non-articulated objects
108  for (const auto& colModel : obstacles->getCollisionModels())
109  {
110  expandBounds(colModel->getBoundingBox());
111  }
112 
113  ARMARX_VERBOSE << "non-articulated objects: scene bounds: " << bounds.min << " and "
114  << bounds.max;
115 
116  // articulated objects
117  for (const auto& articulatedObject : articulatedObjects)
118  {
119  if (articulatedObject->getCollisionModels().empty())
120  {
121  ARMARX_WARNING << "The articulated object `" << articulatedObject->getType()
122  << "/" << articulatedObject->getName()
123  << "` does not provide any collision model!";
124  }
125 
126  ARMARX_VERBOSE << "Articulated object `" << articulatedObject->getType() << "/"
127  << articulatedObject->getName();
128 
129  for (const auto& colModel : articulatedObject->getCollisionModels())
130  {
131  expandBounds(colModel->getBoundingBox());
132  ARMARX_VERBOSE << VAROUT(colModel->getBoundingBox().getMin()) << " and "
133  << VAROUT(colModel->getBoundingBox().getMax());
134  ARMARX_VERBOSE << bounds.min << " and " << bounds.max;
135  }
136  }
137 
138  ARMARX_VERBOSE << "articulated objects: scene bounds: " << bounds.min << " and "
139  << bounds.max;
140  }
141 
142  // expand bounds by margin
143  bounds.min.x() -= margin;
144  bounds.min.y() -= margin;
145  bounds.max.x() += margin;
146  bounds.max.y() += margin;
147 
148  return bounds;
149  }
150 
151  SceneBounds
152  toSceneBounds(const VirtualRobot::WorkspaceGrid::Extends& extends)
153  {
154  return {.min = Eigen::Vector2f{extends.minX, extends.minY},
155  .max = Eigen::Vector2f{extends.maxX, extends.maxY}};
156  }
157 
158  SceneBounds
159  merge(const std::vector<SceneBounds>& sceneBounds)
160  {
161  SceneBounds bounds;
162 
163  const auto expandBounds = [&bounds](const SceneBounds& sb)
164  {
165  bounds.min.x() = std::min(bounds.min.x(), sb.min.x());
166  bounds.min.y() = std::min(bounds.min.y(), sb.min.y());
167 
168  bounds.max.x() = std::max(bounds.max.x(), sb.max.x());
169  bounds.max.y() = std::max(bounds.max.y(), sb.max.y());
170 
171  return bounds;
172  };
173 
174  ranges::for_each(sceneBounds, expandBounds);
175 
176  return bounds;
177  }
178 
179  Costmap
180  toCostmap(const VirtualRobot::WorkspaceGrid& workspaceGrid)
181  {
182  const Costmap::Grid grid(workspaceGrid.getCells().x, workspaceGrid.getCells().y);
183  const SceneBounds sceneBounds = toSceneBounds(workspaceGrid.getExtends());
184 
185  const Costmap::Parameters parameters{.binaryGrid = false,
186  .cellSize = workspaceGrid.getDiscretizeSize()};
187 
188  return {grid, parameters, sceneBounds};
189  }
190 
191  // Costmap
192  // mergeUnaligned(const std::vector<Costmap>& costmaps,
193  // const std::vector<float>& weights,
194  // float cellSize,
195  // const Eigen::Array2f& offset)
196  // {
197  // ARMARX_TRACE;
198 
199  // ARMARX_CHECK_NOT_EMPTY(costmaps);
200  // ARMARX_CHECK_EQUAL(costmaps.size(), weights.size());
201 
202  // // if unset, use the smallest cell size
203  // if (cellSize < 0)
204  // {
205  // const auto compCellSize = [](const Costmap& a, const Costmap& b) -> bool
206  // { return a.params().cellSize < b.params().cellSize; };
207 
208  // cellSize = ranges::min_element(costmaps, compCellSize)->params().cellSize;
209  // }
210 
211  // // scale each costmap
212  // std::vector<Costmap> scaledCostmaps =
213  // simox::alg::apply(costmaps,
214  // [&cellSize](const Costmap& costmap) -> Costmap
215  // { return scaleCostmap(costmap, cellSize); });
216 
217  // // merge costmaps into one
218  // ARMARX_VERBOSE << "Merging " << scaledCostmaps.size() << " costmaps";
219 
220  // // - combine scene bounds
221  // std::vector<SceneBounds> sceneBoundsAll = simox::alg::apply(
222  // scaledCostmaps, [](const Costmap& costmap) { return costmap.getSceneBounds(); });
223 
224  // SceneBounds sceneBounds = merge(sceneBoundsAll);
225  // sceneBounds.min.x() -= 0.5 * cellSize;
226  // sceneBounds.max.x() -= 0.5 * cellSize;
227  // sceneBounds.min.y() -= cellSize;
228  // sceneBounds.max.y() -= cellSize;
229 
230 
231  // sceneBounds.min.x() += offset.x();
232  // sceneBounds.max.x() += offset.x();
233  // sceneBounds.min.y() += offset.y();
234  // sceneBounds.max.y() += offset.y();
235 
236  // // - create grid
237  // const auto largeGrid = CostmapBuilder::createUniformGrid(
238  // sceneBounds, Costmap::Parameters{.binaryGrid = false, .cellSize = cellSize});
239 
240  // Costmap largeCostmap(
241  // largeGrid, Costmap::Parameters{.binaryGrid = false, .cellSize = cellSize}, sceneBounds);
242 
243  // // TODO save each combo individually
244 
245  // static int i = 0;
246  // // - add all costmaps to this one
247  // ranges::for_each(ranges::views::zip(scaledCostmaps, weights),
248  // [&largeCostmap](const auto& p)
249  // {
250  // ARMARX_INFO << "adding to costmap";
251  // const auto& [costmap, weight] = p;
252  // largeCostmap.add(costmap, weight);
253 
254  // armarx::navigation::algorithms::save(
255  // costmap, "/tmp/costmap" + std::to_string(i++));
256  // });
257 
258  // armarx::navigation::algorithms::save(largeCostmap,
259  // "/tmp/large-costmap" + std::to_string(i++));
260 
261  // return largeCostmap;
262  // }
263 
264  void
265  checkSameSize(const std::vector<Costmap>& costmaps)
266  {
267  ARMARX_CHECK_NOT_EMPTY(costmaps);
268 
269  const auto assertSameSize = [&costmaps](const Costmap& costmap)
270  {
271  ARMARX_CHECK_EQUAL(costmap.getGrid().rows(), costmaps.front().getGrid().rows());
272  ARMARX_CHECK_EQUAL(costmap.getGrid().rows(), costmaps.front().getGrid().rows());
273  };
274 
275  ranges::for_each(costmaps, assertSameSize);
276  }
277 
278  Costmap
279  mergeAligned(const std::vector<Costmap>& costmaps, CostmapMergeMode mode)
280  {
281  ARMARX_CHECK_NOT_EMPTY(costmaps);
282  checkSameSize(costmaps);
283 
284 
285  // average mode: fixed weight
286  if (mode == CostmapMergeMode::AVERAGE)
287  {
288  const std::vector<float> weights(costmaps.size(), 1.F / costmaps.size());
289  return mergeAligned(costmaps, weights);
290  }
291 
292  const auto grid = CostmapBuilder::createUniformGrid(costmaps.front().getLocalSceneBounds(),
293  costmaps.front().params());
294 
295  Costmap mergedCostmap(
296  grid, costmaps.front().params(), costmaps.front().getLocalSceneBounds());
297 
298  const auto addMode = [&]() -> Costmap::AddMode
299  {
300  Costmap::AddMode addMode{};
301 
302  switch (mode)
303  {
305  ARMARX_CHECK(false) << "This case should already be handled.";
306  break;
308  addMode = Costmap::AddMode::MIN;
309  break;
311  addMode = Costmap::AddMode::MAX;
312  break;
313  }
314 
315  return addMode;
316  }();
317 
318  ranges::for_each(costmaps,
319  [&mergedCostmap, &addMode](const auto& costmap)
320  { mergedCostmap.add(costmap, addMode); });
321 
322  return mergedCostmap;
323  }
324 
325  Costmap
326  mergeAligned(const std::vector<Costmap>& costmaps, const std::vector<float>& weights)
327  {
328  ARMARX_CHECK_EQUAL(costmaps.size(), weights.size());
329  ARMARX_CHECK_NOT_EMPTY(costmaps);
330  checkSameSize(costmaps);
331 
332 
333  const auto grid = CostmapBuilder::createUniformGrid(costmaps.front().getLocalSceneBounds(),
334  costmaps.front().params());
335 
336  Costmap mergedCostmap(
337  grid, costmaps.front().params(), costmaps.front().getLocalSceneBounds());
338 
339  // foreach pair (costmap, weight): add it to there merged costmap
340  ranges::for_each(ranges::views::zip(costmaps, weights),
341  [&mergedCostmap](const auto& p)
342  {
343  const auto& [costmap, weight] = p;
344  mergedCostmap.add(costmap, weight);
345  });
346 
347  return mergedCostmap;
348  }
349 
350  Costmap
351  scaleCostmap(const Costmap& costmap, float cellSize)
352  {
353  const float scale = costmap.params().cellSize / cellSize;
354  ARMARX_VERBOSE << "Scaling grid by a factor of " << scale;
355 
356  cv::Mat src;
357  cv::eigen2cv(costmap.getGrid(), src);
358 
359  cv::Mat dst;
360  cv::resize(src, dst, cv::Size{0, 0}, scale, scale);
361 
362  Eigen::MatrixXf scaledGrid;
363  cv::cv2eigen(dst, scaledGrid);
364 
365 
366  std::optional<Costmap::Mask> scaledMask;
367  if (costmap.getMask().has_value())
368  {
369  ARMARX_TRACE;
370  ARMARX_VERBOSE << "Scaling mask by a factor of " << scale;
371 
372  // just an int large enough to check if resizing causes
373  // issues on boundary to invalid regions
374  constexpr int someInt = 100;
375 
377  someInt * costmap.getMask().value().cast<std::uint8_t>();
378 
379  cv::Mat maskMat;
380  cv::eigen2cv(mask, maskMat);
381 
382  cv::Mat maskScaled;
383  cv::resize(maskMat, maskScaled, cv::Size{0, 0}, scale, scale);
384 
385  // only those points are valid where the original value '10' is preserved.
386  // all other points are boundary points to the invalid area
387  // cv::threshold(maskMat, maskMat, 10.9, 1, cv::THRESH_BINARY);
388 
389  Eigen::MatrixXi scaledMaskInt; //(scaledGrid.rows(), scaledGrid.cols());
390  cv::cv2eigen(maskScaled, scaledMaskInt);
391 
392  Costmap::Mask m = scaledMaskInt.array() >= someInt;
393 
394  scaledMask = m;
395  }
396 
397  ARMARX_VERBOSE << "Original size (" << costmap.getGrid().rows() << ", "
398  << costmap.getGrid().cols() << ")";
399  ARMARX_VERBOSE << "Resized to (" << scaledGrid.rows() << ", " << scaledGrid.cols() << ")";
400 
401  return {scaledGrid,
402  Costmap::Parameters{.binaryGrid = false, .cellSize = cellSize},
403  costmap.getLocalSceneBounds(),
404  scaledMask};
405  }
406 
407  std::optional<core::Pose2D>
409  {
410  const auto sizeX = costmap.getGrid().cols();
411  const auto sizeY = costmap.getGrid().rows();
412 
413  constexpr std::size_t maxIterations = 1000;
414 
415  // sample a valid pose in the costmap
416 
417  for (std::size_t iteration = 0; iteration < maxIterations; iteration++)
418  {
419  const float iX = VirtualRobot::RandomFloat() * static_cast<float>(sizeX);
420  const float iY = VirtualRobot::RandomFloat() * static_cast<float>(sizeY);
421 
422  algorithms::Costmap::Index idx(iX, iY);
423 
424  if (not costmap.isValid(idx))
425  {
426  continue;
427  }
428 
430  pose.translation() = costmap.toPositionGlobal(idx);
431  return pose;
432  }
433 
434  ARMARX_ERROR << "Failed to sample pose in costmap!";
435  return std::nullopt;
436  }
437 
438  void
439  invalidateOutsideRooms(const std::vector<Room>& rooms, Costmap& costmap)
440  {
441  const std::size_t c_x = costmap.getGrid().rows();
442  const std::size_t c_y = costmap.getGrid().cols();
443 
444  const auto isInsideRoom = [&rooms](const Eigen::Vector2f& pos) -> bool
445  {
446  return std::any_of(rooms.begin(),
447  rooms.end(),
448  [&pos](const Room& room) -> bool { return room.isInside(pos); });
449  };
450 
451 
452  // FIXME check if mask is already initialized
453  ARMARX_CHECK(costmap.getMutableMask().has_value());
454 
455  for (unsigned int x = 0; x < c_x; x++)
456  {
457  for (unsigned int y = 0; y < c_y; y++)
458  {
459  const Costmap::Index index{x, y};
460  const Costmap::Position position = costmap.toPositionGlobal(index);
461 
462  costmap.getMutableMask().value()(x, y) = isInsideRoom(position);
463  }
464  }
465  }
466 } // namespace armarx::navigation::algorithms
armarx::navigation::algorithms::Costmap::getMask
const std::optional< Mask > & getMask() const noexcept
Definition: Costmap.cpp:399
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
armarx::navigation::algorithms::Costmap::Index
Eigen::Array2i Index
Definition: Costmap.h:54
armarx::navigation::algorithms::Costmap::getMutableMask
std::optional< Costmap::Mask > & getMutableMask() noexcept
Definition: Costmap.cpp:405
armarx::navigation::algorithms::CostmapMergeMode
CostmapMergeMode
Defines how the.
Definition: util.h:65
armarx::navigation::algorithms::Costmap::AddMode::MAX
@ MAX
util.h
index
uint8_t index
Definition: EtherCATFrame.h:59
basic_types.h
armarx::navigation::algorithms::merge
SceneBounds merge(const std::vector< SceneBounds > &sceneBounds)
Definition: util.cpp:159
CostmapBuilder.h
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
armarx::objpose::util::articulatedObjects
objpose::ObjectPoseSeq articulatedObjects(objpose::ObjectPoseSeq objects)
Definition: util.cpp:82
armarx::navigation::algorithms::CostmapMergeMode::AVERAGE
@ AVERAGE
trace.h
armarx::navigation::algorithms::computeSceneBounds
SceneBounds computeSceneBounds(const VirtualRobot::SceneObjectSetPtr &obstacles, const std::vector< VirtualRobot::RobotPtr > &articulatedObjects, const SceneBounds &init, const std::vector< Room > &rooms, const float margin, const bool restrictToRooms)
Definition: util.cpp:62
armarx::navigation::algorithms
This file is part of ArmarX.
Definition: aron_conversions.cpp:25
armarx::navigation::algorithms::checkSameSize
void checkSameSize(const std::vector< Costmap > &costmaps)
Definition: util.cpp:265
armarx::rooms
Brief description of class rooms.
Definition: rooms.h:38
armarx::navigation::algorithms::Costmap::Grid
Eigen::MatrixXf Grid
Definition: Costmap.h:56
armarx::navigation::algorithms::mergeAligned
Costmap mergeAligned(const std::vector< Costmap > &costmaps, CostmapMergeMode mode)
Definition: util.cpp:279
ARMARX_CHECK_NOT_EMPTY
#define ARMARX_CHECK_NOT_EMPTY(c)
Definition: ExpressionException.h:224
armarx::navigation::algorithms::Costmap::toPositionGlobal
Position toPositionGlobal(const Index &index) const
Definition: Costmap.cpp:69
armarx::navigation::algorithms::CostmapBuilder::createUniformGrid
static Eigen::MatrixXf createUniformGrid(const SceneBounds &sceneBounds, const Costmap::Parameters &parameters)
Definition: CostmapBuilder.cpp:190
StringHelpers.h
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
armarx::navigation::algorithms::Costmap::getLocalSceneBounds
const SceneBounds & getLocalSceneBounds() const noexcept
Definition: Costmap.cpp:130
armarx::navigation::core::Pose2D
Eigen::Isometry2f Pose2D
Definition: basic_types.h:34
Costmap.h
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
GfxTL::Identity
void Identity(MatrixXX< N, N, T > *a)
Definition: MatrixXX.h:570
armarx::navigation::algorithms::Costmap::Parameters::binaryGrid
bool binaryGrid
Definition: Costmap.h:27
armarx::navigation::algorithms::Costmap::AddMode::MIN
@ MIN
armarx::navigation::algorithms::Costmap::params
const Parameters & params() const noexcept
Definition: Costmap.cpp:237
armarx::navigation::algorithms::Room
Definition: Room.h:33
armarx::navigation::algorithms::scaleCostmap
Costmap scaleCostmap(const Costmap &costmap, float cellSize)
Definition: util.cpp:351
armarx::navigation::algorithms::Costmap::getGrid
const Grid & getGrid() const
Definition: Costmap.cpp:243
armarx::conversions::cv2eigen
Eigen::Vector2f cv2eigen(const cv::Point2f &pt)
Definition: opencv_eigen.h:54
armarx::navigation::algorithms::invalidateOutsideRooms
void invalidateOutsideRooms(const std::vector< Room > &rooms, Costmap &costmap)
Definition: util.cpp:439
armarx::navigation::algorithms::Costmap::Position
Eigen::Vector2f Position
Definition: Costmap.h:55
max
T max(T t1, T t2)
Definition: gdiam.h:51
armarx::navigation::algorithms::SceneBounds
Definition: types.h:29
ARMARX_ERROR
#define ARMARX_ERROR
Definition: Logging.h:196
ExpressionException.h
armarx::navigation::algorithms::CostmapMergeMode::MAX
@ MAX
armarx::navigation::algorithms::CostmapMergeMode::MIN
@ MIN
armarx::navigation::algorithms::SceneBounds::min
Eigen::Vector2f min
Definition: types.h:31
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::navigation::algorithms::Costmap::AddMode
AddMode
Definition: Costmap.h:117
VAROUT
#define VAROUT(x)
Definition: StringHelpers.h:198
armarx::navigation::algorithms::Costmap::Parameters::cellSize
float cellSize
How big each cell is in the uniform grid.
Definition: Costmap.h:30
armarx::navigation::algorithms::sampleValidPositionInMap
std::optional< core::Pose2D > sampleValidPositionInMap(const algorithms::Costmap &costmap)
Definition: util.cpp:408
armarx::navigation::algorithms::Costmap::Parameters
Definition: Costmap.h:24
Eigen::Matrix
Definition: EigenForwardDeclarations.h:27
armarx::navigation::algorithms::toSceneBounds
SceneBounds toSceneBounds(const VirtualRobot::WorkspaceGrid::Extends &extends)
Definition: util.cpp:152
armarx::navigation::algorithms::toCostmap
Costmap toCostmap(const VirtualRobot::WorkspaceGrid &workspaceGrid)
Definition: util.cpp:180
Logging.h
min
T min(T t1, T t2)
Definition: gdiam.h:44
ARMARX_CHECK_EQUAL
#define ARMARX_CHECK_EQUAL(lhs, rhs)
This macro evaluates whether lhs is equal (==) rhs and if it turns out to be false it will throw an E...
Definition: ExpressionException.h:130
ARMARX_WARNING
#define ARMARX_WARNING
Definition: Logging.h:193
armarx::navigation::algorithms::Costmap::isValid
bool isValid(const Index &index) const noexcept
checks whether the cell is masked out
Definition: Costmap.cpp:76
armarx::navigation::algorithms::Costmap::add
bool add(const Costmap &other, float weight=1.0)
Definition: Costmap.cpp:256
Room.h
types.h
armarx::navigation::algorithms::SceneBounds::max
Eigen::Vector2f max
Definition: types.h:32
armarx::navigation::algorithms::Costmap
Definition: Costmap.h:16
armarx::conversions::eigen2cv
cv::Point2f eigen2cv(const Eigen::Vector2f &pt)
Definition: opencv_eigen.h:33