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 
10 #include <mrpt/graphslam/levmarq.h>
13 #include <mrpt/img/TColor.h>
17 #include <mrpt/random.h>
19 #include <iostream>
20 
21 using namespace mrpt;
22 using namespace mrpt::graphs;
23 using namespace mrpt::graphslam;
24 using namespace mrpt::poses;
25 using namespace mrpt::math;
26 using namespace mrpt::gui;
27 using namespace mrpt::opengl;
28 using namespace mrpt::random;
29 using namespace mrpt::img;
30 using namespace std;
31 using namespace mrpt::system;
32 
33 // Level of noise in nodes initial positions:
34 const double STD_NOISE_NODE_XYZ = 0.5;
35 const double STD_NOISE_NODE_ANG = 5.0_deg;
36 
37 // Level of noise in edges:
38 const double STD_NOISE_EDGE_XYZ = 0.001;
39 const double STD_NOISE_EDGE_ANG = 0.01_deg;
40 
41 const double STD4EDGES_COV_MATRIX = 10;
42 const double ERROR_IN_INCOMPATIBLE_EDGE = 0.3; // ratio [0,1]
43 
44 /**
45  * Generic struct template
46  * Auxiliary class to add a new edge to the graph. The edge is annotated with
47  * the relative position of the two nodes
48  */
49 template <class GRAPH, bool EDGES_ARE_PDF = GRAPH::edge_t::is_PDF_val>
50 struct EdgeAdders;
51 
52 /**
53  * Specific templates based on the above EdgeAdders template
54  * Non-PDF version:
55  */
56 template <class GRAPH>
57 struct EdgeAdders<GRAPH, false>
58 {
59  static const int DIM = GRAPH::edge_t::type_value::static_size;
60  typedef CMatrixFixed<double, DIM, DIM> cov_t;
61 
62  static void addEdge(
63  TNodeID from, TNodeID to,
64  const typename GRAPH::global_poses_t& real_poses, GRAPH& graph,
65  const cov_t& COV_MAT)
66  {
67  /**
68  * No covariance argument here (although it is passed in the function
69  * declaration above)
70  * See also :
71  * https://groups.google.com/d/msg/mrpt-users/Sr9LSydArgY/vRNM5V_uA-oJ
72  * for a more detailed explanation on how it is treated
73  */
74  typename GRAPH::edge_t RelativePose(
75  real_poses.find(to)->second - real_poses.find(from)->second);
76  graph.insertEdge(from, to, RelativePose);
77  }
78 };
79 // PDF version:
80 template <class GRAPH>
81 struct EdgeAdders<GRAPH, true>
82 {
83  static const int DIM = GRAPH::edge_t::type_value::static_size;
84  typedef CMatrixFixed<double, DIM, DIM> cov_t;
85 
86  static void addEdge(
87  TNodeID from, TNodeID to,
88  const typename GRAPH::global_poses_t& real_poses, GRAPH& graph,
89  const cov_t& COV_MAT)
90  {
91  typename GRAPH::edge_t RelativePose(
92  real_poses.find(to)->second - real_poses.find(from)->second,
93  COV_MAT);
94  graph.insertEdge(from, to, RelativePose);
95  }
96 };
97 
98 // Container to handle the propagation of the square root error of the problem
99 vector<double> log_sq_err_evolution;
100 
101 // This example lives inside this template class, which can be instanced for
102 // different kind of graphs (see main()):
103 template <class my_graph_t>
105 {
106  template <class GRAPH_T>
107  static void my_levmarq_feedback(
108  const GRAPH_T& graph, const size_t iter, const size_t max_iter,
109  const double cur_sq_error)
110  {
111  log_sq_err_evolution.push_back(std::log(cur_sq_error));
112  if ((iter % 100) == 0)
113  cout << "Progress: " << iter << " / " << max_iter
114  << ", total sq err = " << cur_sq_error << endl;
115  }
116 
117  // ------------------------------------------------------
118  // GraphSLAMDemo
119  // ------------------------------------------------------
120  void run(bool add_extra_tightening_edge)
121  {
122  // The graph: nodes + edges:
123  my_graph_t graph;
124 
125  // The global poses of the graph nodes (without covariance):
126  typename my_graph_t::global_poses_t real_node_poses;
127 
128  /**
129  * Initialize the PRNG from the given random seed.
130  * Method used to initially randomise the generator
131  */
133 
134  // ----------------------------
135  // Create a random graph:
136  // ----------------------------
137  const size_t N_VERTEX = 50;
138  const double DIST_THRES = 7;
139  const double NODES_XY_MAX = 20;
140 
141  /**
142  * First add all the nodes (so that, when I add edges, I can refer to
143  * them
144  */
145  for (TNodeID j = 0; j < N_VERTEX; j++)
146  {
147  // Use evenly distributed nodes:
148  static double ang = 2 * M_PI / N_VERTEX;
149  const double R = NODES_XY_MAX + 2 * (j % 2 ? 1 : -1);
150  CPose2D p(R * cos(ang * j), R * sin(ang * j), ang);
151 
152  // Save real pose:
153  real_node_poses[j] = p;
154 
155  // Copy the nodes to the graph, and add some noise:
156  graph.nodes[j] = p;
157  }
158 
159  /**
160  * Add some edges
161  * Also initialize the information matrix used for EACH constraint. For
162  * simplicity the same information matrix is used for each one of the
163  * edges. This information matrix is RELATIVE to each constraint/edge
164  * (not in global ref. frame) see also:
165  * https://groups.google.com/d/msg/mrpt-users/Sr9LSydArgY/wYFeU2BXr4kJ
166  */
167  typedef EdgeAdders<my_graph_t> edge_adder_t;
168  typename edge_adder_t::cov_t inf_matrix;
169  inf_matrix.setDiagonal(
170  edge_adder_t::cov_t::RowsAtCompileTime,
172 
173  /**
174  * add the edges using the node ids added to the graph before
175  */
176  for (TNodeID i = 0; i < N_VERTEX; i++)
177  {
178  for (TNodeID j = i + 1; j < N_VERTEX; j++)
179  {
180  if (real_node_poses[i].distanceTo(real_node_poses[j]) <
181  DIST_THRES)
183  i, j, real_node_poses, graph, inf_matrix);
184  }
185  }
186 
187  // Add an additional edge to deform the graph?
188  if (add_extra_tightening_edge)
189  {
190  // inf_matrix.setIdentity(square(1.0/(STD4EDGES_COV_MATRIX)));
192  0, N_VERTEX / 2, real_node_poses, graph, inf_matrix);
193 
194  // Tweak this last node to make it incompatible with the rest:
195  // It must exist, don't check errors...
196  typename my_graph_t::edge_t& ed =
197  graph.edges.find(make_pair(TNodeID(0), TNodeID(N_VERTEX / 2)))
198  ->second;
199  ed.getPoseMean().x(
200  (1 - ERROR_IN_INCOMPATIBLE_EDGE) * ed.getPoseMean().x());
201  }
202 
203  // The root node (the origin of coordinates):
204  graph.root = TNodeID(0);
205 
206  // This is the ground truth graph (make a copy for later use):
207  const my_graph_t graph_GT = graph;
208 
209  cout << "graph nodes: " << graph_GT.nodeCount() << endl;
210  cout << "graph edges: " << graph_GT.edgeCount() << endl;
211 
212  // Add noise to edges & nodes:
213  for (typename my_graph_t::edges_map_t::iterator itEdge =
214  graph.edges.begin();
215  itEdge != graph.edges.end(); ++itEdge)
216  {
217  const typename my_graph_t::edge_t::type_value delta_noise(CPose3D(
218  getRandomGenerator().drawGaussian1D(0, STD_NOISE_EDGE_XYZ),
219  getRandomGenerator().drawGaussian1D(0, STD_NOISE_EDGE_XYZ),
220  getRandomGenerator().drawGaussian1D(0, STD_NOISE_EDGE_XYZ),
221  getRandomGenerator().drawGaussian1D(0, STD_NOISE_EDGE_ANG),
222  getRandomGenerator().drawGaussian1D(0, STD_NOISE_EDGE_ANG),
223  getRandomGenerator().drawGaussian1D(0, STD_NOISE_EDGE_ANG)));
224  itEdge->second.getPoseMean() +=
225  typename my_graph_t::edge_t::type_value(delta_noise);
226  }
227 
228  for (typename my_graph_t::global_poses_t::iterator itNode =
229  graph.nodes.begin();
230  itNode != graph.nodes.end(); ++itNode)
231  if (itNode->first != graph.root)
232  itNode->second.getPoseMean() +=
233  typename my_graph_t::edge_t::type_value(CPose3D(
234  getRandomGenerator().drawGaussian1D(
235  0, STD_NOISE_NODE_XYZ),
236  getRandomGenerator().drawGaussian1D(
237  0, STD_NOISE_NODE_XYZ),
238  getRandomGenerator().drawGaussian1D(
239  0, STD_NOISE_NODE_XYZ),
240  getRandomGenerator().drawGaussian1D(
241  0, STD_NOISE_NODE_ANG),
242  getRandomGenerator().drawGaussian1D(
243  0, STD_NOISE_NODE_ANG),
244  getRandomGenerator().drawGaussian1D(
245  0, STD_NOISE_NODE_ANG)));
246 
247  // This is the initial input graph (make a copy for later use):
248  const my_graph_t graph_initial = graph;
249  // graph_GT.saveToTextFile("test_GT.graph");
250  // graph_initial.saveToTextFile("test.graph");
251 
252  // ----------------------------
253  // Run graph slam:
254  // ----------------------------
256  // params["verbose"] = 1;
257  params["profiler"] = 1;
258  params["max_iterations"] = 500;
259  params["scale_hessian"] = 0.1; // If <1, will "exagerate" the scale of
260  // the gradient and, normally, will
261  // converge much faster.
262  params["tau"] = 1e-3;
263 
264  // e2: Lev-marq algorithm iteration stopping criterion #2: |delta_incr|
265  // < e2*(x_norm+e2)
266  // params["e1"] = 1e-6;
267  // params["e2"] = 1e-6;
268 
270  log_sq_err_evolution.clear();
271 
272  cout << "Global graph RMS error / edge = "
273  << std::sqrt(graph.getGlobalSquareError(false) / graph.edgeCount())
274  << endl;
275  cout << "Global graph RMS error / edge = "
276  << std::sqrt(graph.getGlobalSquareError(true) / graph.edgeCount())
277  << " (ignoring information matrices)." << endl;
278 
279  // Do the optimization
281  graph, levmarq_info,
282  nullptr, // List of nodes to optimize. nullptr -> all but the root
283  // node.
284  params, &my_levmarq_feedback<my_graph_t>);
285 
286  cout << "Global graph RMS error / edge = "
287  << std::sqrt(graph.getGlobalSquareError(false) / graph.edgeCount())
288  << endl;
289  cout << "Global graph RMS error / edge = "
290  << std::sqrt(graph.getGlobalSquareError(true) / graph.edgeCount())
291  << " (ignoring information matrices)." << endl;
292 
293  // ----------------------------
294  // Display results:
295  // ----------------------------
296  CDisplayWindow3D win("graph-slam demo");
297 
298  // The final optimized graph:
299  TParametersDouble graph_render_params1;
300  graph_render_params1["show_edges"] = 1;
301  graph_render_params1["edge_width"] = 1;
302  graph_render_params1["nodes_corner_scale"] = 1;
303  CSetOfObjects::Ptr gl_graph1 =
305  graph, graph_render_params1);
306 
307  // The initial noisy graph:
308  TParametersDouble graph_render_params2;
309  graph_render_params2["show_ground_grid"] = 0;
310  graph_render_params2["show_edges"] = 0;
311  graph_render_params2["show_node_corners"] = 0;
312  graph_render_params2["nodes_point_size"] = 7;
313 
314  CSetOfObjects::Ptr gl_graph2 =
316  graph_initial, graph_render_params2);
317 
318  graph_render_params2["nodes_point_size"] = 5;
319  CSetOfObjects::Ptr gl_graph5 =
321  graph, graph_render_params2);
322 
323  // The ground truth graph:
324  TParametersDouble graph_render_params3;
325  graph_render_params3["show_ground_grid"] = 0;
326  graph_render_params3["show_ID_labels"] = 1;
327  graph_render_params3["show_edges"] = 1;
328  graph_render_params3["edge_width"] = 3;
329  graph_render_params3["nodes_corner_scale"] = 2;
330  CSetOfObjects::Ptr gl_graph3 =
332  graph_GT, graph_render_params3);
333  CSetOfObjects::Ptr gl_graph4 =
335  graph_initial, graph_render_params3);
336 
337  win.addTextMessage(
338  5, 5, "Ground truth: Big corners & thick edges",
339  1000 /* arbitrary, unique text label ID */);
340  win.addTextMessage(
341  5, 5 + 15, "Initial graph: Gray thick points.",
342  1001 /* arbitrary, unique text label ID */);
343  win.addTextMessage(
344  5, 5 + 30, "Final graph: Small corners & thin edges",
345  1002 /* arbitrary, unique text label ID */);
346 
347  {
348  COpenGLScene::Ptr& scene = win.get3DSceneAndLock();
349  scene->insert(gl_graph1);
350  scene->insert(gl_graph3);
351  scene->insert(gl_graph2);
352  scene->insert(gl_graph5);
353  win.unlockAccess3DScene();
354  win.repaint();
355  }
356 
357  // Show progress of error:
358  CDisplayWindowPlots win_err("Evolution of log(sq. error)");
359  win_err.plot(log_sq_err_evolution, "-b");
360  win_err.axis_fit();
361 
362  // wait end:
363  cout << "Close any window to end...\n";
364  while (win.isOpen() && win_err.isOpen() && !mrpt::system::os::kbhit())
365  {
366  std::this_thread::sleep_for(10ms);
367  }
368  }
369 };
370 
371 int main()
372 {
373  try
374  {
375  // typedef CNetworkOfPoses<CPose2D,map_traits_map_as_vector> my_graph_t;
376 
377  cout << "Select the type of graph to optimize:\n"
378  "1. CNetworkOfPoses2D \n"
379  "2. CNetworkOfPoses2DInf \n"
380  "3. CNetworkOfPoses3D \n"
381  "4. CNetworkOfPoses3DInf \n";
382 
383  cout << ">> ";
384 
385  int i = 0;
386  {
387  string l;
388  std::getline(cin, l);
389  l = mrpt::system::trim(l);
390  i = atoi(&l[0]);
391  }
392 
393  bool add_extra_tightening_edge;
394  cout << "Add an extra, incompatible tightening edge? [y/N] ";
395  {
396  string l;
397  std::getline(cin, l);
398  l = mrpt::system::trim(l);
399  add_extra_tightening_edge = l[0] == 'Y' || l[0] == 'y';
400  }
401 
402  switch (i)
403  {
404  case 1:
405  {
407  demo.run(add_extra_tightening_edge);
408  }
409  break;
410  case 2:
411  {
413  demo.run(add_extra_tightening_edge);
414  }
415  break;
416  case 3:
417  {
419  demo.run(add_extra_tightening_edge);
420  }
421  break;
422  case 4:
423  {
425  demo.run(add_extra_tightening_edge);
426  }
427  break;
428  };
429 
430  std::this_thread::sleep_for(20ms);
431  return 0;
432  }
433  catch (exception& e)
434  {
435  cout << "MRPT exception caught: " << e.what() << endl;
436  return -1;
437  }
438  catch (...)
439  {
440  printf("Another exception!!");
441  return -1;
442  }
443 }
A namespace of pseudo-random numbers generators of diferent distributions.
Generic struct template Auxiliary class to add a new edge to the graph.
void addEdge(TNodeID from, TNodeID to, const std::map< TNodeID, CPose2D > &real_poses, CNetworkOfPoses2D &graph_links)
void optimize_graph_spa_levmarq(GRAPH_T &graph, TResultInfoSpaLevMarq &out_info, const std::set< mrpt::graphs::TNodeID > *in_nodes_to_optimize=nullptr, const mrpt::system::TParametersDouble &extra_params=mrpt::system::TParametersDouble(), FEEDBACK_CALLABLE functor_feedback=FEEDBACK_CALLABLE())
Optimize a graph of pose constraints using the Sparse Pose Adjustment (SPA) sparse representation and...
Definition: levmarq.h:79
Create a GUI window and display plots with MATLAB-like interfaces and commands.
Abstract graph and tree data structures, plus generic graph algorithms.
void randomize(const uint32_t seed)
Initialize the PRNG from the given random seed.
mrpt::vision::TStereoCalibParams params
STL namespace.
SLAM methods related to graphs of pose constraints.
const double STD_NOISE_NODE_XYZ
This base provides a set of functions for maths stuff.
mrpt::gui::CDisplayWindow3D::Ptr win
Classes for 2D/3D geometry representation, both of single values and probability density distribution...
const double STD4EDGES_COV_MATRIX
return_t square(const num_t x)
Inline function for the square of a number.
const double STD_NOISE_EDGE_ANG
This is the global namespace for all Mobile Robot Programming Toolkit (MRPT) libraries.
void run(bool add_extra_tightening_edge)
A class used to store a 2D pose, including the 2D coordinate point and a heading (phi) angle...
Definition: CPose2D.h:39
const float R
A class used to store a 3D pose (a 3D translation + a rotation in 3D).
Definition: CPose3D.h:85
Output information for mrpt::graphslam::optimize_graph_spa_levmarq()
The namespace for 3D scene representation and rendering.
Definition: CGlCanvasBase.h:13
bool kbhit() noexcept
An OS-independent version of kbhit, which returns true if a key has been pushed.
Definition: os.cpp:392
CSetOfObjects::Ptr graph_visualize(const GRAPH_T &g, const mrpt::system::TParametersDouble &extra_params=mrpt::system::TParametersDouble())
Returns an opengl objects representation of an arbitrary graph, as a network of 3D pose frames...
uint64_t TNodeID
A generic numeric type for unique IDs of nodes or entities.
Definition: TNodeID.h:16
std::string trim(const std::string &str)
Removes leading and trailing spaces.
const double STD_NOISE_NODE_ANG
Classes for creating GUI windows for 2D and 3D visualization.
Definition: about_box.h:14
CRandomGenerator & getRandomGenerator()
A static instance of a CRandomGenerator class, for use in single-thread applications.
const double STD_NOISE_EDGE_XYZ
A graphical user interface (GUI) for efficiently rendering 3D scenes in real-time.
const double ERROR_IN_INCOMPATIBLE_EDGE
vector< double > log_sq_err_evolution



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