msgpack_parser.cpp
Go to the documentation of this file.
1 /*
2  * @brief msgpack_parser unpacks and parses msgpack data for the sick 3D lidar multiScan136.
3  *
4  * Usage example:
5  *
6  * std::ifstream msgpack_istream("polarscan_testdata_000.msg", std::ios::binary);
7  * sick_scansegment_xd::ScanSegmentParserOutput msgpack_output;
8  * sick_scansegment_xd::MsgPackParser::Parse(msgpack_istream, msgpack_output);
9  *
10  * sick_scansegment_xd::MsgPackParser::WriteCSV({ msgpack_output }, "polarscan_testdata_000.csv")
11  *
12  * for (int groupIdx = 0; groupIdx < msgpack_output.scandata.size(); groupIdx++)
13  * {
14  * for (int echoIdx = 0; echoIdx < msgpack_output.scandata[groupIdx].size(); echoIdx++)
15  * {
16  * std::vector<sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint>& scanline = msgpack_output.scandata[groupIdx][echoIdx];
17  * std::cout << (groupIdx + 1) << ". group, " << (echoIdx + 1) << ". echo: ";
18  * for (int pointIdx = 0; pointIdx < scanline.size(); pointIdx++)
19  * {
20  * sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint& point = scanline[pointIdx];
21  * std::cout << (pointIdx > 0 ? "," : "") << "(" << point.x << "," << point.y << "," << point.z << "," << point.i << ")";
22  * }
23  * std::cout << std::endl;
24  * }
25  * }
26  *
27  * Copyright (C) 2020,2021 Ing.-Buero Dr. Michael Lehning, Hildesheim
28  * Copyright (C) 2020,2021 SICK AG, Waldkirch
29  *
30  * Licensed under the Apache License, Version 2.0 (the "License");
31  * you may not use this file except in compliance with the License.
32  * You may obtain a copy of the License at
33  *
34  * http://www.apache.org/licenses/LICENSE-2.0
35  *
36  * Unless required by applicable law or agreed to in writing, software
37  * distributed under the License is distributed on an "AS IS" BASIS,
38  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
39  * See the License for the specific language governing permissions and
40  * limitations under the License.
41  *
42  * All rights reserved.
43  *
44  * Redistribution and use in source and binary forms, with or without
45  * modification, are permitted provided that the following conditions are met:
46  *
47  * * Redistributions of source code must retain the above copyright
48  * notice, this list of conditions and the following disclaimer.
49  * * Redistributions in binary form must reproduce the above copyright
50  * notice, this list of conditions and the following disclaimer in the
51  * documentation and/or other materials provided with the distribution.
52  * * Neither the name of SICK AG nor the names of its
53  * contributors may be used to endorse or promote products derived from
54  * this software without specific prior written permission
55  * * Neither the name of Ing.-Buero Dr. Michael Lehning nor the names of its
56  * contributors may be used to endorse or promote products derived from
57  * this software without specific prior written permission
58  *
59  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
60  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
61  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
62  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
63  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
64  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
65  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
66  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
67  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
68  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
69  * POSSIBILITY OF SUCH DAMAGE.
70  *
71  * Authors:
72  * Michael Lehning <michael.lehning@lehning.de>
73  *
74  * Copyright 2020,2021 SICK AG
75  * Copyright 2020,2021 Ing.-Buero Dr. Michael Lehning
76  *
77  */
78 #include <ctime>
79 #include <fstream>
80 #include <msgpack11.hpp>
81 #include "sick_scan/softwarePLL.h"
84 
86 static float normalizeAngle(float angle_rad)
87 {
88  while (angle_rad > (float)(M_PI))
89  angle_rad -= (float)(2.0 * M_PI);
90  while (angle_rad < (float)(-M_PI))
91  angle_rad += (float)(2.0 * M_PI);
92  return angle_rad;
93 }
94 
95  /*
96  * @brief Counter for each message (each scandata decoded from msgpack data)
97  */
100 
101 /*
102  * @brief Returns the tokenized integer of a msgpack key.
103  * Example: MsgpackKeyToInt("data") returns 0x11.
104  */
105 /* static int MsgpackKeyToInt(const std::string& key)
106 {
107  static std::map<std::string, int> s_msgpack_keys = {
108  {"class" , 0x10},
109  {"data" , 0x11},
110  {"numOfElems" , 0x12},
111  {"elemSz" , 0x13},
112  {"endian" , 0x14},
113  {"elemTypes" , 0x15},
114  {"little" , 0x30},
115  {"float32" , 0x31},
116  {"uint16" , 0x34},
117  {"ChannelTheta" , 0x50},
118  {"ChannelPhi" , 0x51},
119  {"DistValues" , 0x52},
120  {"RssiValues" , 0x53},
121  {"PropertiesValues" , 0x54},
122  {"Scan" , 0x70},
123  {"TimestampStart" , 0x71},
124  {"TimestampStop" , 0x72},
125  {"ThetaStart" , 0x73},
126  {"ThetaStop" , 0x74},
127  {"ScanNumber" , 0x75},
128  {"ModuleId" , 0x76},
129  {"BeamCount" , 0x77},
130  {"EchoCount" , 0x78},
131  {"ScanSegment" , 0x90},
132  {"SegmentCounter" , 0x91},
133  {"FrameNumber" , 0x92},
134  {"Availability" , 0x93},
135  {"SenderId" , 0x94},
136  {"SegmentSize" , 0x95},
137  {"SegmentData" , 0x96},
138  {"TelegramCounter" , 0xB0},
139  {"TimestampTransmit" , 0xB1}
140  };
141  return s_msgpack_keys[key];
142 } */
143 
144 /*
145  * @brief Define msgpack keys by precompiler define (faster than calling sick_scansegment_xd::MsgpackKeyToInt)
146  */
147 #define MsgpackKeyToInt_class 0x10 // sick_scansegment_xd::MsgpackKeyToInt("class")
148 #define MsgpackKeyToInt_data 0x11 // sick_scansegment_xd::MsgpackKeyToInt("data")
149 #define MsgpackKeyToInt_numOfElems 0x12 // sick_scansegment_xd::MsgpackKeyToInt("numOfElems")
150 #define MsgpackKeyToInt_elemSz 0x13 // sick_scansegment_xd::MsgpackKeyToInt("elemSz")
151 #define MsgpackKeyToInt_endian 0x14 // sick_scansegment_xd::MsgpackKeyToInt("endian")
152 #define MsgpackKeyToInt_elemTypes 0x15 // sick_scansegment_xd::MsgpackKeyToInt("elemTypes")
153 #define MsgpackKeyToInt_little 0x30 // sick_scansegment_xd::MsgpackKeyToInt("little")
154 #define MsgpackKeyToInt_float32 0x31 // sick_scansegment_xd::MsgpackKeyToInt("float32")
155 #define MsgpackKeyToInt_uint32 0x32 // sick_scansegment_xd::MsgpackKeyToInt("uint32")
156 #define MsgpackKeyToInt_uint8 0x33 // sick_scansegment_xd::MsgpackKeyToInt("uint8")
157 #define MsgpackKeyToInt_uint16 0x34 // sick_scansegment_xd::MsgpackKeyToInt("uint16")
158 #define MsgpackKeyToInt_ChannelTheta 0x50 // sick_scansegment_xd::MsgpackKeyToInt("ChannelTheta")
159 #define MsgpackKeyToInt_ChannelPhi 0x51 // sick_scansegment_xd::MsgpackKeyToInt("ChannelPhi")
160 #define MsgpackKeyToInt_DistValues 0x52 // sick_scansegment_xd::MsgpackKeyToInt("DistValues")
161 #define MsgpackKeyToInt_RssiValues 0x53 // sick_scansegment_xd::MsgpackKeyToInt("RssiValues")
162 #define MsgpackKeyToInt_PropertiesValues 0x54 // sick_scansegment_xd::MsgpackKeyToInt("PropertiesValues")
163 #define MsgpackKeyToInt_Scan 0x70 // sick_scansegment_xd::MsgpackKeyToInt("Scan")
164 #define MsgpackKeyToInt_TimestampStart 0x71 // sick_scansegment_xd::MsgpackKeyToInt("TimestampStart")
165 #define MsgpackKeyToInt_TimestampStop 0x72 // sick_scansegment_xd::MsgpackKeyToInt("TimestampStop")
166 #define MsgpackKeyToInt_ThetaStart 0x73 // sick_scansegment_xd::MsgpackKeyToInt("ThetaStart")
167 #define MsgpackKeyToInt_ThetaStop 0x74 // sick_scansegment_xd::MsgpackKeyToInt("ThetaStop")
168 #define MsgpackKeyToInt_ScanNumber 0x75 // sick_scansegment_xd::MsgpackKeyToInt("ScanNumber")
169 #define MsgpackKeyToInt_ModuleId 0x76 // sick_scansegment_xd::MsgpackKeyToInt("ModuleId")
170 #define MsgpackKeyToInt_BeamCount 0x77 // sick_scansegment_xd::MsgpackKeyToInt("BeamCount")
171 #define MsgpackKeyToInt_EchoCount 0x78 // sick_scansegment_xd::MsgpackKeyToInt("EchoCount")
172 #define MsgpackKeyToInt_ScanSegment 0x90 // sick_scansegment_xd::MsgpackKeyToInt("ScanSegment")
173 #define MsgpackKeyToInt_SegmentCounter 0x91 // sick_scansegment_xd::MsgpackKeyToInt("SegmentCounter")
174 #define MsgpackKeyToInt_FrameNumber 0x92 // sick_scansegment_xd::MsgpackKeyToInt("FrameNumber")
175 #define MsgpackKeyToInt_Availability 0x93 // sick_scansegment_xd::MsgpackKeyToInt("Availability")
176 #define MsgpackKeyToInt_SenderId 0x94 // sick_scansegment_xd::MsgpackKeyToInt("SenderId")
177 #define MsgpackKeyToInt_SegmentSize 0x95 // sick_scansegment_xd::MsgpackKeyToInt("SegmentSize")
178 #define MsgpackKeyToInt_SegmentData 0x96 // sick_scansegment_xd::MsgpackKeyToInt("SegmentData")
179 #define MsgpackKeyToInt_LayerId 0xA0 // sick_scansegment_xd::MsgpackKeyToInt("LayerId")
180 #define MsgpackKeyToInt_TelegramCounter 0xB0 // sick_scansegment_xd::MsgpackKeyToInt("TelegramCounter")
181 #define MsgpackKeyToInt_TimestampTransmit 0xB1 // sick_scansegment_xd::MsgpackKeyToInt("TimestampTransmit")
182 #define MsgpackKeyToInt_MaxValue 0xB2 // max allowed value of a msgpack key
183 
184 /*
185  * @brief static defined key values of type msgpack11::MsgPack, avoids new and delete for each call to msgpack11::MsgPack::object::find()
186  */
188 {
189 public:
191  {
192  values = std::vector<msgpack11::MsgPack>(MsgpackKeyToInt_MaxValue);
193  for (int n = 0; n < MsgpackKeyToInt_MaxValue; n++)
194  values[n] = msgpack11::MsgPack(n);
195  }
196  std::vector<msgpack11::MsgPack> values;
197 };
199 
200 /*
201  * @brief Returns the name of a tokenized msgpack key.
202  * Example: MsgpackKeyToStr(0x11) returns "data".
203  */
204 /* static std::string MsgpackKeyToStr(int key)
205 {
206  static std::map<int, std::string> s_msgpack_keys = {
207  {0x10 , "class"},
208  {0x11 , "data"},
209  {0x12 , "numOfElems"},
210  {0x13 , "elemSz"},
211  {0x14 , "endian"},
212  {0x15 , "elemTypes"},
213  {0x30 , "little"},
214  {0x31 , "float32"},
215  {0x34 , "uint16"},
216  {0x50 , "ChannelTheta"},
217  {0x51 , "ChannelPhi"},
218  {0x52 , "DistValues"},
219  {0x53 , "RssiValues"},
220  {0x54 , "PropertiesValues"},
221  {0x70 , "Scan"},
222  {0x71 , "TimestampStart"},
223  {0x72 , "TimestampStop"},
224  {0x73 , "ThetaStart"},
225  {0x74 , "ThetaStop"},
226  {0x75 , "ScanNumber"},
227  {0x76 , "ModuleId"},
228  {0x77 , "BeamCount"},
229  {0x78 , "EchoCount"},
230  {0x90 , "ScanSegment"},
231  {0x91 , "SegmentCounter"},
232  {0x92 , "FrameNumber"},
233  {0x93 , "Availability"},
234  {0x94 , "SenderId"},
235  {0x95 , "SegmentSize"},
236  {0x96 , "SegmentData"},
237  {0xB0 , "TelegramCounter"},
238  {0xB1 , "TimestampTransmit"}
239  };
240  return s_msgpack_keys[key];
241 } */
242 
243 /*
244  * Prints MsgPack raw data to string (debug and development only)
245  */
246 static std::string printMsgPack(const msgpack11::MsgPack& msg)
247 {
248  std::stringstream s;
249  if (msg.is_number())
250  s << msg.number_value();
251  if (msg.is_string())
252  s << "\"" << msg.string_value() << "\"";
253  if (msg.is_bool())
254  s << (msg.bool_value() ? "true" : "false");
255  if (!msg.array_items().empty())
256  {
257  s << "array[";
258  for (int n = 0; n < msg.array_items().size(); n++)
259  s << (n > 0 ? "," : "") << printMsgPack(msg.array_items()[n]);
260  s << "]";
261  }
262  if (!msg.binary_items().empty())
263  {
264  s << "binary[";
265  for (int n = 0; n < msg.binary_items().size(); n++)
266  s << (n > 0 ? "," : "") << printMsgPack(msg.binary_items()[n]);
267  s << "]";
268  }
269  if (!msg.object_items().empty())
270  {
271  s << "object{";
272  int n = 0;
273  for (msgpack11::MsgPack::object::const_iterator iter_item = msg.object_items().cbegin(); iter_item != msg.object_items().cend(); iter_item++, n++)
274  {
275  s << (n > 0 ? "," : "") << "\"" << printMsgPack(iter_item->first) << "\":\"" << printMsgPack(iter_item->second) << "\"";
276  }
277  s << "}";
278  }
279  return s.str();
280 }
281 
282 /*
283  * @brief class MsgPackElement is a quadruple of MsgPack
284  * for data, size, type and endianess of a MsgPack element.
285  * Note: For performance reasons, the pointer to the objects
286  * are stored. The caller has to insure objects are allocated
287  * properly while accessing MsgPackElement members.
288  */
290 {
291 public:
294  {
295  data = &object_items.find(s_msgpack_keys.values[MsgpackKeyToInt_data])->second;
296  elemSz = &object_items.find(s_msgpack_keys.values[MsgpackKeyToInt_elemSz])->second;
297  endian = &object_items.find(s_msgpack_keys.values[MsgpackKeyToInt_endian])->second;
298  elemTypes = &object_items.find(s_msgpack_keys.values[MsgpackKeyToInt_elemTypes])->second;
299  if (elemTypes->is_array())
301  }
306 };
307 
308 /*
309  * @brief class MsgPackToFloat32VectorConverter decodes a MsgPackElement into an array of float data.
310  */
312 {
313 public:
315  MsgPackToFloat32VectorConverter(const MsgPackElement& msgpack, bool dstIsBigEndian)
316  {
317  union FLOAT_4BYTE_UNION
318  {
319  uint8_t u8_bytes[4];
320  uint32_t u32_bytes;
321  float value;
322  };
323 
324  union UINT_2BYTE_UNION
325  {
326  uint8_t u8_bytes[2];
327  uint16_t u16_bytes;
328  };
329 
330  assert(msgpack.data && msgpack.elemSz && msgpack.elemTypes && msgpack.endian
331  && msgpack.elemSz->is_number()
332  && msgpack.data->binary_items().size() > 0
333  && ((msgpack.data->binary_items().size()) % (msgpack.elemSz->int_value())) == 0);
334  assert((msgpack.elemSz->int_value() == 4 && msgpack.elemTypes->int_value() == MsgpackKeyToInt_float32)
335  || (msgpack.elemSz->int_value() == 2 && msgpack.elemTypes->int_value() == MsgpackKeyToInt_uint16));
336 
337  bool srcIsBigEndian = (msgpack.endian->string_value() == "big");
338  const msgpack11::MsgPack::binary& binary_items = msgpack.data->binary_items();
339  int elem_size = msgpack.elemSz->int_value();
340  int binary_size = (int)(binary_items.size());
341  m_data.reserve(binary_size / elem_size);
342  if (msgpack.elemSz->int_value() == 4 && msgpack.elemTypes->int_value() == MsgpackKeyToInt_float32) // Decode 4 bytes as float
343  {
344  FLOAT_4BYTE_UNION elem_buffer;
345  if (srcIsBigEndian == dstIsBigEndian) // src and dst have identical endianess: reinterprete 4 bytes as float
346  {
347  for (int n = 0; n < binary_size; n += 4)
348  {
349  elem_buffer.u32_bytes = *((uint32_t*)(&binary_items[n]));
350  m_data.push_back(elem_buffer.value);
351  }
352  }
353  else // src and dst have different endianess: reorder 4 bytes and interprete as float
354  {
355  for (int n = 0; n < binary_size; n += 4)
356  {
357  elem_buffer.u8_bytes[3] = binary_items[n + 0];
358  elem_buffer.u8_bytes[2] = binary_items[n + 1];
359  elem_buffer.u8_bytes[1] = binary_items[n + 2];
360  elem_buffer.u8_bytes[0] = binary_items[n + 3];
361  m_data.push_back(elem_buffer.value);
362  }
363  }
364  }
365  else if (msgpack.elemSz->int_value() == 2 && msgpack.elemTypes->int_value() == MsgpackKeyToInt_uint16) // Decode 2 bytes as uint16 and convert to float
366  {
367  UINT_2BYTE_UNION elem_buffer;
368  if (srcIsBigEndian == dstIsBigEndian) // src and dst have identical endianess: reinterprete 2 bytes as uint16 and convert to float
369  {
370  for (int n = 0; n < binary_size; n += 2)
371  {
372  elem_buffer.u16_bytes = *((uint16_t*)(&binary_items[n]));
373  m_data.push_back((float)elem_buffer.u16_bytes);
374  }
375  }
376  else // src and dst have different endianess: reorder 2 bytes (uint16) and convert to float
377  {
378  for (int n = 0; n < binary_size; n += 2)
379  {
380  elem_buffer.u8_bytes[1] = binary_items[n + 0];
381  elem_buffer.u8_bytes[0] = binary_items[n + 1];
382  m_data.push_back((float)elem_buffer.u16_bytes);
383  }
384  }
385  }
386  else
387  {
388  std::cerr << "## ERROR MsgPackToFloat32VectorConverter: invalid or unsupported elemSz or elemTypes:" << std::endl
389  << " msgpack.data = " << (msgpack.data ? printMsgPack(*msgpack.data) : "NULL") << std::endl
390  << " msgpack.elemSz = " << (msgpack.elemSz ? printMsgPack(*msgpack.elemSz) : "NULL") << std::endl
391  << " msgpack.elemTypes = " << (msgpack.elemTypes ? printMsgPack(*msgpack.elemTypes) : "NULL") << std::endl
392  << " msgpack.endian = " << (msgpack.endian ? printMsgPack(*msgpack.endian) : "NULL") << std::endl;
393  }
394  }
395  std::string print(void)
396  {
397  std::stringstream s;
398  if (!m_data.empty())
399  {
400  s << m_data[0];
401  for(int n = 1; n < m_data.size(); n++)
402  s << "," << m_data[n];
403  }
404  return s.str();
405  }
406  float rad2deg(float angle) const { return angle * (float)(180.0 / M_PI); }
407  std::string printRad2Deg(void)
408  {
409  std::stringstream s;
410  if (!m_data.empty())
411  {
412  s << rad2deg(m_data[0]);
413  for(int n = 1; n < m_data.size(); n++)
414  s << "," << rad2deg(m_data[n]);
415  }
416  return s.str();
417  }
418  std::vector<float>& data(void)
419  {
420  return m_data;
421  }
422 protected:
423  std::vector<float> m_data;
424 };
425 
426 /*
427  * @brief reads a file in binary mode and returns all bytes.
428  * @param[in] filepath input file incl. path
429  * @param[out] list of bytes
430  */
431 std::vector<uint8_t> sick_scansegment_xd::MsgPackParser::ReadFile(const std::string& filepath)
432 {
433  std::ifstream file_istream(filepath, std::ios::binary);
434  if (file_istream.is_open())
435  return std::vector<uint8_t>((std::istreambuf_iterator<char>(file_istream)), std::istreambuf_iterator<char>());
436  return std::vector<uint8_t>();
437 }
438 
439 /*
440  * @brief Returns a hexdump of a msgpack. To get a well formatted json struct from a msgpack,
441  * just paste the returned string to https://toolslick.com/conversion/data/messagepack-to-json
442  */
443 std::string sick_scansegment_xd::MsgPackParser::MsgpackToHexDump(const std::vector<uint8_t>& msgpack_data, bool pretty_print)
444 {
445  std::stringstream msgpack_hexdump;
446  for (size_t n = 0; n < msgpack_data.size(); n++)
447  {
448  msgpack_hexdump << std::setfill('0') << std::setw(2) << std::hex << (int)(msgpack_data[n] & 0xFF);
449  if(pretty_print)
450  {
451  if ((n % 20) == 19)
452  msgpack_hexdump << std::endl;
453  else
454  msgpack_hexdump << " ";
455  }
456  }
457  return msgpack_hexdump.str();
458 }
459 
460 /*
461  * @brief unpacks and parses msgpack data from a binary input stream.
462  *
463  * Usage example:
464  *
465  * std::vector<uint8_t> msgpack_data = sick_scansegment_xd::MsgPackParser::ReadFile("polarscan_testdata_000.msg");
466  * sick_scansegment_xd::ScanSegmentParserOutput msgpack_output;
467  * sick_scansegment_xd::MsgPackParser::Parse(msgpack_data, msgpack_output);
468  * sick_scansegment_xd::MsgPackParser::WriteCSV({ msgpack_output }, "polarscan_testdata_000.csv")
469  *
470  * @param[in+out] msgpack_ifstream the binary input stream delivering the binary msgpack data
471  * @param[in] msgpack_timestamp receive timestamp of msgpack_data
472  * @param[in] add_transform_xyz_rpy Apply an additional transform to the cartesian pointcloud, default: "0,0,0,0,0,0" (i.e. no transform)
473  * @param[out] result msgpack data converted to scanlines of type ScanSegmentParserOutput
474  * @param[in+out] msgpack_validator_data_collector collects MsgPackValidatorData over N msgpacks
475  * @param[in] msgpack_validator msgpack validation, see MsgPackValidator for details
476  * @param[in] msgpack_validator_enabled true: check msgpack data for out of bounds and missing scan data, false: no msgpack validation
477  * @param[in] discard_msgpacks_not_validated true: msgpacks are discarded if not validated, false: error message if a msgpack is not validated
478  * @param[in] use_software_pll true (default): result timestamp from sensor ticks by software pll, false: result timestamp from msg receiving
479  * @param[in] verbose true: enable debug output, false: quiet mode
480  */
481 bool sick_scansegment_xd::MsgPackParser::Parse(const std::vector<uint8_t>& msgpack_data, fifo_timestamp msgpack_timestamp,
482  sick_scan_xd::SickCloudTransform& add_transform_xyz_rpy, ScanSegmentParserOutput& result,
483  sick_scansegment_xd::MsgPackValidatorData& msgpack_validator_data_collector, const sick_scansegment_xd::MsgPackValidator& msgpack_validator,
484  bool msgpack_validator_enabled, bool discard_msgpacks_not_validated,
485  bool use_software_pll, bool verbose)
486 {
487  // To debug, print and visual msgpack_data, just paste hex dump to
488  // https://toolslick.com/conversion/data/messagepack-to-json
489  // to get a well formatted json struct.
490  // std::string msgpack_hexdump = MsgpackToHexDump(msgpack_data, false);
491  // std::ofstream msgpack_dumpfile("msgpack-debug.msgpack.hex");
492  // if(msgpack_dumpfile.is_open())
493  // msgpack_dumpfile << msgpack_hexdump;
494  // std::string msgpack_hexdump = MsgpackToHexDump(msgpack_data, true);
495  // std::cout << std::endl << "MsgPack hexdump: " << std::endl << msgpack_hexdump << std::endl << std::endl;
496  std::string msgpack_string((char*)msgpack_data.data(), msgpack_data.size());
497  std::istringstream msgpack_istream(msgpack_string);
498  return Parse(msgpack_istream, msgpack_timestamp, add_transform_xyz_rpy, result, msgpack_validator_data_collector, msgpack_validator, msgpack_validator_enabled, discard_msgpacks_not_validated, use_software_pll, verbose);
499 }
500 
501 /*
502  * @brief unpacks and parses msgpack data from a binary input stream.
503  *
504  * Usage example:
505  *
506  * std::ifstream msgpack_istream("polarscan_testdata_000.msg", std::ios::binary);
507  * sick_scansegment_xd::ScanSegmentParserOutput msgpack_output;
508  * sick_scansegment_xd::MsgPackParser::Parse(msgpack_istream, msgpack_output);
509  *
510  * sick_scansegment_xd::MsgPackParser::WriteCSV({ msgpack_output }, "polarscan_testdata_000.csv")
511  *
512  * for (int groupIdx = 0; groupIdx < msgpack_output.scandata.size(); groupIdx++)
513  * {
514  * for (int echoIdx = 0; echoIdx < msgpack_output.scandata[groupIdx].size(); echoIdx++)
515  * {
516  * std::vector<sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint>& scanline = msgpack_output.scandata[groupIdx][echoIdx];
517  * std::cout << (groupIdx + 1) << ". group, " << (echoIdx + 1) << ". echo: ";
518  * for (int pointIdx = 0; pointIdx < scanline.size(); pointIdx++)
519  * {
520  * sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint& point = scanline[pointIdx];
521  * std::cout << (pointIdx > 0 ? "," : "") << "(" << point.x << "," << point.y << "," << point.z << "," << point.i << ")";
522  * }
523  * std::cout << std::endl;
524  * }
525  * }
526  *
527  * @param[in+out] msgpack_ifstream the binary input stream delivering the binary msgpack data
528  * @param[in] msgpack_timestamp receive timestamp of msgpack_data
529  * @param[in] add_transform_xyz_rpy Apply an additional transform to the cartesian pointcloud, default: "0,0,0,0,0,0" (i.e. no transform)
530  * @param[out] result msgpack data converted to scanlines of type ScanSegmentParserOutput
531  * @param[in] discard_msgpacks_not_validated true: msgpacks are discarded if not validated, false: error message if a msgpack is not validated
532  * @param[in] msgpack_validator msgpack validation, see MsgPackValidator for details
533  * @param[in] msgpack_validator_enabled true: check msgpack data for out of bounds and missing scan data, false: no msgpack validation
534  * @param[in+out] msgpack_validator_data_collector collects MsgPackValidatorData over N msgpacks
535  * @param[in] use_software_pll true (default): result timestamp from sensor ticks by software pll, false: result timestamp from msg receiving
536  * @param[in] verbose true: enable debug output, false: quiet mode
537  */
538 bool sick_scansegment_xd::MsgPackParser::Parse(std::istream& msgpack_istream, fifo_timestamp msgpack_timestamp,
539  sick_scan_xd::SickCloudTransform& add_transform_xyz_rpy, ScanSegmentParserOutput& result,
540  sick_scansegment_xd::MsgPackValidatorData& msgpack_validator_data_collector,
541  const sick_scansegment_xd::MsgPackValidator& msgpack_validator,
542  bool msgpack_validator_enabled, bool discard_msgpacks_not_validated,
543  bool use_software_pll, bool verbose)
544 {
545  int64_t systemtime_nanoseconds = msgpack_timestamp.time_since_epoch().count();
546  uint32_t systemtime_sec = (uint32_t)(systemtime_nanoseconds / 1000000000); // seconds part of timestamp
547  uint32_t systemtime_nsec = (uint32_t)(systemtime_nanoseconds % 1000000000); // nanoseconds part of timestamp
548  result.timestamp = sick_scansegment_xd::Timestamp(systemtime_sec, systemtime_nsec); // Timestamp(std::chrono::system_clock::now()); // default timestamp: msgpack receive time, overwritten by timestamp from msgpack data
549  result.timestamp_sec = systemtime_sec;
550  result.timestamp_nsec = systemtime_nsec;
551  int32_t segment_idx = messageCount++; // default value: counter for each message (each scandata decoded from msgpack data), overwritten by msgpack data
552  int32_t telegram_cnt = telegramCount++; // default value: counter for each message (each scandata decoded from msgpack data), overwritten by msgpack data
553  msgpack11::MsgPack msg_unpacked;
554  try
555  {
556  // Unpack the binary msgpack data
557  std::string msg_parse_error;
558  msg_unpacked = msgpack11::MsgPack::parse(msgpack_istream, msg_parse_error);
559  if (!msg_parse_error.empty())
560  {
561  ROS_ERROR_STREAM("## ERROR msgpack11::MsgPack::parse(): " << msg_parse_error);
562  return false;
563  }
564  // std::cout << printMsgPack(msg_unpacked) << std::endl << std::endl;
565  }
566  catch(const std::exception & exc)
567  {
568  ROS_ERROR_STREAM("## ERROR msgpack11::MsgPack::parse(): exception " << exc.what());
569  return false;
570  }
571 
572  // Get endianess of the system (destination target)
573  bool dstIsBigEndian = sick_scansegment_xd::Config::SystemIsBigEndian();
574 
575  // Parse the unpacked msgpack data, see sick_scansegment_xd/python/polarscan_reader_test/polarscan_receiver_test.py for multiScan136 message format
576  // and https://github.com/SICKAG/msgpack11/blob/master/msgpack11.hpp or https://github.com/SICKAG/msgpack11/blob/master/example.cpp
577  // for details about decoding and paring MsgPack data.
578  try
579  {
580  sick_scansegment_xd::MsgPackValidatorData msgpack_validator_data;
581  // std::cout << "root_object_items: " << printMsgPack(msg_unpacked.object_items()) << std::endl;
582  msgpack11::MsgPack::object::const_iterator root_data_iter = msg_unpacked.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_data]);
583  if (root_data_iter == msg_unpacked.object_items().end())
584  {
585  ROS_WARN_STREAM("## ERROR MsgPackParser::Parse(): \"data\" not found");
586  return false;
587  }
588  const msgpack11::MsgPack& root_data = root_data_iter->second;
589  msgpack11::MsgPack::object::const_iterator group_data_iter = root_data.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_SegmentData]);
590  msgpack11::MsgPack::object::const_iterator timestamp_data_iter = root_data.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_TimestampTransmit]);
591  if (group_data_iter == root_data.object_items().end() || timestamp_data_iter == root_data.object_items().end())
592  {
593  ROS_WARN_STREAM("## ERROR MsgPackParser::Parse(): \"SegmentData\" and/or \"TimestampTransmit\" not found");
594  return false;
595  }
596  const msgpack11::MsgPack& group_data = group_data_iter->second;
597  const msgpack11::MsgPack& timestamp_data = timestamp_data_iter->second;
598  if (use_software_pll && timestamp_data.is_number())
599  {
600  // result.timestamp = std::to_string(timestamp_data.int64_value());
601  // Calculate system time from sensor ticks using SoftwarePLL
602  // result.timestamp = std::to_string(timestamp_data.int64_value());
603  SoftwarePLL& software_pll = SoftwarePLL::instance();
604  uint32_t curtick = timestamp_data.int32_value();
605  software_pll.updatePLL(systemtime_sec, systemtime_nsec, curtick);
606  if (software_pll.IsInitialized())
607  {
608  uint32_t pll_sec = 0, pll_nsec = 0;
609  software_pll.getCorrectedTimeStamp(pll_sec, pll_nsec, curtick);
610  result.timestamp = sick_scansegment_xd::Timestamp(pll_sec, pll_nsec);
611  result.timestamp_sec = pll_sec;
612  result.timestamp_nsec = pll_nsec;
613  if (verbose)
614  ROS_INFO_STREAM("MsgPackParser::Parse(): sensor_ticks=" << curtick << ", system_time=" << sick_scansegment_xd::Timestamp(systemtime_sec, systemtime_nsec) << " sec, timestamp=" << result.timestamp << " sec");
615  }
616  else if (verbose)
617  ROS_INFO_STREAM("MsgPackParser::Parse(): sensor_ticks=" << curtick << ", system_time=" << sick_scansegment_xd::Timestamp(systemtime_sec, systemtime_nsec) << " sec, SoftwarePLL not yet initialized");
618  }
619  msgpack11::MsgPack::object::const_iterator segment_counter_iter = root_data.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_SegmentCounter]);
620  if (segment_counter_iter == root_data.object_items().end())
621  {
622  ROS_WARN_STREAM("## ERROR MsgPackParser::Parse(): \"SegmentCounter\" not found");
623  return false;
624  }
625  const msgpack11::MsgPack& segment_idx_data = segment_counter_iter->second;
626  segment_idx = segment_idx_data.int32_value();
627 
628  msgpack11::MsgPack::object::const_iterator telegram_counter_iter = root_data.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_TelegramCounter]);
629  if (telegram_counter_iter != root_data.object_items().end())
630  {
631  const msgpack11::MsgPack& telegram_cnt_data = telegram_counter_iter->second;
632  telegramCount = telegram_cnt_data.int32_value();
633  telegram_cnt = telegramCount;
634  }
635 
636  // std::cout << "root_data: " << printMsgPack(root_data) << std::endl << "root_data.array_items().size(): " << root_data.array_items().size() << ", root_data.object_items().size(): " << root_data.object_items().size() << std::endl;
637  // std::cout << "group_data.array_items().size(): " << group_data.array_items().size() << ", group_data.object_items().size(): " << group_data.object_items().size() << std::endl;
638  result.scandata.reserve(group_data.array_items().size());
639  for (int groupIdx = 0; groupIdx < group_data.array_items().size(); groupIdx++)
640  {
641  // Get ChannelPhi, ChannelTheta, DistValues and RssiValues for each group
642  const msgpack11::MsgPack& groupMsg = group_data.array_items()[groupIdx];
643  // std::cout << "groupMsg: " << printMsgPack(groupMsg) << std::endl << "groupMsg.array_items().size(): " << groupMsg.array_items().size() << ", groupMsg.object_items().size(): " << groupMsg.object_items().size() << std::endl;
644  msgpack11::MsgPack::object::const_iterator dataMsg = groupMsg.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_data]);
645  if (dataMsg == groupMsg.object_items().end())
646  {
647  continue; // ok if missing
648  }
649  msgpack11::MsgPack::object::const_iterator echoCountMsg = dataMsg->second.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_EchoCount]);
650  msgpack11::MsgPack::object::const_iterator channelPhiMsg = dataMsg->second.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_ChannelPhi]);
651  msgpack11::MsgPack::object::const_iterator channelThetaMsg = dataMsg->second.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_ChannelTheta]);
652  msgpack11::MsgPack::object::const_iterator distValuesMsg = dataMsg->second.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_DistValues]);
653  msgpack11::MsgPack::object::const_iterator rssiValuesMsg = dataMsg->second.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_RssiValues]);
654  msgpack11::MsgPack::object::const_iterator timestampStartMsg = dataMsg->second.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_TimestampStart]);
655  msgpack11::MsgPack::object::const_iterator timestampStopMsg = dataMsg->second.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_TimestampStop]);
656  msgpack11::MsgPack::object::const_iterator propertiesMsg = dataMsg->second.object_items().find(s_msgpack_keys.values[MsgpackKeyToInt_PropertiesValues]);
657  if (echoCountMsg == dataMsg->second.object_items().end() ||
658  channelPhiMsg == dataMsg->second.object_items().end() || channelThetaMsg == dataMsg->second.object_items().end() ||
659  distValuesMsg == dataMsg->second.object_items().end() || rssiValuesMsg == dataMsg->second.object_items().end() ||
660  timestampStartMsg == dataMsg->second.object_items().end() || timestampStopMsg == dataMsg->second.object_items().end())
661  {
662  ROS_WARN_STREAM("## ERROR MsgPackParser::Parse(): Entries in data segment missing");
663  continue;
664  }
665  uint32_t u32TimestampStart = timestampStartMsg->second.uint32_value();
666  uint32_t u32TimestampStop = timestampStopMsg->second.uint32_value();
667  uint32_t u32TimestampStart_sec = 0, u32TimestampStart_nsec = 0;
668  uint32_t u32TimestampStop_sec = 0, u32TimestampStop_nsec = 0;
669  if (use_software_pll && SoftwarePLL::instance().IsInitialized())
670  {
671  SoftwarePLL& software_pll = SoftwarePLL::instance();
672  software_pll.getCorrectedTimeStamp(u32TimestampStart_sec, u32TimestampStart_nsec, u32TimestampStart);
673  software_pll.getCorrectedTimeStamp(u32TimestampStop_sec, u32TimestampStop_nsec, u32TimestampStop);
674  }
675 
676  // Get data, elemSz, elemTypes and endian for each MsgPack object
677  MsgPackElement channelPhiMsgElement(channelPhiMsg->second.object_items());
678  MsgPackElement channelThetaMsgElement(channelThetaMsg->second.object_items());
679  std::vector<MsgPackElement> distValuesDataMsg(distValuesMsg->second.array_items().size());
680  std::vector<MsgPackElement> rssiValuesDataMsg(rssiValuesMsg->second.array_items().size());
681  for (int n = 0; n < distValuesMsg->second.array_items().size(); n++)
682  distValuesDataMsg[n] = MsgPackElement(distValuesMsg->second.array_items()[n].object_items());
683  for (int n = 0; n < rssiValuesMsg->second.array_items().size(); n++)
684  rssiValuesDataMsg[n] = MsgPackElement(rssiValuesMsg->second.array_items()[n].object_items());
685 
686  // Get optional property values
687  std::vector<std::vector<uint8_t>> propertyValues; // uint8_t property = propertyValues[echoIdx][pointIdx]
688  if (propertiesMsg != dataMsg->second.object_items().end()) // property values available
689  {
690  propertyValues = std::vector<std::vector<uint8_t>>(propertiesMsg->second.array_items().size());
691  for (int n = 0; n < propertiesMsg->second.array_items().size(); n++)
692  {
693  const MsgPackElement& propertyMsgPackElement = MsgPackElement(propertiesMsg->second.array_items()[n].object_items());
694  propertyValues[n] = std::vector<uint8_t>(propertyMsgPackElement.data->binary_items().size(), 0);
695  if (propertyMsgPackElement.elemSz->int_value() == 1 && propertyMsgPackElement.elemTypes->int_value() == MsgpackKeyToInt_uint8 && propertyMsgPackElement.data->binary_items().size() > 0)
696  {
697  for(int m = 0; m < propertyValues[n].size(); m++)
698  {
699  propertyValues[n][m] = propertyMsgPackElement.data->binary_items()[m];
700  }
701  }
702  else
703  {
704  ROS_WARN_STREAM("## ERROR MsgPackParser::Parse(): invalid property array");
705  }
706  }
707  }
708 
709  // Convert all data to float values
710  int iEchoCount = echoCountMsg->second.int32_value();
711  MsgPackToFloat32VectorConverter channelPhi(channelPhiMsgElement, dstIsBigEndian);
712  MsgPackToFloat32VectorConverter channelTheta(channelThetaMsgElement, dstIsBigEndian);
713  std::vector<MsgPackToFloat32VectorConverter> distValues(distValuesDataMsg.size());
714  std::vector<MsgPackToFloat32VectorConverter> rssiValues(rssiValuesDataMsg.size());
715  for (int n = 0; n < distValuesDataMsg.size(); n++)
716  distValues[n] = MsgPackToFloat32VectorConverter(distValuesDataMsg[n], dstIsBigEndian);
717  for (int n = 0; n < rssiValuesDataMsg.size(); n++)
718  rssiValues[n] = MsgPackToFloat32VectorConverter(rssiValuesDataMsg[n], dstIsBigEndian);
719  assert(channelPhi.data().size() == 1 && channelTheta.data().size() > 0 && distValues.size() == iEchoCount && rssiValues.size() == iEchoCount);
720 
721  // Check optional propertyValues: if available, we expect as many properties as we have points
722  for (int n = 0; n < propertyValues.size(); n++)
723  {
724  // ROS_DEBUG_STREAM("MsgPackParser::Parse(): " << (distValues[n].data().size()) << " dist values, " << (rssiValues[n].data().size()) << " rssi values, " << (propertyValues[n].size()) << " property values (" << (n+1) << ". echo)");
725  if (propertyValues[n].size() != distValues[n].data().size())
726  ROS_WARN_STREAM("## ERROR MsgPackParser::Parse(): invalid property values");
727  }
728 
729  // Convert to cartesian coordinates
731  result.scandata.back().timestampStart_sec = u32TimestampStart_sec;
732  result.scandata.back().timestampStart_nsec = u32TimestampStart_nsec;
733  result.scandata.back().timestampStop_sec = u32TimestampStop_sec;
734  result.scandata.back().timestampStop_nsec = u32TimestampStop_nsec;
735  iEchoCount = std::min((int)distValuesDataMsg.size(), iEchoCount);
736  iEchoCount = std::min((int)rssiValuesDataMsg.size(), iEchoCount);
737  std::vector<sick_scansegment_xd::ScanSegmentParserOutput::Scanline>& groupData = result.scandata.back().scanlines;
738  groupData.reserve(iEchoCount);
739  int iPointCount = (int)channelTheta.data().size();
740  // Precompute sin and cos values of azimuth and elevation
741  float elevation = -channelPhi.data()[0]; // elevation must be negated, a positive pitch-angle yields negative z-coordinates
742  float cos_elevation = std::cos(elevation);
743  float sin_elevation = std::sin(elevation);
744  std::vector<float> cos_azimuth(iPointCount);
745  std::vector<float> sin_azimuth(iPointCount);
746  std::vector<uint64_t> lut_lidar_timestamp_microsec(iPointCount);
747  for (int pointIdx = 0; pointIdx < iPointCount; pointIdx++)
748  {
749  float azimuth = channelTheta.data()[pointIdx] + add_transform_xyz_rpy.azimuthOffset();
750  cos_azimuth[pointIdx] = std::cos(azimuth);
751  sin_azimuth[pointIdx] = std::sin(azimuth);
752  lut_lidar_timestamp_microsec[pointIdx] = ((pointIdx * (u32TimestampStop - u32TimestampStart)) / (iPointCount - 1)) + u32TimestampStart;
753  // if (pointIdx > 0)
754  // SCANSEGMENT_XD_DEBUG_STREAM("azimuth[" << pointIdx << "] = " << (azimuth * 180 / M_PI) << " [deg], delta_azimuth = " << ((channelTheta.data[pointIdx] - channelTheta.data[pointIdx-1]) * 180 / M_PI) << " [deg]");
755  }
756  for (int echoIdx = 0; echoIdx < iEchoCount; echoIdx++)
757  {
758  assert(iPointCount == channelTheta.data().size() && iPointCount == distValues[echoIdx].data().size() && iPointCount == rssiValues[echoIdx].data().size());
760  sick_scansegment_xd::ScanSegmentParserOutput::Scanline& scanline = groupData.back();
761  scanline.points.reserve(iPointCount);
762  for (int pointIdx = 0; pointIdx < iPointCount; pointIdx++)
763  {
764  uint8_t reflectorbit = 0;
765  for (int n = 0; n < propertyValues.size(); n++)
766  if (pointIdx < propertyValues[n].size())
767  reflectorbit |= ((propertyValues[n][pointIdx]) & 0x01); // reflector bit is set, if a reflector is detected on any number of echos
768  float dist = 0.001f * distValues[echoIdx].data()[pointIdx]; // convert distance to meter
769  float intensity = rssiValues[echoIdx].data()[pointIdx];
770  float x = dist * cos_azimuth[pointIdx] * cos_elevation;
771  float y = dist * sin_azimuth[pointIdx] * cos_elevation;
772  float z = dist * sin_elevation;
773  add_transform_xyz_rpy.applyTransform(x, y, z);
774  float azimuth = channelTheta.data()[pointIdx];
775  float azimuth_norm = normalizeAngle(azimuth);
776  if (msgpack_validator_enabled)
777  {
778  msgpack_validator_data.update(echoIdx, segment_idx, azimuth_norm, elevation);
779  msgpack_validator_data_collector.update(echoIdx, segment_idx, azimuth_norm, elevation);
780  }
781  uint64_t lidar_timestamp_microsec = lut_lidar_timestamp_microsec[pointIdx];
782  scanline.points.push_back(sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint(x, y, z, intensity, dist, azimuth, elevation, groupIdx, echoIdx, pointIdx, lidar_timestamp_microsec, reflectorbit));
783  }
784  }
785 
786  // debug output
787  if (verbose)
788  {
789  ROS_INFO_STREAM((groupIdx + 1) << ". group: EchoCount = " << iEchoCount);
790  ROS_INFO_STREAM((groupIdx + 1) << ". group: phi (elevation, rad) = [" << channelPhi.print() << "], " << channelPhi.data().size() << " element");
791  ROS_INFO_STREAM((groupIdx + 1) << ". group: phi (elevation, deg) = [" << channelPhi.printRad2Deg() << "], " << channelPhi.data().size() << " element");
792  ROS_INFO_STREAM((groupIdx + 1) << ". group: theta (azimuth, rad) = [" << channelTheta.print() << "], " << channelTheta.data().size() << " elements");
793  ROS_INFO_STREAM((groupIdx + 1) << ". group: theta (azimuth, deg) = [" << channelTheta.printRad2Deg() << "], " << channelTheta.data().size() << " elements");
794  ROS_INFO_STREAM((groupIdx + 1) << ". group: timestampStart = " << u32TimestampStart << " = " << sick_scansegment_xd::Timestamp(u32TimestampStart_sec, u32TimestampStart_nsec));
795  ROS_INFO_STREAM((groupIdx + 1) << ". group: timestampStop = " << u32TimestampStop << " = " << sick_scansegment_xd::Timestamp(u32TimestampStop_sec, u32TimestampStop_nsec));
796  for (int n = 0; n < distValues.size(); n++)
797  ROS_INFO_STREAM((groupIdx + 1) << ". group: dist[" << n << "] = [" << distValues[n].print() << "], " << distValues[n].data().size() << " elements");
798  for (int n = 0; n < rssiValues.size(); n++)
799  ROS_INFO_STREAM((groupIdx + 1) << ". group: rssi[" << n << "] = [" << rssiValues[n].print() << "], " << rssiValues[n].data().size() << " elements");
800  // std::cout << (groupIdx + 1) << ". group: ChannelPhiMsg.data = " << printMsgPack(*channelPhiMsgElement.data) << std::endl;
801  // std::cout << (groupIdx + 1) << ". group: ChannelPhiMsg.elemSz = " << printMsgPack(*channelPhiMsgElement.elemSz) << std::endl;
802  // std::cout << (groupIdx + 1) << ". group: ChannelPhiMsg.elemType = " << printMsgPack(*channelPhiMsgElement.elemTypes) << std::endl;
803  // std::cout << (groupIdx + 1) << ". group: ChannelPhiMsg.endian = " << printMsgPack(*channelPhiMsgElement.endian) << std::endl;
804  // std::cout << (groupIdx + 1) << ". group: ChannelThetaMsg.data = " << printMsgPack(*channelThetaMsgElement.data) << std::endl;
805  // std::cout << (groupIdx + 1) << ". group: ChannelThetaMsg.elemSz = " << printMsgPack(*channelThetaMsgElement.elemSz) << std::endl;
806  // std::cout << (groupIdx + 1) << ". group: ChannelThetaMsg.elemType = " << printMsgPack(*channelThetaMsgElement.elemTypes) << std::endl;
807  // std::cout << (groupIdx + 1) << ". group: ChannelThetaMsg.endian = " << printMsgPack(*channelThetaMsgElement.endian) << std::endl;
808  // for (int n = 0; n < distValuesDataMsg.size(); n++)
809  // {
810  // std::cout << (groupIdx + 1) << ". group: DistValuesDataMsg[" << n << "].data = " << printMsgPack(*distValuesDataMsg[n].data) << std::endl;
811  // std::cout << (groupIdx + 1) << ". group: DistValuesDataMsg[" << n << "].elemSz = " << printMsgPack(*distValuesDataMsg[n].elemSz) << std::endl;
812  // std::cout << (groupIdx + 1) << ". group: DistValuesDataMsg[" << n << "].elemTypes = " << printMsgPack(*distValuesDataMsg[n].elemTypes) << std::endl;
813  // std::cout << (groupIdx + 1) << ". group: DistValuesDataMsg[" << n << "].endian = " << printMsgPack(*distValuesDataMsg[n].endian) << std::endl;
814  // }
815  // for (int n = 0; n < rssiValuesDataMsg.size(); n++)
816  // {
817  // std::cout << (groupIdx + 1) << ". group: RssiValuesDataMsg[" << n << "].data = " << printMsgPack(*rssiValuesDataMsg[n].data) << std::endl;
818  // std::cout << (groupIdx + 1) << ". group: RssiValuesDataMsg[" << n << "].elemSz = " << printMsgPack(*rssiValuesDataMsg[n].elemSz) << std::endl;
819  // std::cout << (groupIdx + 1) << ". group: RssiValuesDataMsg[" << n << "].elemTypes = " << printMsgPack(*rssiValuesDataMsg[n].elemTypes) << std::endl;
820  // std::cout << (groupIdx + 1) << ". group: RssiValuesDataMsg[" << n << "].endian = " << printMsgPack(*rssiValuesDataMsg[n].endian) << std::endl;
821  // }
822  ROS_INFO_STREAM("");
823  }
824  }
825  // msgpack validation
826  if(msgpack_validator_enabled)
827  {
828  if (verbose)
829  {
830  std::vector<std::string> messages = msgpack_validator_data.print();
831  for(int n = 0; n < messages.size(); n++)
832  ROS_INFO_STREAM(messages[n]);
833  }
834  if (msgpack_validator.validateNotOutOfBound(msgpack_validator_data) == false)
835  {
836  if (discard_msgpacks_not_validated)
837  {
838  ROS_ERROR_STREAM("## ERROR MsgPackParser::Parse(): msgpack out of bounds validation failed, pointcloud discarded");
839  return false;
840  }
841  ROS_ERROR_STREAM("## ERROR MsgPackParser::Parse(): msgpack out of bounds validation failed");
842  }
843  else if (verbose)
844  {
845  ROS_INFO_STREAM("msgpack out of bounds validation passed.");
846  }
847  }
848  }
849  catch (const std::exception& exc)
850  {
851  ROS_ERROR_STREAM("## ERROR msgpack11::MsgPack::parse(): exception " << exc.what());
852  return false;
853  }
854  result.segmentIndex = segment_idx;
855  result.telegramCnt = telegram_cnt;
856  return true;
857 }
858 
859 /*
860  * @brief exports msgpack data to csv file.
861  *
862  * Usage example:
863  *
864  * std::ifstream msgpack_istream("polarscan_testdata_000.msg", std::ios::binary);
865  * sick_scansegment_xd::ScanSegmentParserOutput msgpack_output;
866  * sick_scansegment_xd::MsgPackParser::Parse(msgpack_istream, msgpack_output);
867  *
868  * sick_scansegment_xd::MsgPackParser::WriteCSV({ msgpack_output }, "polarscan_testdata_000.csv")
869  *
870  * @param[in] result converted msgpack data, output from Parse function
871  * @param[in] csvFile name of output csv file incl. optional file path
872  * @param[in] overwrite_existing_file if overwrite_existing_file is true and csvFile already exists, the file will be overwritten. otherwise all results are appended to the file.
873  */
874 bool sick_scansegment_xd::MsgPackParser::WriteCSV(const std::vector<ScanSegmentParserOutput>& results, const std::string& csvFile, bool overwrite_existing_file)
875 {
876  if (results.empty())
877  return false;
878  bool write_header = false;
879  std::ios::openmode openmode = std::ios::app;
880  if (overwrite_existing_file || results[0].segmentIndex == 0 || !sick_scansegment_xd::FileReadable(csvFile))
881  {
882  write_header = true; // Write a csv header once for all new csv files
883  openmode = std::ios::trunc; // Create a new file in overwrite mode or at the first message, otherwise append all further messages
884  }
885  std::ofstream csv_ostream(csvFile, openmode);
886  if (!csv_ostream.is_open())
887  {
888  ROS_ERROR_STREAM("## ERRORMsgPackParser::WriteCSV(): can't open output file \"" << csvFile << "\" for writing.");
889  return false;
890  }
891  if (write_header)
892  {
893  csv_ostream << "SegmentIndex; Timestamp; GroupIdx; EchoIdx; PointIdx; X; Y; Z; Range; Azimuth; Elevation; Intensity" << std::endl;
894  }
895  for (int msgCnt = 0; msgCnt < results.size(); msgCnt++)
896  {
897  const ScanSegmentParserOutput& result = results[msgCnt];
898  for (int groupIdx = 0; groupIdx < result.scandata.size(); groupIdx++)
899  {
900  for (int echoIdx = 0; echoIdx < result.scandata[groupIdx].scanlines.size(); echoIdx++)
901  {
902  const std::vector<sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint>& scanline = result.scandata[groupIdx].scanlines[echoIdx].points;
903  for (int pointIdx = 0; pointIdx < scanline.size(); pointIdx++)
904  {
905  const sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint& point = scanline[pointIdx];
906  csv_ostream << std::setw(12) << result.segmentIndex;
907  csv_ostream << ";" << std::setw(24) << result.timestamp;
908  csv_ostream << ";" << std::setw(12) << point.groupIdx;
909  csv_ostream << ";" << std::setw(12) << point.echoIdx;
910  csv_ostream << ";" << std::setw(12) << point.pointIdx;
911  csv_ostream << ";" << std::setw(12) << std::fixed << std::setprecision(3) << point.x;
912  csv_ostream << ";" << std::setw(12) << std::fixed << std::setprecision(3) << point.y;
913  csv_ostream << ";" << std::setw(12) << std::fixed << std::setprecision(3) << point.z;
914  csv_ostream << ";" << std::setw(12) << std::fixed << std::setprecision(3) << point.range;
915  csv_ostream << ";" << std::setw(12) << std::fixed << std::setprecision(8) << point.azimuth;
916  csv_ostream << ";" << std::setw(12) << std::fixed << std::setprecision(8) << point.elevation;
917  csv_ostream << ";" << std::setw(12) << std::fixed << std::setprecision(3) << point.i;
918  csv_ostream << ";" << std::setw(24) << point.lidar_timestamp_microsec;
919  csv_ostream << std::endl;
920  }
921  }
922  }
923  }
924  return true;
925 }
926 
927 /*
928  * @brief exports x, y, z, intensity, group, echo and message index of msgpack data.
929  * @param[in] result converted msgpack data, output from Parse function
930  * Note: All output vectors x, y, z, i, group_idx, echo_idx, msg_idx identical size, i.e. it's safe to
931  * assert(x.size() == y.size() && x.size() == z.size() && x.size() == i.size() && x.size() == group_idx.size() && echo_idx.size() == msg_idx.size());
932  */
933 bool sick_scansegment_xd::MsgPackParser::ExportXYZI(const std::vector<ScanSegmentParserOutput>& results, std::vector<float>& x, std::vector<float>& y, std::vector<float>& z, std::vector<float>& i, std::vector<int>& group_idx, std::vector<int>& echo_idx, std::vector<int>& msg_idx)
934 {
935  if (results.empty())
936  return false;
937  size_t data_length = 0;
938  for (int groupIdx = 0; groupIdx < results[0].scandata.size(); groupIdx++)
939  {
940  for (int echoIdx = 0; echoIdx < results[0].scandata[groupIdx].scanlines.size(); echoIdx++)
941  {
942  data_length += results[0].scandata[groupIdx].scanlines[echoIdx].points.size();
943  }
944  }
945  x.reserve(data_length);
946  y.reserve(data_length);
947  z.reserve(data_length);
948  i.reserve(data_length);
949  group_idx.reserve(data_length);
950  echo_idx.reserve(data_length);
951  msg_idx.reserve(data_length);
952  for (int msgCnt = 0; msgCnt < results.size(); msgCnt++)
953  {
954  const ScanSegmentParserOutput& result = results[msgCnt];
955  for (int groupIdx = 0; groupIdx < result.scandata.size(); groupIdx++)
956  {
957  for (int echoIdx = 0; echoIdx < result.scandata[groupIdx].scanlines.size(); echoIdx++)
958  {
959  const std::vector<sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint>& scanline = result.scandata[groupIdx].scanlines[echoIdx].points;
960  for (int pointIdx = 0; pointIdx < scanline.size(); pointIdx++)
961  {
962  const sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint& point = scanline[pointIdx];
963  x.push_back(point.x);
964  y.push_back(point.y);
965  z.push_back(point.z);
966  i.push_back(point.i);
967  group_idx.push_back(point.groupIdx);
968  echo_idx.push_back(point.echoIdx);
969  msg_idx.push_back(result.segmentIndex);
970  }
971  }
972  }
973  }
974  return true;
975 
976 }
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::range
float range
Definition: scansegment_parser_output.h:139
msgpack11::MsgPack::is_number
bool is_number() const
Definition: msgpack11.hpp:122
msgpack11::MsgPack::string_value
const std::string & string_value() const
Definition: msgpack11.cpp:742
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::elevation
float elevation
Definition: scansegment_parser_output.h:141
MsgPackToFloat32VectorConverter::printRad2Deg
std::string printRad2Deg(void)
Definition: msgpack_parser.cpp:407
sick_scansegment_xd::Timestamp
std::string Timestamp(uint32_t sec, uint32_t nsec)
Definition: scansegment_parser_output.cpp:100
MsgPackElement::elemSz
const msgpack11::MsgPack * elemSz
Definition: msgpack_parser.cpp:303
SoftwarePLL
class SoftwarePLL implements synchronisation between ticks and timestamp. See https://github....
Definition: softwarePLL.h:21
MsgPackToFloat32VectorConverter::MsgPackToFloat32VectorConverter
MsgPackToFloat32VectorConverter(const MsgPackElement &msgpack, bool dstIsBigEndian)
Definition: msgpack_parser.cpp:315
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::echoIdx
int echoIdx
Definition: scansegment_parser_output.h:143
msg
msg
MsgpackKeyToInt_uint16
#define MsgpackKeyToInt_uint16
Definition: msgpack_parser.cpp:157
MsgpackKeyToInt_ChannelPhi
#define MsgpackKeyToInt_ChannelPhi
Definition: msgpack_parser.cpp:159
MsgpackKeyToInt_MaxValue
#define MsgpackKeyToInt_MaxValue
Definition: msgpack_parser.cpp:182
SoftwarePLL::updatePLL
bool updatePLL(uint32_t sec, uint32_t nanoSec, uint32_t curtick)
Definition: softwarePLL.cpp:191
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::z
float z
Definition: scansegment_parser_output.h:137
angle
TFSIMD_FORCE_INLINE tfScalar angle(const Quaternion &q1, const Quaternion &q2)
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::azimuth
float azimuth
Definition: scansegment_parser_output.h:140
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::lidar_timestamp_microsec
uint64_t lidar_timestamp_microsec
Definition: scansegment_parser_output.h:145
sick_scan_xd::SickCloudTransform
Definition: sick_cloud_transform.h:85
MsgpackKeyToInt_endian
#define MsgpackKeyToInt_endian
Definition: msgpack_parser.cpp:151
sick_scansegment_xd::MsgPackParser::telegramCount
static int telegramCount
Definition: msgpack_parser.h:236
sick_scansegment_xd::ScanSegmentParserOutput::timestamp_nsec
uint32_t timestamp_nsec
Definition: scansegment_parser_output.h:187
MsgPackToFloat32VectorConverter::m_data
std::vector< float > m_data
Definition: msgpack_parser.cpp:423
s
XmlRpcServer s
sick_scansegment_xd::MsgPackValidatorData
Definition: msgpack_validator.h:86
MsgpackKeyToInt_RssiValues
#define MsgpackKeyToInt_RssiValues
Definition: msgpack_parser.cpp:161
MsgpackKeyToInt_PropertiesValues
#define MsgpackKeyToInt_PropertiesValues
Definition: msgpack_parser.cpp:162
MsgPackToFloat32VectorConverter::rad2deg
float rad2deg(float angle) const
Definition: msgpack_parser.cpp:406
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::pointIdx
int pointIdx
Definition: scansegment_parser_output.h:144
SoftwarePLL::instance
static SoftwarePLL & instance()
Definition: softwarePLL.h:24
msgpack11::MsgPack::int_value
int32_t int_value() const
Definition: msgpack11.cpp:732
msgpack11::MsgPack::is_array
bool is_array() const
Definition: msgpack11.hpp:135
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::groupIdx
int groupIdx
Definition: scansegment_parser_output.h:142
msgpack11::MsgPack::parse
static MsgPack parse(const std::string &in, std::string &err)
roswrap::console::print
ROSCONSOLE_DECL void print(FilterBase *filter, void *logger, Level level, const char *file, int line, const char *function, const char *fmt,...) ROSCONSOLE_PRINTF_ATTRIBUTE(7
Don't call this directly. Use the ROS_LOG() macro instead.
MsgpackKeyToInt_data
#define MsgpackKeyToInt_data
Definition: msgpack_parser.cpp:148
MsgpackKeyToInt_elemTypes
#define MsgpackKeyToInt_elemTypes
Definition: msgpack_parser.cpp:152
SoftwarePLL::getCorrectedTimeStamp
bool getCorrectedTimeStamp(uint32_t &sec, uint32_t &nanoSec, uint32_t tick)
Definition: softwarePLL.cpp:226
MsgPackElement::elemTypes
const msgpack11::MsgPack * elemTypes
Definition: msgpack_parser.cpp:304
sick_scansegment_xd::MsgPackValidator::validateNotOutOfBound
bool validateNotOutOfBound(const MsgPackValidatorData &data_received) const
Definition: msgpack_validator.cpp:210
data
data
msgpack11::MsgPack::binary
std::vector< uint8_t > binary
Definition: msgpack11.hpp:60
sick_scansegment_xd::ScanSegmentParserOutput::segmentIndex
int segmentIndex
Definition: scansegment_parser_output.h:192
multiscan_pcap_player.verbose
int verbose
Definition: multiscan_pcap_player.py:142
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::y
float y
Definition: scansegment_parser_output.h:136
MsgpackKeyToInt_EchoCount
#define MsgpackKeyToInt_EchoCount
Definition: msgpack_parser.cpp:171
MsgPackToFloat32VectorConverter::print
std::string print(void)
Definition: msgpack_parser.cpp:395
ROS_WARN_STREAM
#define ROS_WARN_STREAM(args)
Definition: sick_scan_logging.h:123
ROS_INFO_STREAM
#define ROS_INFO_STREAM(...)
Definition: sick_scan_ros2_example.cpp:71
softwarePLL.h
sick_scansegment_xd::ScanSegmentParserOutput
Definition: scansegment_parser_output.h:118
msgpack11.hpp
msgpack_parser.h
MsgPackElement
Definition: msgpack_parser.cpp:289
MsgpackKeyToInt_SegmentCounter
#define MsgpackKeyToInt_SegmentCounter
Definition: msgpack_parser.cpp:173
sick_scansegment_xd::ScanSegmentParserOutput::telegramCnt
int telegramCnt
Definition: scansegment_parser_output.h:193
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::x
float x
Definition: scansegment_parser_output.h:135
sick_scansegment_xd::MsgPackParser::messageCount
static int messageCount
Definition: msgpack_parser.h:235
sick_scansegment_xd::MsgPackParser::ExportXYZI
static bool ExportXYZI(const std::vector< ScanSegmentParserOutput > &results, std::vector< float > &x, std::vector< float > &y, std::vector< float > &z, std::vector< float > &i, std::vector< int > &group_idx, std::vector< int > &echo_idx, std::vector< int > &msg_idx)
Definition: msgpack_parser.cpp:933
sick_scansegment_xd::MsgPackParser::Parse
static bool Parse(const std::vector< uint8_t > &msgpack_data, fifo_timestamp msgpack_timestamp, sick_scan_xd::SickCloudTransform &add_transform_xyz_rpy, ScanSegmentParserOutput &result, sick_scansegment_xd::MsgPackValidatorData &msgpack_validator_data_collector, const sick_scansegment_xd::MsgPackValidator &msgpack_validator=sick_scansegment_xd::MsgPackValidator(), bool msgpack_validator_enabled=false, bool discard_msgpacks_not_validated=false, bool use_software_pll=true, bool verbose=false)
Definition: msgpack_parser.cpp:481
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint
Definition: scansegment_parser_output.h:129
fifo_timestamp
std::chrono::time_point< std::chrono::system_clock > fifo_timestamp
Definition: fifo.h:68
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::i
float i
Definition: scansegment_parser_output.h:138
sick_scansegment_xd::FileReadable
static bool FileReadable(const std::string &filename)
Definition: include/sick_scansegment_xd/common.h:170
normalizeAngle
static float normalizeAngle(float angle_rad)
Definition: msgpack_parser.cpp:86
MsgPackKeyValues::values
std::vector< msgpack11::MsgPack > values
Definition: msgpack_parser.cpp:196
MsgPackElement::endian
const msgpack11::MsgPack * endian
Definition: msgpack_parser.cpp:305
msgpack11::MsgPack::int32_value
int32_t int32_value() const
Definition: msgpack11.cpp:735
msgpack11::MsgPack::object_items
const object & object_items() const
Definition: msgpack11.cpp:746
sick_scansegment_xd::MsgPackValidatorData::print
std::vector< std::string > print(void) const
Definition: msgpack_validator.cpp:125
MsgpackKeyToInt_TimestampStop
#define MsgpackKeyToInt_TimestampStop
Definition: msgpack_parser.cpp:165
s_msgpack_keys
static MsgPackKeyValues s_msgpack_keys
Definition: msgpack_parser.cpp:198
MsgPackToFloat32VectorConverter::data
std::vector< float > & data(void)
Definition: msgpack_parser.cpp:418
MsgpackKeyToInt_DistValues
#define MsgpackKeyToInt_DistValues
Definition: msgpack_parser.cpp:160
sick_scansegment_xd::ScanSegmentParserOutput::timestamp_sec
uint32_t timestamp_sec
Definition: scansegment_parser_output.h:186
MsgpackKeyToInt_SegmentData
#define MsgpackKeyToInt_SegmentData
Definition: msgpack_parser.cpp:178
sick_scansegment_xd::MsgPackParser::ReadFile
static std::vector< uint8_t > ReadFile(const std::string &filepath)
Definition: msgpack_parser.cpp:431
sick_scansegment_xd::MsgPackParser::MsgpackToHexDump
static std::string MsgpackToHexDump(const std::vector< uint8_t > &msgpack_data, bool pretty_print=true)
Definition: msgpack_parser.cpp:443
SoftwarePLL::IsInitialized
bool IsInitialized() const
Definition: softwarePLL.h:47
sick_scansegment_xd::MsgPackValidatorData::update
void update(int echo_idx, int segment_idx, float azimuth, float elevation)
Definition: msgpack_validator.cpp:85
MsgpackKeyToInt_uint8
#define MsgpackKeyToInt_uint8
Definition: msgpack_parser.cpp:156
MsgPackElement::MsgPackElement
MsgPackElement()
Definition: msgpack_parser.cpp:292
MsgpackKeyToInt_elemSz
#define MsgpackKeyToInt_elemSz
Definition: msgpack_parser.cpp:150
msgpack11::MsgPack::object
std::map< MsgPack, MsgPack > object
Definition: msgpack11.hpp:57
MsgPackToFloat32VectorConverter
Definition: msgpack_parser.cpp:311
MsgpackKeyToInt_TelegramCounter
#define MsgpackKeyToInt_TelegramCounter
Definition: msgpack_parser.cpp:180
config.h
sick_scansegment_xd::MsgPackValidator
Definition: msgpack_validator.h:201
sick_scansegment_xd::ScanSegmentParserOutput::scandata
std::vector< Scangroup > scandata
Definition: scansegment_parser_output.h:175
msgpack11::MsgPack::binary_items
const binary & binary_items() const
Definition: msgpack11.cpp:744
sick_scansegment_xd::ScanSegmentParserOutput::Scanline
Definition: scansegment_parser_output.h:152
ROS_ERROR_STREAM
#define ROS_ERROR_STREAM(...)
Definition: sick_scan_ros2_example.cpp:72
sick_scan_xd::SickCloudTransform::applyTransform
void applyTransform(float_type &x, float_type &y, float_type &z)
Definition: sick_cloud_transform.h:100
MsgpackKeyToInt_ChannelTheta
#define MsgpackKeyToInt_ChannelTheta
Definition: msgpack_parser.cpp:158
MsgpackKeyToInt_TimestampStart
#define MsgpackKeyToInt_TimestampStart
Definition: msgpack_parser.cpp:164
MsgPackKeyValues::MsgPackKeyValues
MsgPackKeyValues()
Definition: msgpack_parser.cpp:190
MsgpackKeyToInt_float32
#define MsgpackKeyToInt_float32
Definition: msgpack_parser.cpp:154
printMsgPack
static std::string printMsgPack(const msgpack11::MsgPack &msg)
Definition: msgpack_parser.cpp:246
MsgPackKeyValues
Definition: msgpack_parser.cpp:187
msgpack11::MsgPack::array_items
const array & array_items() const
Definition: msgpack11.cpp:743
msgpack11::MsgPack
Definition: msgpack11.hpp:29
sick_scansegment_xd::MsgPackParser::WriteCSV
static bool WriteCSV(const std::vector< ScanSegmentParserOutput > &results, const std::string &csvFile, bool overwrite_existing_file)
Definition: msgpack_parser.cpp:874
MsgPackElement::data
const msgpack11::MsgPack * data
Definition: msgpack_parser.cpp:302
sick_scansegment_xd::Config::SystemIsBigEndian
static bool SystemIsBigEndian(void)
Definition: config.cpp:127
MsgPackToFloat32VectorConverter::MsgPackToFloat32VectorConverter
MsgPackToFloat32VectorConverter()
Definition: msgpack_parser.cpp:314
sick_scan_xd::SickCloudTransform::azimuthOffset
float azimuthOffset(void) const
Definition: sick_cloud_transform.h:133
sick_scansegment_xd::ScanSegmentParserOutput::Scanline::points
std::vector< LidarPoint > points
Definition: scansegment_parser_output.h:155
sick_scansegment_xd::ScanSegmentParserOutput::Scangroup
Definition: scansegment_parser_output.h:161
sick_scansegment_xd::ScanSegmentParserOutput::timestamp
std::string timestamp
Definition: scansegment_parser_output.h:185
MsgpackKeyToInt_TimestampTransmit
#define MsgpackKeyToInt_TimestampTransmit
Definition: msgpack_parser.cpp:181
MsgPackElement::MsgPackElement
MsgPackElement(const msgpack11::MsgPack::object &object_items)
Definition: msgpack_parser.cpp:293


sick_scan_xd
Author(s): Michael Lehning , Jochen Sprickerhof , Martin Günther
autogenerated on Fri Oct 25 2024 02:47:09