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 /*
11  Example : kinect-to-2d-laser-demo
12  Web page : https://www.mrpt.org/Example_Kinect_To_2D_laser_scan (includes
13  video demo)
14 
15  Purpose : Demonstrate grabbing from CKinect, multi-threading
16  and converting the 3D range data into an equivalent
17  2D planar scan.
18 */
19 
22 #include <mrpt/hwdrivers/CKinect.h>
23 #include <mrpt/img/TColor.h>
24 #include <mrpt/opengl/CFrustum.h>
29 #include <mrpt/system/CTicTac.h>
30 #include <mrpt/system/filesystem.h>
31 #include <iostream>
32 
33 using namespace mrpt;
34 using namespace mrpt::hwdrivers;
35 using namespace mrpt::math;
36 using namespace mrpt::gui;
37 using namespace mrpt::obs;
38 using namespace mrpt::maps;
39 using namespace mrpt::opengl;
40 using namespace mrpt::system;
41 using namespace mrpt::img;
42 using namespace std;
43 
44 // Thread for grabbing: Do this is another thread so we divide rendering and
45 // grabbing
46 // and exploit multicore CPUs.
47 struct TThreadParam
48 {
49  TThreadParam() {}
50  volatile bool quit{false};
51  volatile int pushed_key{0};
52  volatile double Hz{0};
53 
54  CObservation3DRangeScan::Ptr new_obs; // RGB+D (+3D points)
55  CObservationIMU::Ptr new_obs_imu; // Accelerometers
56 };
57 
59 {
60  try
61  {
62  CKinect kinect;
63 
64  // Set params:
65  // kinect.enableGrab3DPoints(true);
66  // kinect.enablePreviewRGB(true);
67  //...
68  const std::string cfgFile = "kinect_calib.cfg";
69  if (mrpt::system::fileExists(cfgFile))
70  {
71  cout << "Loading calibration from: " << cfgFile << endl;
72  kinect.loadConfig(mrpt::config::CConfigFile(cfgFile), "KINECT");
73  }
74  else
75  cerr << "Warning: Calibration file [" << cfgFile
76  << "] not found -> Using default params.\n";
77 
78  // Open:
79  cout << "Calling CKinect::initialize()...";
80  kinect.initialize();
81  cout << "OK\n";
82 
83  CTicTac tictac;
84  int nImgs = 0;
85  bool there_is_obs = true, hard_error = false;
86 
87  while (!hard_error && !p.quit)
88  {
89  // Grab new observation from the camera:
90  auto obs = CObservation3DRangeScan::Create(); // Smart pointers to
91  // observations
92  CObservationIMU::Ptr obs_imu = CObservationIMU::Create();
93 
94  kinect.getNextObservation(*obs, *obs_imu, there_is_obs, hard_error);
95 
96  if (!hard_error && there_is_obs)
97  {
98  std::atomic_store(&p.new_obs, obs);
99  std::atomic_store(&p.new_obs_imu, obs_imu);
100  }
101 
102  if (p.pushed_key != 0)
103  {
104  switch (p.pushed_key)
105  {
106  case 27:
107  p.quit = true;
108  break;
109  }
110 
111  // Clear pushed key flag:
112  p.pushed_key = 0;
113  }
114 
115  nImgs++;
116  if (nImgs > 10)
117  {
118  p.Hz = nImgs / tictac.Tac();
119  nImgs = 0;
120  tictac.Tic();
121  }
122  }
123  }
124  catch (const std::exception& e)
125  {
126  cout << "Exception in Kinect thread: " << e.what() << endl;
127  p.quit = true;
128  }
129 }
130 
131 // ------------------------------------------------------
132 // Test_Kinect
133 // ------------------------------------------------------
134 void Test_Kinect()
135 {
136  // Launch grabbing thread:
137  // --------------------------------------------------------
138  TThreadParam thrPar;
139  std::thread thHandle(thread_grabbing, std::ref(thrPar));
140 
141  // Wait until data stream starts so we can say for sure the sensor has been
142  // initialized OK:
143  cout << "Waiting for sensor initialization...\n";
144  do
145  {
146  CObservation3DRangeScan::Ptr possiblyNewObs =
147  std::atomic_load(&thrPar.new_obs);
148  if (possiblyNewObs && possiblyNewObs->timestamp != INVALID_TIMESTAMP)
149  break;
150  else
151  std::this_thread::sleep_for(10ms);
152  } while (!thrPar.quit);
153 
154  // Check error condition:
155  if (thrPar.quit) return;
156 
157  // Create window and prepare OpenGL object in the scene:
158  // --------------------------------------------------------
159  mrpt::gui::CDisplayWindow3D win3D("Kinect 3D -> 2D laser scan", 800, 600);
160 
161  win3D.setCameraAzimuthDeg(140);
162  win3D.setCameraElevationDeg(30);
163  win3D.setCameraZoom(10.0);
164  win3D.setFOV(90);
165  win3D.setCameraPointingToPoint(2.5, 0, 0);
166 
167  // The 3D point cloud OpenGL object:
170  gl_points->setPointSize(2.5);
171 
172  // The 2D "laser scan" OpenGL object:
175  gl_2d_scan->enablePoints(true);
176  gl_2d_scan->enableLine(true);
177  gl_2d_scan->enableSurface(true);
178  gl_2d_scan->setSurfaceColor(0, 0, 1, 0.3); // RGBA
179 
181  0.2f, 5.0f, 90.0f, 5.0f, 2.0f, true, true);
182 
183  const double aspect_ratio =
184  480.0 / 640.0; // kinect.rows() / double( kinect.cols() );
185 
187  viewInt; // Extra viewports for the RGB & D images.
188  {
189  mrpt::opengl::COpenGLScene::Ptr& scene = win3D.get3DSceneAndLock();
190 
191  // Create the Opengl object for the point cloud:
192  scene->insert(gl_points);
193  scene->insert(gl_2d_scan);
194  scene->insert(gl_frustum);
195 
196  {
199  gl_grid->setColor(0.6, 0.6, 0.6);
200  scene->insert(gl_grid);
201  }
202  {
205  gl_corner->setScale(0.2);
206  scene->insert(gl_corner);
207  }
208 
209  const int VW_WIDTH =
210  250; // Size of the viewport into the window, in pixel units.
211  const int VW_HEIGHT = aspect_ratio * VW_WIDTH;
212  const int VW_GAP = 30;
213 
214  // Create the Opengl objects for the planar images, as textured planes,
215  // each in a separate viewport:
216  win3D.addTextMessage(
217  30, -25 - 1 * (VW_GAP + VW_HEIGHT), "Range data", 1);
218  viewRange = scene->createViewport("view2d_range");
219  viewRange->setViewportPosition(
220  5, -10 - 1 * (VW_GAP + VW_HEIGHT), VW_WIDTH, VW_HEIGHT);
221 
222  win3D.addTextMessage(
223  30, -25 - 2 * (VW_GAP + VW_HEIGHT), "Intensity data", 2);
224  viewInt = scene->createViewport("view2d_int");
225  viewInt->setViewportPosition(
226  5, -10 - 2 * (VW_GAP + VW_HEIGHT), VW_WIDTH, VW_HEIGHT);
227 
228  win3D.unlockAccess3DScene();
229  win3D.repaint();
230  }
231 
233  CObservationIMU::Ptr last_obs_imu;
234 
235  while (win3D.isOpen() && !thrPar.quit)
236  {
237  CObservation3DRangeScan::Ptr possiblyNewObs =
238  std::atomic_load(&thrPar.new_obs);
239  if (possiblyNewObs && possiblyNewObs->timestamp != INVALID_TIMESTAMP &&
240  (!last_obs || possiblyNewObs->timestamp != last_obs->timestamp))
241  {
242  // It IS a new observation:
243  last_obs = possiblyNewObs;
244  last_obs_imu = std::atomic_load(&thrPar.new_obs_imu);
245 
246  // Update visualization ---------------------------------------
247  bool do_refresh = false;
248 
249  // Show 2D ranges as a grayscale image:
250  if (last_obs->hasRangeImage)
251  {
252  mrpt::img::CImage img;
253 
254  // Normalize the image
255  static CMatrixFloat range2D; // Static to save time allocating
256  // the matrix in every iteration
257  range2D = last_obs->rangeImage;
258  range2D *= (1.0 / last_obs->maxRange);
259 
260  img.setFromMatrix(range2D);
261 
262  win3D.get3DSceneAndLock();
263  viewRange->setImageView(std::move(img));
264  win3D.unlockAccess3DScene();
265  do_refresh = true;
266  }
267 
268  // Convert ranges to an equivalent 2D "fake laser" scan:
269  if (last_obs->hasRangeImage)
270  {
271  // Convert to scan:
273  CObservation2DRangeScan::Create();
274  const float vert_FOV = DEG2RAD(gl_frustum->getVertFOV());
275 
277  sp.angle_inf = .5f * vert_FOV;
278  sp.angle_sup = .5f * vert_FOV;
279  sp.sensorLabel = "KINECT_2D_SCAN";
280 
281  last_obs->convertTo2DScan(*obs_2d, sp);
282 
283  // And load scan in the OpenGL object:
284  gl_2d_scan->setScan(*obs_2d);
285  }
286 
287  // Show intensity image:
288  if (last_obs->hasIntensityImage)
289  {
290  win3D.get3DSceneAndLock();
291  viewInt->setImageView(
292  last_obs->intensityImage); // This is not "_fast" since the
293  // intensity image is used below
294  // in the coloured point cloud.
295  win3D.unlockAccess3DScene();
296  do_refresh = true;
297  }
298 
299  // Show 3D points:
300  if (last_obs->hasPoints3D)
301  {
302  win3D.get3DSceneAndLock();
305 
306  last_obs->unprojectInto(*gl_points, pp);
307  win3D.unlockAccess3DScene();
308  do_refresh = true;
309  }
310 
311  // Some text messages:
312  win3D.get3DSceneAndLock();
313  // Estimated grabbing rate:
314  win3D.addTextMessage(-100, -20, format("%.02f Hz", thrPar.Hz), 100);
315 
316  win3D.addTextMessage(
317  10, 10,
318  "'o'/'i'-zoom out/in, '2'/'3'/'f':show/hide 2D/3D/frustum "
319  "data, mouse: orbit 3D, ESC: quit",
320  110);
321 
322  win3D.addTextMessage(
323  10, 25,
324  format(
325  "Show: 3D=%s 2D=%s Frustum=%s (vert. FOV=%.1fdeg)",
326  gl_points->isVisible() ? "YES" : "NO",
327  gl_2d_scan->isVisible() ? "YES" : "NO",
328  gl_frustum->isVisible() ? "YES" : "NO",
329  gl_frustum->getVertFOV()),
330  111);
331  win3D.unlockAccess3DScene();
332 
333  // Do we have accelerometer data?
334  if (last_obs_imu && last_obs_imu->dataIsPresent[IMU_X_ACC])
335  {
336  win3D.get3DSceneAndLock();
337  win3D.addTextMessage(
338  10, 65,
339  format(
340  "Acc: x=%.02f y=%.02f z=%.02f",
341  last_obs_imu->rawMeasurements[IMU_X_ACC],
342  last_obs_imu->rawMeasurements[IMU_Y_ACC],
343  last_obs_imu->rawMeasurements[IMU_Z_ACC]),
344  102);
345  win3D.unlockAccess3DScene();
346  do_refresh = true;
347  }
348 
349  // Force opengl repaint:
350  if (do_refresh) win3D.repaint();
351 
352  } // end update visualization:
353 
354  // Process possible keyboard commands:
355  // --------------------------------------
356  if (win3D.keyHit() && thrPar.pushed_key == 0)
357  {
358  const int key = tolower(win3D.getPushedKey());
359 
360  switch (key)
361  {
362  // Some of the keys are processed in this thread:
363  case 'o':
364  win3D.setCameraZoom(win3D.getCameraZoom() * 1.2);
365  win3D.repaint();
366  break;
367  case 'i':
368  win3D.setCameraZoom(win3D.getCameraZoom() / 1.2);
369  win3D.repaint();
370  break;
371  case '2':
372  gl_2d_scan->setVisibility(!gl_2d_scan->isVisible());
373  break;
374  case '3':
375  gl_points->setVisibility(!gl_points->isVisible());
376  break;
377  case 'F':
378  case 'f':
379  gl_frustum->setVisibility(!gl_frustum->isVisible());
380  break;
381  case '+':
382  gl_frustum->setVertFOV(gl_frustum->getVertFOV() + 1);
383  break;
384  case '-':
385  gl_frustum->setVertFOV(gl_frustum->getVertFOV() - 1);
386  break;
387  // ...and the rest in the kinect thread:
388  default:
389  thrPar.pushed_key = key;
390  break;
391  };
392  }
393 
394  std::this_thread::sleep_for(1ms);
395  }
396 
397  cout << "Waiting for grabbing thread to exit...\n";
398  thrPar.quit = true;
399  thHandle.join();
400  cout << "Bye!\n";
401 }
402 
403 int main(int argc, char** argv)
404 {
405  try
406  {
407  Test_Kinect();
408 
409  std::this_thread::sleep_for(50ms);
410  return 0;
411  }
412  catch (const std::exception& e)
413  {
414  std::cerr << "MRPT error: " << mrpt::exception_to_str(e) << std::endl;
415  return -1;
416  }
417  catch (...)
418  {
419  printf("Another exception!!");
420  return -1;
421  }
422 }
double Tac() noexcept
Stops the stopwatch.
Definition: CTicTac.cpp:86
static Ptr Create(Args &&... args)
std::string std::string format(std::string_view fmt, ARGS &&... args)
Definition: format.h:26
static Ptr Create(Args &&... args)
This class allows loading and storing values and vectors of different types from ".ini" files easily.
bool fileExists(const std::string &fileName)
Test if a given file (or directory) exists.
Definition: filesystem.cpp:128
A high-performance stopwatch, with typical resolution of nanoseconds.
CObservationIMU::Ptr new_obs_imu
Contains classes for various device interfaces.
y-axis acceleration (local/vehicle frame) (m/sec2)
STL namespace.
z-axis acceleration (local/vehicle frame) (m/sec2)
const float vert_FOV
volatile bool quit
In/Out variable: Forces the thread to exit or indicates an error in the thread that caused it to end...
A class for grabing "range images", intensity images (either RGB or IR) and other information from an...
Definition: CKinect.h:265
This base provides a set of functions for maths stuff.
constexpr double DEG2RAD(const double x)
Degrees to radians.
Used in CObservation3DRangeScan::convertTo2DScan()
This namespace contains representation of robot actions and observations.
void initialize() override
Initializes the 3D camera - should be invoked after calling loadConfig() or setting the different par...
Definition: CKinect.cpp:135
void setFromMatrix(const MAT &m, bool matrix_is_normalized=true)
Set the image from a matrix, interpreted as grayscale intensity values, in the range [0...
Definition: img/CImage.h:844
static Ptr Create(Args &&... args)
Definition: CFrustum.h:55
void loadConfig(const mrpt::config::CConfigFileBase &configSource, const std::string &section)
Loads the generic settings common to any sensor (See CGenericSensor), then call to "loadConfig_sensor...
void Test_Kinect()
void getNextObservation(mrpt::obs::CObservation3DRangeScan &out_obs, bool &there_is_obs, bool &hardware_error)
The main data retrieving function, to be called after calling loadConfig() and initialize().
Definition: CKinect.cpp:504
This is the global namespace for all Mobile Robot Programming Toolkit (MRPT) libraries.
const char * argv[]
CSetOfObjects::Ptr CornerXYZ(float scale=1.0)
Returns three arrows representing a X,Y,Z 3D corner.
void thread_grabbing(TThreadParam &p)
Used in CObservation3DRangeScan::unprojectInto()
The namespace for 3D scene representation and rendering.
Definition: CGlCanvasBase.h:13
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
Classes for creating GUI windows for 2D and 3D visualization.
Definition: about_box.h:14
const int argc
CObservation3DRangeScan::Ptr new_obs
RGB+D (+ optionally, 3D point cloud)
void Tic() noexcept
Starts the stopwatch.
Definition: CTicTac.cpp:75
This template class provides the basic functionality for a general 2D any-size, resizable container o...
x-axis acceleration (local/vehicle frame) (m/sec2)
static Ptr Create(Args &&... args)
Definition: CGridPlaneXY.h:31
volatile double Hz
Out variable: Approx.
#define INVALID_TIMESTAMP
Represents an invalid timestamp, where applicable.
Definition: datetime.h:43
bool takeIntoAccountSensorPoseOnRobot
(Default: false) If false, local (sensor-centric) coordinates of points are generated.
A class for storing images as grayscale or RGB bitmaps.
Definition: img/CImage.h:148
A graphical user interface (GUI) for efficiently rendering 3D scenes in real-time.



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