MRPT  2.0.1
test.cpp
Go to the documentation of this file.
1 /* +------------------------------------------------------------------------+
2  | Mobile Robot Programming Toolkit (MRPT) |
3  | https://www.mrpt.org/ |
4  | |
5  | Copyright (c) 2005-2020, Individual contributors, see AUTHORS file |
6  | See: https://www.mrpt.org/Authors - All rights reserved. |
7  | Released under BSD License. See: https://www.mrpt.org/License |
8  +------------------------------------------------------------------------+ */
9 
12 #include <mrpt/math/geometry.h>
16 #include <mrpt/poses/CPose2D.h>
18 #include <mrpt/poses/CPosePDFSOG.h>
19 #include <mrpt/random.h>
20 #include <mrpt/system/CTicTac.h>
22 #include <mrpt/tfest/se2.h>
23 #include <iostream>
24 
25 // Method explained in paper:
26 // J.L. Blanco, J. Gonzalez-Jimenez, J.A. Fernandez-Madrigal,
27 // "A Robust, Multi-Hypothesis Approach to Matching Occupancy Grid Maps",
28 // Robotica, 2013.
29 // http://dx.doi.org/10.1017/S0263574712000732
30 
31 // ============= PARAMETERS ===================
32 const size_t NUM_OBSERVATIONS_TO_SIMUL = 10;
33 const size_t RANSAC_MINIMUM_INLIERS = 9; // Min. # of inliers to accept
34 
35 #define LOAD_MAP_FROM_FILE 0 // 1: load from "sMAP_FILE", 0: random map.
36 #define SHOW_POINT_LABELS 0
37 
38 const float normalizationStd = 0.15f; // 1 sigma noise (meters)
39 const float ransac_mahalanobisDistanceThreshold = 5.0f;
40 const size_t MINIMUM_RANSAC_ITERS = 100000;
41 
42 #if !LOAD_MAP_FROM_FILE
43 const size_t NUM_MAP_FEATS = 100;
44 const double MAP_SIZE_X = 50;
45 const double MAP_SIZE_Y = 25;
46 #else
47 // Expected format of the 2D map is, for each line (one per landmark):
48 // ID X Y
49 const std::string sMAP_FILE = string("./DLRMap.txt");
50 #endif
51 // ==============================================
52 
53 using namespace mrpt;
54 using namespace mrpt::math;
55 using namespace mrpt::random;
56 using namespace mrpt::maps;
57 using namespace mrpt::tfest;
58 using namespace mrpt::img;
59 using namespace std;
60 
61 struct TObs
62 {
63  size_t ID; // Ground truth ID
64  double x, y;
65 };
66 
67 // ------------------------------------------------------
68 // TestRANSAC
69 // ------------------------------------------------------
70 void TestRANSAC()
71 {
73  "MRPT example: ransac-data-association", 800, 600);
74 
75  mrpt::system::CTimeLogger timelog; // For dumping stats at the end
77 
78  getRandomGenerator().randomize(); // randomize with time
79 
80  // --------------------------------
81  // Load feature map:
82  // --------------------------------
83  CSimplePointsMap the_map;
84 #if LOAD_MAP_FROM_FILE
85  {
86  CMatrixDouble M;
87  M.loadFromTextFile(sMAP_FILE); // Launch except. on error
88  ASSERT_(M.cols() == 3 && M.rows() > 2)
89 
90  const size_t nPts = M.rows();
91  the_map.resize(nPts);
92  for (size_t i = 0; i < nPts; i++) the_map.setPoint(i, M(i, 1), M(i, 2));
93  }
94 #else
95  // Generate random MAP:
96  the_map.resize(NUM_MAP_FEATS);
97  for (size_t i = 0; i < NUM_MAP_FEATS; i++)
98  {
99  the_map.setPoint(
100  i, getRandomGenerator().drawUniform(0, MAP_SIZE_X),
101  getRandomGenerator().drawUniform(0, MAP_SIZE_Y));
102  }
103 #endif
104 
105  const size_t nMapPts = the_map.size();
106  cout << "Loaded/generated map with " << nMapPts << " landmarks.\n";
107 
108  const size_t nObs = NUM_OBSERVATIONS_TO_SIMUL;
109 
110  mrpt::opengl::CPointCloud::Ptr gl_obs_map =
120  {
121  mrpt::opengl::COpenGLScene::Ptr& scene = win.get3DSceneAndLock();
122 
123  scene->getViewport("main")->setCustomBackgroundColor(
124  TColorf(0.8f, 0.8f, 0.8f));
125  win.setCameraPointingToPoint(MAP_SIZE_X * 0.5, MAP_SIZE_Y * 0.5, 0);
126  win.setCameraZoom(2 * MAP_SIZE_X);
127 
128  //
130 
131  //
134  gl_map->loadFromPointsMap(&the_map);
135  gl_map->setColor(0, 0, 1);
136  gl_map->setPointSize(3);
137 
138  scene->insert(gl_map);
139 
140 #if SHOW_POINT_LABELS
141  for (size_t i = 0; i < the_map.size(); i++)
142  {
144  mrpt::format("%u", static_cast<unsigned int>(i)));
145  double x, y;
146  the_map.getPoint(i, x, y);
147  gl_txt->setLocation(x + 0.05, y + 0.05, 0.01);
148 
149  scene->insert(gl_txt);
150  }
151 #endif
152 
153  //
154  scene->insert(gl_lines);
155 
156  //
157  gl_obs_map->setColor(1, 0, 0);
158  gl_obs_map->setPointSize(5);
159 
160  gl_result->setColor(0, 1, 0);
161  gl_result->setPointSize(4);
162 
163  //
164  gl_obs->insert(mrpt::opengl::stock_objects::CornerXYZ(0.6f));
165  gl_obs->insert(gl_obs_map);
166  gl_obs->insert(gl_obs_txts);
167  scene->insert(gl_obs);
168  scene->insert(gl_result);
169 
170  win.unlockAccess3DScene();
171  win.repaint();
172  }
173 
174  // Repeat for each set of observations in the input file
175  while (win.isOpen())
176  {
177  // Read the observations themselves:
178  vector<TObs> observations;
179  observations.resize(nObs);
180 
181  const mrpt::poses::CPose2D GT_pose(
182  mrpt::random::getRandomGenerator().drawUniform(
183  -10, 10 + MAP_SIZE_X),
184  mrpt::random::getRandomGenerator().drawUniform(
185  -10, 10 + MAP_SIZE_Y),
186  mrpt::random::getRandomGenerator().drawUniform(-M_PI, M_PI));
187 
188  const mrpt::poses::CPose2D GT_pose_inv = -GT_pose;
189 
190  std::vector<std::pair<size_t, float>> idxs;
191  the_map.kdTreeRadiusSearch2D(GT_pose.x(), GT_pose.y(), 1000, idxs);
192  ASSERT_(idxs.size() >= nObs);
193 
194  for (size_t i = 0; i < nObs; i++)
195  {
196  double gx, gy;
197  the_map.getPoint(idxs[i].first, gx, gy);
198 
199  double lx, ly;
200  GT_pose_inv.composePoint(gx, gy, lx, ly);
201 
202  observations[i].ID = idxs[i].first;
203  observations[i].x =
205  0, normalizationStd);
206  observations[i].y =
208  0, normalizationStd);
209  }
210 
211  // ----------------------------------------------------
212  // Generate list of individual-compatible pairings
213  // ----------------------------------------------------
214  TMatchingPairList all_correspondences;
215 
216  all_correspondences.reserve(nMapPts * nObs);
217 
218  // ALL possibilities:
219  for (size_t j = 0; j < nObs; j++)
220  {
221  TMatchingPair match;
222 
223  match.other_idx = j;
224  match.other_x = observations[j].x;
225  match.other_y = observations[j].y;
226 
227  for (size_t i = 0; i < nMapPts; i++)
228  {
229  match.this_idx = i;
230  the_map.getPoint(i, match.this_x, match.this_y);
231 
232  all_correspondences.push_back(match);
233  }
234  }
235  cout << "Generated " << all_correspondences.size()
236  << " potential pairings.\n";
237 
238  // ----------------------------------------------------
239  // Run RANSAC-based D-A
240  // ----------------------------------------------------
241  timelog.enter("robustRigidTransformation");
242  timer.Tic();
243 
246 
247  params.ransac_minSetSize =
248  RANSAC_MINIMUM_INLIERS; // ransac_minSetSize (to add the solution
249  // to the SOG)
250  params.ransac_maxSetSize =
251  all_correspondences
252  .size(); // ransac_maxSetSize: Test with all data points
253  params.ransac_mahalanobisDistanceThreshold =
255  params.ransac_nSimulations = 0; // 0=auto
256  params.ransac_fuseByCorrsMatch = true;
257  params.ransac_fuseMaxDiffXY = 0.01f;
258  params.ransac_fuseMaxDiffPhi = 0.1_deg;
259  params.ransac_algorithmForLandmarks = true;
260  params.probability_find_good_model = 0.999999;
261  params.ransac_min_nSimulations =
262  MINIMUM_RANSAC_ITERS; // (a lower limit to the auto-detected value
263  // of ransac_nSimulations)
264  params.verbose = true;
265 
266  // Run ransac data-association:
268  all_correspondences, normalizationStd, params, results);
269 
270  timelog.leave("robustRigidTransformation");
271 
272  mrpt::poses::CPosePDFSOG& best_poses = results.transformation;
273  TMatchingPairList& out_best_pairings = results.largestSubSet;
274 
275  const double tim = timer.Tac();
276  cout << "RANSAC time: " << mrpt::system::formatTimeInterval(tim)
277  << endl;
278 
279  cout << "# of SOG modes: " << best_poses.size() << endl;
280  cout << "Best match has " << out_best_pairings.size() << " features:\n";
281  for (size_t i = 0; i < out_best_pairings.size(); i++)
282  cout << out_best_pairings[i].this_idx << " <-> "
283  << out_best_pairings[i].other_idx << endl;
284  cout << endl;
285 
286  // Generate "association vector":
287  vector<int> obs2map_pairings(nObs, -1);
288  for (size_t i = 0; i < out_best_pairings.size(); i++)
289  obs2map_pairings[out_best_pairings[i].other_idx] =
290  out_best_pairings[i].this_idx == ((unsigned int)-1)
291  ? -1
292  : out_best_pairings[i].this_idx;
293 
294  cout << "1\n";
295  for (size_t i = 0; i < nObs; i++) cout << obs2map_pairings[i] << " ";
296  cout << endl;
297 
298  gl_result->clear();
299 
300  // Reconstruct the SE(2) transformation for these pairings:
301  mrpt::poses::CPosePDFGaussian solution_pose;
302  mrpt::tfest::se2_l2(out_best_pairings, solution_pose);
303 
304  // Normalized covariance: scale!
305  solution_pose.cov *= square(normalizationStd);
306 
307  cout << "Solution pose: " << solution_pose.mean << endl;
308  cout << "Ground truth pose: " << GT_pose << endl;
309 
310  {
311  // mrpt::opengl::COpenGLScene::Ptr &scene =
312  win.get3DSceneAndLock();
313 
314  win.addTextMessage(
315  5, 5,
316  "Blue: map landmarks | Red: Observations | White lines: Found "
317  "correspondences",
318  0);
319 
320  //
321  gl_obs_map->clear();
322  for (size_t k = 0; k < nObs; k++)
323  gl_obs_map->insertPoint(
324  observations[k].x, observations[k].y, 0.0);
325 
326  gl_obs->setPose(solution_pose.mean);
327 
328 #if SHOW_POINT_LABELS
329  gl_obs_txts->clear();
330  for (size_t i = 0; i < nObs; i++)
331  {
333  mrpt::format("%u", static_cast<unsigned int>(i)));
334  const double x = observations[i].x;
335  const double y = observations[i].y;
336  gl_txt->setLocation(x + 0.05, y + 0.05, 0.01);
337  gl_obs_txts->insert(gl_txt);
338  }
339 #endif
340 
341  //
342  gl_lines->clear();
343  double sqerr = 0;
344  size_t nPairs = 0;
345  for (size_t k = 0; k < nObs; k++)
346  {
347  int map_idx = obs2map_pairings[k];
348  if (map_idx < 0) continue;
349  nPairs++;
350 
351  double map_x, map_y;
352  the_map.getPoint(map_idx, map_x, map_y);
353 
354  double obs_x, obs_y;
355  solution_pose.mean.composePoint(
356  observations[k].x, observations[k].y, obs_x, obs_y);
357 
358  const double z = 0;
359 
360  gl_lines->appendLine(map_x, map_y, 0, obs_x, obs_y, z);
361 
362  sqerr += mrpt::math::distanceSqrBetweenPoints<double>(
363  map_x, map_y, obs_x, obs_y);
364  }
365 
366  win.addTextMessage(
367  5, 20, "Ground truth pose : " + GT_pose.asString(), 1);
368  win.addTextMessage(
369  5, 35,
370  "RANSAC estimated pose: " + solution_pose.mean.asString() +
371  mrpt::format(" | RMSE=%f", (nPairs ? sqerr / nPairs : 0.0)),
372  2);
373 
374  win.unlockAccess3DScene();
375  win.repaint();
376 
377  cout << "nPairings: " << nPairs
378  << " RMSE = " << (nPairs ? sqerr / nPairs : 0.0) << endl;
379 
380  win.waitForKey();
381  }
382 
383  } // end of for each set of observations
384 }
385 
386 // ------------------------------------------------------
387 // MAIN
388 // ------------------------------------------------------
389 int main()
390 {
391  try
392  {
393  TestRANSAC();
394  return 0;
395  }
396  catch (const std::exception& e)
397  {
398  std::cerr << "MRPT error: " << mrpt::exception_to_str(e) << std::endl;
399  return -1;
400  }
401  catch (...)
402  {
403  printf("Untyped exception!!");
404  return -1;
405  }
406 }
A namespace of pseudo-random numbers generators of diferent distributions.
double Tac() noexcept
Stops the stopwatch.
Definition: CTicTac.cpp:86
void getPoint(size_t index, float &x, float &y, float &z) const
Access to a given point from map, as a 2D point.
Definition: CPointsMap.cpp:192
CPose2D mean
The mean value.
const float normalizationStd
void asString(std::string &s) const
Returns a human-readable textual representation of the object (eg: "[x y yaw]", yaw in degrees) ...
Definition: CPose2D.cpp:445
static Ptr Create(Args &&... args)
Definition: CText.h:37
Declares a class that represents a Probability Density function (PDF) of a 2D pose ...
Definition: CPosePDFSOG.h:34
std::string std::string format(std::string_view fmt, ARGS &&... args)
Definition: format.h:26
bool se2_l2(const mrpt::tfest::TMatchingPairList &in_correspondences, mrpt::math::TPose2D &out_transformation, mrpt::math::CMatrixDouble33 *out_estimateCovariance=nullptr)
Least-squares (L2 norm) solution to finding the optimal SE(2) (x,y,yaw) between two reference frames...
Definition: se2_l2.cpp:92
Parameters for se2_l2_robust().
Definition: se2.h:73
void randomize(const uint32_t seed)
Initialize the PRNG from the given random seed.
const size_t MINIMUM_RANSAC_ITERS
A high-performance stopwatch, with typical resolution of nanoseconds.
const float ransac_mahalanobisDistanceThreshold
mrpt::vision::TStereoCalibParams params
A cloud of points in 2D or 3D, which can be built from a sequence of laser scans. ...
STL namespace.
static Ptr Create(Args &&... args)
Definition: CSetOfObjects.h:28
std::string formatTimeInterval(const double timeSeconds)
Returns a formated string with the given time difference (passed as the number of seconds)...
Definition: datetime.cpp:124
void composePoint(double lx, double ly, double &gx, double &gy) const
An alternative, slightly more efficient way of doing with G and L being 2D points and P this 2D pose...
Definition: CPose2D.cpp:199
mrpt::math::CMatrixDouble33 cov
The 3x3 covariance matrix.
static Ptr Create(Args &&... args)
Definition: CPointCloud.h:49
#define ASSERT_(f)
Defines an assertion mechanism.
Definition: exceptions.h:120
This base provides a set of functions for maths stuff.
size_t kdTreeRadiusSearch2D(const num_t x0, const num_t y0, const num_t maxRadiusSqr, std::vector< std::pair< size_t, num_t >> &out_indices_dist) const
KD Tree-based search for all the points within a given radius of some 2D point.
void TestRANSAC()
bool se2_l2_robust(const mrpt::tfest::TMatchingPairList &in_correspondences, const double in_normalizationStd, const TSE2RobustParams &in_ransac_params, TSE2RobustResult &out_results)
Robust least-squares (L2 norm) solution to finding the optimal SE(2) (x,y,yaw) between two reference ...
map< string, CVectorDouble > results
Declares a class that represents a Probability Density function (PDF) of a 2D pose ...
A list of TMatchingPair.
Definition: TMatchingPair.h:70
return_t drawGaussian1D(const double mean, const double std)
Generate a normally distributed pseudo-random number.
mrpt::gui::CDisplayWindow3D::Ptr win
size_type rows() const
Number of rows in the matrix.
size_type cols() const
Number of columns in the matrix.
A structure for holding correspondences between two sets of points or points-like entities in 2D or 3...
Definition: TMatchingPair.h:31
const size_t NUM_OBSERVATIONS_TO_SIMUL
return_t square(const num_t x)
Inline function for the square of a number.
void enter(const std::string_view &func_name) noexcept
Start of a named section.
This is the global namespace for all Mobile Robot Programming Toolkit (MRPT) libraries.
void setPoint(size_t index, float x, float y, float z)
Changes a given point from map, with Z defaulting to 0 if not provided.
Definition: CPointsMap.h:492
A versatile "profiler" that logs the time spent within each pair of calls to enter(X)-leave(X), among other stats.
A class used to store a 2D pose, including the 2D coordinate point and a heading (phi) angle...
Definition: CPose2D.h:39
const double MAP_SIZE_X
CSetOfObjects::Ptr CornerXYZ(float scale=1.0)
Returns three arrows representing a X,Y,Z 3D corner.
An RGBA color - floats in the range [0,1].
Definition: TColor.h:88
double leave(const std::string_view &func_name) noexcept
End of a named section.
std::string exception_to_str(const std::exception &e)
Builds a nice textual representation of a nested exception, which if generated using MRPT macros (THR...
Definition: exceptions.cpp:59
size_t size() const
Return the number of Gaussian modes.
Definition: CPosePDFSOG.h:85
bool verbose
Show progress messages to std::cout console (default=true)
void Tic() noexcept
Starts the stopwatch.
Definition: CTicTac.cpp:75
CRandomGenerator & getRandomGenerator()
A static instance of a CRandomGenerator class, for use in single-thread applications.
const double MAP_SIZE_Y
size_t size() const
Save the point cloud as a PCL PCD file, in either ASCII or binary format.
Definition: CPointsMap.h:440
void resize(size_t newLength) override
Resizes all point buffers so they can hold the given number of points: newly created points are set t...
Functions for estimating the optimal transformation between two frames of references given measuremen...
Output placeholder for se2_l2_robust()
Definition: se2.h:130
const size_t NUM_MAP_FEATS
static Ptr Create(Args &&... args)
Definition: CSetOfLines.h:35
A graphical user interface (GUI) for efficiently rendering 3D scenes in real-time.
const size_t RANSAC_MINIMUM_INLIERS
void loadFromTextFile(std::istream &f)
Loads a vector/matrix from a text file, compatible with MATLAB text format.



Page generated by Doxygen 1.8.14 for MRPT 2.0.1 Git: 0fef1a6d7 Fri Apr 3 23:00:21 2020 +0200 at vie abr 3 23:20:28 CEST 2020