Component.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  * @package armarx::navigation::components::costmap_merger
17  * @author Tobias Gröger ( tobias dot groeger at student dot kit dot edu )
18  * @date 2025
19  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 
24 #include "Component.h"
25 
26 #include <algorithm>
27 #include <cstddef>
28 #include <limits>
29 #include <mutex>
30 #include <string>
31 #include <utility>
32 #include <vector>
33 
34 #include <SimoxUtility/algorithm/string/string_tools.h>
35 
44 
46 
51 
52 #include <range/v3/range/conversion.hpp>
53 #include <range/v3/view/filter.hpp>
54 
56 {
57  using algorithms::Costmap;
58 
60  {
61  addPlugin(costmapReaderPlugin);
62  addPlugin(costmapWriterPlugin);
63  }
64 
65  const std::string Component::defaultName = "costmap_merger";
66 
69  {
72 
73  def->required(
74  properties.costmapsToMerge,
75  "p.costmapsToMerge",
76  "The costmaps to merge, they are prioritized in the order they are specified, i.e. the "
77  "first costmap will take priority over the second one etc.");
78 
79  def->optional(
80  properties.outputCostmapName, "p.outputCostmapName", "The name of the merged costmap.");
81 
82  return def;
83  }
84 
85  void
87  {
88  // This should not be necessary but seems to be. ToDo: Look into this.
90 
94  this,
96  }
97 
98  void
100  {
101  ARMARX_VERBOSE << "Creating initial merge of costmaps";
103  }
104 
105  void
107  const std::vector<armem::MemoryID>& snapshotIDs)
108  {
109  ARMARX_VERBOSE << "A costmap instance has changed!";
110 
111  const std::vector<armem::MemoryID> relevantCostmaps =
112  snapshotIDs |
113  ranges::views::filter(
114  [&](const armem::MemoryID& id)
115  {
116  std::string name = id.providerSegmentName + "/" + id.entityName;
117  return std::find(properties.costmapsToMerge.cbegin(),
118  properties.costmapsToMerge.cend(),
119  name) != properties.costmapsToMerge.cend();
120  }) |
121  ranges::to_vector;
122 
123  if (relevantCostmaps.empty())
124  {
125  return;
126  }
127 
128  ARMARX_INFO << "Relevant costmaps changed: " << relevantCostmaps;
129 
130  // Re-merge the costmaps and store them
132  }
133 
134  void
136  {
137  std::lock_guard g{mergeCostmapMtx};
138 
139  ARMARX_VERBOSE << "Reading costmaps";
140 
141  const armarx::DateTime timestamp = Clock::Now();
142 
143  std::vector<Costmap> costmaps;
144 
145  for (const auto& costmap : properties.costmapsToMerge)
146  {
147  ARMARX_VERBOSE << "Reading costmap `" << costmap << "`.";
148  const std::vector<std::string> splits = simox::alg::split(costmap, "/");
149  ARMARX_CHECK_EQUAL(splits.size(), 2) << "Invalid costmap name: " << costmap;
150 
152  .providerName = splits.front(), .name = splits.back(), .timestamp = timestamp};
153  auto result = costmapReaderPlugin->get().query(query);
154 
155  if (not result)
156  {
157  ARMARX_WARNING << "Attempting to merge costmap `" << costmap
158  << "` which doesn't exist (yet) (" << result.errorMessage << ")";
159  }
160  else
161  {
162  ARMARX_CHECK_NOT_NULL(result.costmap);
163  costmaps.emplace_back(std::move(*result.costmap));
164  }
165  }
166 
167  if (costmaps.empty())
168  {
169  ARMARX_WARNING << "Nothing to merge";
170  return;
171  }
172 
173  ARMARX_TRACE;
174  ARMARX_VERBOSE << "Merging costmaps";
175 
176  auto mergedCostmap = [&]()
177  {
178  // calculate the union of the two costmaps
179  // and the cell size of the most detailed costmap
180  Eigen::AlignedBox2f aabb;
181  float minCellSize = std::numeric_limits<float>::max();
182  for (const auto& c : costmaps)
183  {
184  // origin transformation might mix up where min/max lies, therefore not min/max anymore
185  Eigen::Vector2f globalCornerA = c.origin() * c.getLocalSceneBounds().min;
186  Eigen::Vector2f globalCornerB = c.origin() * c.getLocalSceneBounds().max;
187 
188  aabb.extend(globalCornerA);
189  aabb.extend(globalCornerB);
190 
191  minCellSize = std::min(minCellSize, c.params().cellSize);
192  }
193  ARMARX_TRACE;
194  ARMARX_VERBOSE << "Scene bounds intersections is " << aabb.min() << " and "
195  << aabb.max() << " with minCellSize as " << minCellSize;
196 
197 
198  // create costmap with given cell size and scene bounds
199 
200  //+1 for explicit rounding up
201  Eigen::Vector2f boundingSizes = aabb.sizes();
202  const auto cX = static_cast<std::size_t>((boundingSizes.x() / minCellSize) + 1);
203  const auto cY = static_cast<std::size_t>((boundingSizes.y() / minCellSize) + 1);
204 
205  Eigen::MatrixXf grid(cX, cY);
206  grid.setZero();
207 
208  Costmap costmap(grid,
209  Costmap::Parameters{.cellSize = minCellSize, .sceneBoundsMargin = 0.F},
210  algorithms::SceneBounds{.min = aabb.min(), .max = aabb.max()});
211 
212  // to copy shape from grid to mask
213  costmap.getMutableMask() = costmap.getGrid().array() == 0;
214  costmap.getMutableMask()->setOnes();
215 
216  return costmap;
217  }();
218 
219  ARMARX_CHECK(mergedCostmap.getMask().has_value());
220 
221  // check entry of first valid costmap in order and return whether any costmap was valid
222  const auto& checkInOrder = [&](int x, int y, const Eigen::Vector2f& position)
223  {
224  for (const auto& other : costmaps)
225  {
226  Eigen::AlignedBox2f bounds(other.getLocalSceneBounds().min,
227  other.getLocalSceneBounds().max);
228  const auto otherVertex = other.toVertex(position);
229  // only use this costmap if the current position lies inside it's bounds
230  if (bounds.contains(position) && other.isValid(otherVertex.index))
231  {
232  mergedCostmap.getMutableGrid()(x, y) = *other.value(otherVertex.index);
233  return true;
234  }
235  }
236  return false;
237  };
238 
239  for (int x = 0; x < mergedCostmap.getGrid().rows(); x++)
240  {
241  for (int y = 0; y < mergedCostmap.getGrid().cols(); y++)
242  {
243  ARMARX_TRACE;
244  const Costmap::Position position = mergedCostmap.toPositionGlobal({x, y});
245  mergedCostmap.getMutableMask().value()(x, y) = checkInOrder(x, y, position);
246  }
247  }
248 
249  costmapWriterPlugin->get().store(
250  mergedCostmap, properties.outputCostmapName, getName(), timestamp);
251  }
252 
253  void
255  {
256  }
257 
258  void
260  {
261  }
262 
263  std::string
265  {
266  return Component::defaultName;
267  }
268 
269  std::string
271  {
272  return Component::defaultName;
273  }
274 
276 
277 } // namespace armarx::navigation::components::costmap_merger
armarx::navigation::components::costmap_merger::Component::createPropertyDefinitions
armarx::PropertyDefinitionsPtr createPropertyDefinitions() override
Definition: Component.cpp:68
armarx::navigation::components::costmap_merger::ARMARX_REGISTER_COMPONENT_EXECUTABLE
ARMARX_REGISTER_COMPONENT_EXECUTABLE(Component, Component::GetDefaultName())
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
armarx::navigation::algorithms::Costmap::getMutableMask
std::optional< Costmap::Mask > & getMutableMask() noexcept
Definition: Costmap.cpp:405
armarx::navigation::components::costmap_merger::Component::onInitComponent
void onInitComponent() override
Definition: Component.cpp:86
armarx::navigation::memory::constants::CostmapCoreSegmentName
const std::string CostmapCoreSegmentName
Definition: constants.h:36
armarx::navigation::components::costmap_merger::Component::mergeAndStoreCostmap
void mergeAndStoreCostmap()
Definition: Component.cpp:135
MemoryID.h
armarx::armem::client::plugins::PluginUser::memoryNameSystem
MemoryNameSystem & memoryNameSystem()
Definition: PluginUser.cpp:20
DateTime.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
trace.h
armarx::max
std::vector< T > max(const std::vector< T > &v1, const std::vector< T > &v2)
Definition: VectorHelpers.h:297
Reader.h
c
constexpr T c
Definition: UnscentedKalmanFilterTest.cpp:46
armarx::navigation::memory::constants::NavigationMemoryName
const std::string NavigationMemoryName
Definition: constants.h:29
armarx::ManagedIceObject::addPlugin
PluginT * addPlugin(const std::string prefix="", ParamsT &&... params)
Definition: ManagedIceObject.h:186
armarx::navigation::components::costmap_merger::Component::Component
Component()
Definition: Component.cpp:59
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
Clock.h
armarx::navigation::memory::client::costmap::Reader::Query::providerName
std::string providerName
Definition: Reader.h:44
Costmap.h
armarx::navigation::components::costmap_merger
Definition: Component.cpp:55
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
armarx::armem::MemoryID
A memory ID.
Definition: MemoryID.h:47
armarx::navigation::components::costmap_merger::Component::onExitComponent
void onExitComponent() override
Definition: Component.cpp:259
armarx::navigation::components::costmap_merger::Component::processObjectInstance
void processObjectInstance(const armem::MemoryID &id, const std::vector< armem::MemoryID > &snapshotIDs)
Definition: Component.cpp:106
armarx::navigation::components::costmap_merger::Component
Definition: Component.h:44
armarx::navigation::algorithms::Costmap::Position
Eigen::Vector2f Position
Definition: Costmap.h:55
max
T max(T t1, T t2)
Definition: gdiam.h:51
armarx::armem::client::util::MemoryListener::subscribe
SubscriptionHandle subscribe(const MemoryID &subscriptionID, Callback Callback)
Definition: MemoryListener.cpp:116
armarx::navigation::algorithms::SceneBounds
Definition: types.h:29
armarx::navigation::memory::client::costmap::Reader::Query
Definition: Reader.h:42
Component.h
ExpressionException.h
armarx::navigation::components::costmap_merger::Component::GetDefaultName
static std::string GetDefaultName()
Get the component's default name.
Definition: Component.cpp:270
armarx::navigation::components::costmap_merger::Component::onDisconnectComponent
void onDisconnectComponent() override
Definition: Component.cpp:254
armarx::core::time::DateTime
Represents a point in time.
Definition: DateTime.h:24
armarx::Component::getConfigIdentifier
std::string getConfigIdentifier()
Retrieve config identifier for this component as set in constructor.
Definition: Component.cpp:79
Decoupled.h
armarx::armem::client::MemoryNameSystem::setComponent
void setComponent(ManagedIceObject *component)
Definition: MemoryNameSystem.cpp:462
armarx::ComponentPropertyDefinitions
Default component property definition container.
Definition: Component.h:69
armarx::navigation::algorithms::SceneBounds::min
Eigen::Vector2f min
Definition: types.h:31
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
PropertyDefinitionContainer.h
IceUtil::Handle< class PropertyDefinitionContainer >
armarx::navigation::algorithms::Costmap::Parameters::cellSize
float cellSize
How big each cell is in the uniform grid.
Definition: Costmap.h:30
armarx::navigation::algorithms::Costmap::Parameters
Definition: Costmap.h:24
constants.h
armarx::core::time::Clock::Now
static DateTime Now()
Current time on the virtual clock.
Definition: Clock.cpp:93
armarx::ManagedIceObject::getName
std::string getName() const
Retrieve name of object.
Definition: ManagedIceObject.cpp:108
Logging.h
armarx::navigation::components::costmap_merger::Component::getDefaultName
std::string getDefaultName() const override
Definition: Component.cpp:264
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::components::costmap_merger::Component::onConnectComponent
void onConnectComponent() override
Definition: Component.cpp:99
types.h
armarx::navigation::algorithms::Costmap
Definition: Costmap.h:16
armarx::split
std::vector< std::string > split(const std::string &source, const std::string &splitBy, bool trimElements=false, bool removeEmptyElements=false)
Definition: StringHelpers.cpp:38