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-online-offline-demo
12  Web page :
13  https://www.mrpt.org/Switching_between_reading_live_Kinect_RGBD_dataset_for_debugging
14 
15  Purpose : Demonstrate how to programmatically switch between online Kinect
16  grabbing and offline parsing a Rawlog dataset. Refer to the launch
17  of the grabbing thread in Test_KinectOnlineOffline()
18 */
19 
20 #include <mrpt/gui.h>
21 #include <mrpt/hwdrivers/CKinect.h>
22 #include <mrpt/img/TColor.h>
25 #include <mrpt/obs/CRawlog.h>
26 #include <mrpt/opengl/CFrustum.h>
32 #include <mrpt/system/CTicTac.h>
34 #include <iostream>
35 #include <memory>
36 
37 using namespace mrpt;
38 using namespace mrpt::hwdrivers;
39 using namespace mrpt::gui;
40 using namespace mrpt::obs;
41 using namespace mrpt::system;
42 using namespace mrpt::img;
43 using namespace mrpt::serialization;
44 using namespace mrpt::io;
45 using namespace std;
46 
47 // Thread for online/offline capturing: This should be done in another thread
48 // specially in the online mode, but also in offline to emulate the
49 // possibility
50 // of losing frames if our program doesn't process them quickly.
51 struct TThreadParam
52 {
54  bool _is_online, const string& _rawlog_file = string(),
55  bool _generate_3D_pointcloud_in_this_thread = false)
56  : is_online(_is_online),
57  rawlog_file(_rawlog_file),
58  generate_3D_pointcloud_in_this_thread(
59  _generate_3D_pointcloud_in_this_thread),
60  quit(false),
61  Hz(0)
62  {
63  }
64 
65  /** true: live grabbing from the sensor, false: from rawlog */
66  const bool is_online;
67  /** Only when is_online==false */
68  const string rawlog_file;
69  /** true: populate the 3D point fields in the output observation; false:
70  * only RGB and Depth images. */
71  const bool generate_3D_pointcloud_in_this_thread;
72 
73  /** In/Out variable: Forces the thread to exit or indicates an error in the
74  * thread that caused it to end. */
75  volatile bool quit;
76  /** Out variable: Approx. capturing rate from the thread. */
77  volatile double Hz;
78 
79  /** RGB+D (+ optionally, 3D point cloud) */
81 };
82 
83 // Only for offline operation:
84 // Uncoment to force the simulated sensor to run at this given rate.
85 // If commented out, the simulated sensor will reproduce the real rate
86 // as indicated by timestamps in the dataset.
87 //#define FAKE_KINECT_FPS_RATE 10 // In Hz
88 
89 // ----------------------------------------------------
90 // The online/offline grabbing thread
91 // ----------------------------------------------------
93 {
94  try
95  {
96  typedef std::unique_ptr<CKinect>
97  CKinectPtr; // This assures automatic destruction
98 
99  // Only one of these will be actually used:
100  CKinectPtr kinect;
101  CFileGZInputStream dataset;
102 
103  mrpt::system::TTimeStamp dataset_prev_tim = INVALID_TIMESTAMP;
104  mrpt::system::TTimeStamp my_last_read_obs_tim = INVALID_TIMESTAMP;
105 
106  if (p.is_online)
107  {
108  // Online:
109  // ---------------------
110  kinect.reset(new CKinect());
111 
112  // Set params:
113  kinect->enableGrabRGB(true);
114  kinect->enableGrabDepth(true);
115  kinect->enableGrabAccelerometers(false);
116  kinect->enableGrab3DPoints(p.generate_3D_pointcloud_in_this_thread);
117 
118  // Open:
119  cout << "Calling CKinect::initialize()...";
120  kinect->initialize();
121  cout << "OK\n";
122  }
123  else
124  {
125  // Offline:
126  // ---------------------
127  if (!dataset.open(p.rawlog_file))
128  throw std::runtime_error(
129  "Couldn't open rawlog dataset file for input...");
130 
131  // Set external images directory:
132  CImage::setImagesPathBase(
133  CRawlog::detectImagesDirectory(p.rawlog_file));
134  }
135 
136  CTicTac tictac;
137  int nImgs = 0;
138 
139  while (!p.quit)
140  {
141  if (p.is_online)
142  {
143  // Grab new observation from the camera:
144  bool there_is_obs = true, hard_error = false;
145 
146  // Smart pointers to observations. Memory pooling used inside
147  auto obs = CObservation3DRangeScan::Create();
148  kinect->getNextObservation(*obs, there_is_obs, hard_error);
149 
150  if (hard_error)
151  throw std::runtime_error(
152  "Sensor returned 'hardware error'");
153  else if (there_is_obs)
154  {
155  // Send object to the main thread:
156  std::atomic_store(&p.new_obs, obs);
157  }
158  }
159  else
160  {
161  // Offline:
162  CSerializable::Ptr obs;
163  do
164  {
165  try
166  {
167  archiveFrom(dataset) >> obs;
168  }
169  catch (const std::exception& e)
170  {
171  throw std::runtime_error(
172  string(
173  "\nError reading from dataset file (EOF?):\n") +
174  string(e.what()));
175  }
176  ASSERT_(obs);
177  } while (!IS_CLASS(*obs, CObservation3DRangeScan));
178 
179  // We have one observation:
181  std::dynamic_pointer_cast<CObservation3DRangeScan>(obs);
182  obs3D->load(); // *Important* This is needed to load the range
183  // image if stored as a separate file.
184 
185  // Do we have to wait to emulate real-time behavior?
186  const mrpt::system::TTimeStamp cur_tim = obs3D->timestamp;
188 
189  if (dataset_prev_tim != INVALID_TIMESTAMP &&
190  my_last_read_obs_tim != INVALID_TIMESTAMP)
191  {
192 #ifndef FAKE_KINECT_FPS_RATE
193  const double At_dataset =
194  mrpt::system::timeDifference(dataset_prev_tim, cur_tim);
195 #else
196  const double At_dataset = 1.0 / FAKE_KINECT_FPS_RATE;
197 #endif
198  const double At_actual = mrpt::system::timeDifference(
199  my_last_read_obs_tim, now_tim);
200 
201  const double need_to_wait_ms =
202  1000. * (At_dataset - At_actual);
203  // cout << "[Kinect grab thread] Need to wait (ms): " <<
204  // need_to_wait_ms << endl;
205  if (need_to_wait_ms > 0)
206  std::this_thread::sleep_for(
207  std::chrono::duration<double, std::milli>(
208  need_to_wait_ms));
209  }
210 
211  // Send observation to main thread:
212  std::atomic_store(&p.new_obs, obs3D);
213 
214  dataset_prev_tim = cur_tim;
215  my_last_read_obs_tim = mrpt::system::now();
216  }
217 
218  // Update Hz rate estimate
219  nImgs++;
220  if (nImgs > 10)
221  {
222  p.Hz = nImgs / tictac.Tac();
223  nImgs = 0;
224  tictac.Tic();
225  }
226  }
227  }
228  catch (const std::exception& e)
229  {
230  cout << "Exception in Kinect thread: " << e.what() << endl;
231  p.quit = true;
232  }
233 }
234 
235 // ------------------------------------------------------
236 // Test_KinectOnlineOffline
237 // ------------------------------------------------------
239  bool is_online, const string& rawlog_file = string())
240 {
241  // Launch grabbing thread:
242  // --------------------------------------------------------
243  TThreadParam thrPar(
244  is_online, rawlog_file,
245  false // generate_3D_pointcloud_in_this_thread -> Don't, we'll do it in
246  // this main thread.
247  );
248 
249  std::thread thHandle(thread_grabbing, std::ref(thrPar));
250 
251  // Wait until data stream starts so we can say for sure the sensor has been
252  // initialized OK:
253  cout << "Waiting for sensor initialization...\n";
254  do
255  {
256  CObservation3DRangeScan::Ptr newObs = std::atomic_load(&thrPar.new_obs);
257  if (newObs && newObs->timestamp != INVALID_TIMESTAMP)
258  break;
259  else
260  std::this_thread::sleep_for(10ms);
261  } while (!thrPar.quit);
262 
263  // Check error condition:
264  if (thrPar.quit) return;
265  cout << "OK! Sensor started to emit observations.\n";
266 
267  // Create window and prepare OpenGL object in the scene:
268  // --------------------------------------------------------
269  mrpt::gui::CDisplayWindow3D win3D("Kinect 3D view", 800, 600);
270 
271  win3D.setCameraAzimuthDeg(140);
272  win3D.setCameraElevationDeg(20);
273  win3D.setCameraZoom(8.0);
274  win3D.setFOV(90);
275  win3D.setCameraPointingToPoint(2.5, 0, 0);
276 
279  gl_points->setPointSize(2.5);
280 
282  viewInt; // Extra viewports for the RGB images.
283  {
284  mrpt::opengl::COpenGLScene::Ptr& scene = win3D.get3DSceneAndLock();
285 
286  // Create the Opengl object for the point cloud:
287  scene->insert(gl_points);
288  scene->insert(mrpt::opengl::CGridPlaneXY::Create());
290 
291  const double aspect_ratio = 480.0 / 640.0;
292  const int VW_WIDTH =
293  400; // Size of the viewport into the window, in pixel units.
294  const int VW_HEIGHT = aspect_ratio * VW_WIDTH;
295 
296  // Create an extra opengl viewport for the RGB image:
297  viewInt = scene->createViewport("view2d_int");
298  viewInt->setViewportPosition(5, 30, VW_WIDTH, VW_HEIGHT);
299  win3D.addTextMessage(10, 30 + VW_HEIGHT + 10, "Intensity data", 2);
300 
301  win3D.addTextMessage(5, 5, "'o'/'i'-zoom out/in, ESC: quit", 110);
302 
303  win3D.unlockAccess3DScene();
304  win3D.repaint();
305  }
306 
308 
309  while (win3D.isOpen() && !thrPar.quit)
310  {
311  CObservation3DRangeScan::Ptr newObs = std::atomic_load(&thrPar.new_obs);
312  if (newObs && newObs->timestamp != INVALID_TIMESTAMP &&
313  newObs->timestamp != last_obs_tim)
314  {
315  // It IS a new observation:
316  last_obs_tim = newObs->timestamp;
317 
318  // Update visualization ---------------------------------------
319 
320  win3D.get3DSceneAndLock();
321 
322  // Estimated grabbing rate:
323  win3D.addTextMessage(
324  -350, -13,
325  format(
326  "Timestamp: %s",
327  mrpt::system::dateTimeLocalToString(last_obs_tim).c_str()),
328  100);
329  win3D.addTextMessage(-100, -30, format("%.02f Hz", thrPar.Hz), 101);
330 
331  // Show intensity image:
332  if (newObs->hasIntensityImage)
333  {
334  viewInt->setImageView(newObs->intensityImage); // This is not
335  // "_fast" since
336  // the intensity
337  // image may be
338  // needed later
339  // on.
340  }
341  win3D.unlockAccess3DScene();
342 
343  // -------------------------------------------------------
344  // Create 3D points from RGB+D data
345  //
346  // There are several methods to do this.
347  // Switch the #if's to select among the options:
348  // See also:
349  // https://www.mrpt.org/Generating_3D_point_clouds_from_RGB_D_observations
350  // -------------------------------------------------------
351  if (newObs->hasRangeImage)
352  {
353  static mrpt::system::CTimeLogger logger;
354  logger.enter("RGBD->3D");
355 
356 // Pathway: RGB+D --> PCL <PointXYZ> --> XYZ opengl
357 #if 0
358  static pcl::PointCloud<pcl::PointXYZ> cloud;
359  logger.enter("RGBD->3D.projectInto");
360  newObs->unprojectInto(cloud, false /* without obs.sensorPose */);
361  logger.leave("RGBD->3D.projectInto");
362 
363  win3D.get3DSceneAndLock();
364  logger.enter("RGBD->3D.load in OpenGL");
365  gl_points->loadFromPointsMap(&cloud);
366  logger.leave("RGBD->3D.load in OpenGL");
367  win3D.unlockAccess3DScene();
368 #endif
369 
370 // Pathway: RGB+D --> PCL <PointXYZRGB> --> XYZ+RGB opengl
371 #if 0
372  static pcl::PointCloud<pcl::PointXYZRGB> cloud;
373  logger.enter("RGBD->3D.projectInto");
374  newObs->unprojectInto(cloud, false /* without obs.sensorPose */);
375  logger.leave("RGBD->3D.projectInto");
376 
377  win3D.get3DSceneAndLock();
378  logger.enter("RGBD->3D.load in OpenGL");
379  gl_points->loadFromPointsMap(&cloud);
380  logger.leave("RGBD->3D.load in OpenGL");
381  win3D.unlockAccess3DScene();
382 #endif
383 
384 // Pathway: RGB+D --> XYZ+RGB opengl
385 #if 1
386  win3D.get3DSceneAndLock();
387  logger.enter("RGBD->3D.projectInto");
390 
391  newObs->unprojectInto(*gl_points, pp);
392 
393  logger.leave("RGBD->3D.projectInto");
394  win3D.unlockAccess3DScene();
395 #endif
396 
397 // Pathway: RGB+D --> XYZ+RGB opengl (With a 6D global pose of the robot)
398 #if 0
399  const CPose3D globalPose(1,2,3,10.0_deg,20.0_deg,30.0_deg);
400  win3D.get3DSceneAndLock();
401  logger.enter("RGBD->3D.projectInto");
402  newObs->unprojectInto(*gl_points, false /* without obs.sensorPose */, &globalPose);
403  logger.leave("RGBD->3D.projectInto");
404  win3D.unlockAccess3DScene();
405 #endif
406 
407 // Pathway: RGB+D --> internal local XYZ pointcloud --> XYZ+RGB point cloud map
408 // --> XYZ+RGB opengl
409 #if 0
410  // Project 3D points:
411  if (!newObs->hasPoints3D)
412  {
413  logger.enter("RGBD->3D.projectInto");
414  newObs->unprojectInto();
415  logger.leave("RGBD->3D.projectInto");
416  }
417 
418  CColouredPointsMap pntsMap;
419  pntsMap.colorScheme.scheme = CColouredPointsMap::cmFromIntensityImage;
420  pntsMap.loadFromRangeScan(*newObs);
421 
422  win3D.get3DSceneAndLock();
423  logger.enter("RGBD->3D.load in OpenGL");
424  gl_points->loadFromPointsMap(&pntsMap);
425  logger.leave("RGBD->3D.load in OpenGL");
426  win3D.unlockAccess3DScene();
427 #endif
428 
429  logger.leave("RGBD->3D");
430  }
431 
432  win3D.repaint();
433  } // end update visualization:
434 
435  // Process possible keyboard commands:
436  // --------------------------------------
437  if (win3D.keyHit())
438  {
439  const int key = tolower(win3D.getPushedKey());
440 
441  switch (key)
442  {
443  // Some of the keys are processed in this thread:
444  case 'o':
445  win3D.setCameraZoom(win3D.getCameraZoom() * 1.2);
446  win3D.repaint();
447  break;
448  case 'i':
449  win3D.setCameraZoom(win3D.getCameraZoom() / 1.2);
450  win3D.repaint();
451  break;
452  case 27: // ESC
453  thrPar.quit = true;
454  break;
455  default:
456  break;
457  };
458  }
459 
460  std::this_thread::sleep_for(1ms);
461  }
462 
463  cout << "Waiting for grabbing thread to exit...\n";
464  thrPar.quit = true;
465  thHandle.join();
466  cout << "Bye!\n";
467 }
468 
469 int main(int argc, char** argv)
470 {
471  try
472  {
473  // Determine online/offline operation:
474  if (argc != 1 && argc != 2)
475  {
476  cerr << "Usage:\n"
477  << argv[0] << " --> Online grab from sensor\n"
478  << argv[0] << " [DATASET.rawlog] --> Offline from dataset\n";
479  return 1;
480  }
481 
482  if (argc == 1)
483  {
484  // Online
485  cout << "Using online operation" << endl;
487  }
488  else
489  {
490  // Offline:
491  cout << "Using offline operation with: " << argv[1] << endl;
492  Test_KinectOnlineOffline(false, string(argv[1]));
493  }
494 
495  std::this_thread::sleep_for(50ms);
496  return 0;
497  }
498  catch (const std::exception& e)
499  {
500  std::cerr << "MRPT error: " << mrpt::exception_to_str(e) << std::endl;
501  return -1;
502  }
503  catch (...)
504  {
505  printf("Another exception!!");
506  return -1;
507  }
508 }
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
A high-performance stopwatch, with typical resolution of nanoseconds.
A range or depth 3D scan measurement, as from a time-of-flight range camera or a structured-light dep...
mrpt::system::TTimeStamp now()
A shortcut for system::getCurrentTime.
Definition: datetime.h:86
Contains classes for various device interfaces.
STL namespace.
const string rawlog_file
Only when is_online==false.
CArchiveStreamBase< STREAM > archiveFrom(STREAM &s)
Helper function to create a templatized wrapper CArchive object for a: MRPT&#39;s CStream, std::istream, std::ostream, std::stringstream.
Definition: CArchive.h:592
bool open(const std::string &fileName, mrpt::optional_ref< std::string > error_msg=std::nullopt)
Opens the file for read.
volatile bool quit
In/Out variable: Forces the thread to exit or indicates an error in the thread that caused it to end...
#define ASSERT_(f)
Defines an assertion mechanism.
Definition: exceptions.h:120
mrpt::Clock::time_point TTimeStamp
A system independent time type, it holds the the number of 100-nanosecond intervals since January 1...
Definition: datetime.h:40
A class for grabing "range images", intensity images (either RGB or IR) and other information from an...
Definition: CKinect.h:265
const bool is_online
true: live grabbing from the sensor, false: from rawlog
This namespace contains representation of robot actions and observations.
void Test_KinectOnlineOffline(bool is_online, const string &rawlog_file=string())
#define IS_CLASS(obj, class_name)
True if the given reference to object (derived from mrpt::rtti::CObject) is of the given class...
Definition: CObject.h:146
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.
const char * argv[]
A versatile "profiler" that logs the time spent within each pair of calls to enter(X)-leave(X), among other stats.
const bool generate_3D_pointcloud_in_this_thread
true: populate the 3D point fields in the output observation; false: only RGB and Depth images...
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()
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
Transparently opens a compressed "gz" file and reads uncompressed data from it.
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)
double timeDifference(const mrpt::system::TTimeStamp t_first, const mrpt::system::TTimeStamp t_later)
Returns the time difference from t1 to t2 (positive if t2 is posterior to t1), in seconds...
Definition: datetime.h:123
void Tic() noexcept
Starts the stopwatch.
Definition: CTicTac.cpp:75
std::string dateTimeLocalToString(const mrpt::system::TTimeStamp t)
Convert a timestamp into this textual form (in local time): YEAR/MONTH/DAY,HH:MM:SS.MMM.
Definition: datetime.cpp:176
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 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