compact_parser.cpp
Go to the documentation of this file.
1 /*
2  * @brief compact_parser parses and convertes scandata in compact format
3  *
4  * Copyright (C) 2020,2021 Ing.-Buero Dr. Michael Lehning, Hildesheim
5  * Copyright (C) 2020,2021 SICK AG, Waldkirch
6  *
7  * Licensed under the Apache License, Version 2.0 (the "License");
8  * you may not use this file except in compliance with the License.
9  * You may obtain a copy of the License at
10  *
11  * http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing, software
14  * distributed under the License is distributed on an "AS IS" BASIS,
15  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16  * See the License for the specific language governing permissions and
17  * limitations under the License.
18  *
19  * All rights reserved.
20  *
21  * Redistribution and use in source and binary forms, with or without
22  * modification, are permitted provided that the following conditions are met:
23  *
24  * * Redistributions of source code must retain the above copyright
25  * notice, this list of conditions and the following disclaimer.
26  * * Redistributions in binary form must reproduce the above copyright
27  * notice, this list of conditions and the following disclaimer in the
28  * documentation and/or other materials provided with the distribution.
29  * * Neither the name of SICK AG nor the names of its
30  * contributors may be used to endorse or promote products derived from
31  * this software without specific prior written permission
32  * * Neither the name of Ing.-Buero Dr. Michael Lehning nor the names of its
33  * contributors may be used to endorse or promote products derived from
34  * this software without specific prior written permission
35  *
36  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
37  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
38  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
39  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
40  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
41  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
42  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
43  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
44  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
45  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
46  * POSSIBILITY OF SUCH DAMAGE.
47  *
48  * Authors:
49  * Michael Lehning <michael.lehning@lehning.de>
50  *
51  * Copyright 2020 SICK AG
52  * Copyright 2020 Ing.-Buero Dr. Michael Lehning
53  *
54  */
55 #include "sick_scan/softwarePLL.h"
59 
61 {
62  uint8_t u8_bytes[4];
63  uint16_t u16_bytes[2];
64  uint32_t u32_bytes[1];
65  float f32_val[1];
66 };
67 
69 {
70  uint8_t u8_bytes[8];
71  uint16_t u16_bytes[4];
72  uint32_t u32_bytes[2];
73  uint64_t u64_bytes[1];
74  float f32_val[2];
75  double f64_val[1];
76 };
77 
79 {
83 
84 template <typename T> static inline T readUnsigned(const uint8_t* scandata, uint32_t* byte_cnt)
85 {
86 #if TARGET_IS_LITTLE_ENDIAN // src and dst have identical endianess
87  *byte_cnt += sizeof(T);
88  return (*((T*)scandata));
89 #else // src and dst have different endianess: reorder bytes
90  COMPACT_8BYTE_UNION buffer;
91  buffer.u64_bytes[0] = 0;
92  for (int n = 0, m = sizeof(T) - 1; n < sizeof(T); n++,m--)
93  {
94  buffer.u8_bytes[n] = scandata[m];
95  }
96  *byte_cnt += sizeof(T);
97  return (T)(buffer.u64_bytes[0]);
98 #endif
99 }
100 
101 static inline float readFloat32(const uint8_t* scandata, uint32_t* byte_cnt)
102 {
103 #if TARGET_IS_LITTLE_ENDIAN // src and dst have identical endianess
104  *byte_cnt += sizeof(float);
105  return (*((float*)scandata));
106 #else // src and dst have different endianess: reorder bytes
107  COMPACT_4BYTE_UNION buffer;
108  buffer.u32_bytes[0] = 0;
109  for (int n = 0, m = sizeof(float) - 1; n < sizeof(float); n++,m--)
110  {
111  buffer.u8_bytes[n] = scandata[m];
112  }
113  *byte_cnt += 4;
114  return buffer.f32_val[0];
115 #endif
116 }
117 
118 static inline bool endOfBuffer(uint32_t byte_cnt, size_t bytes_to_read, uint32_t num_bytes)
119 {
120  return ((byte_cnt) + (bytes_to_read) > (num_bytes));
121 }
122 
123 static void print_error(const std::string& err_msg, int line_number, double print_rate = 1)
124 {
125  static std::map<int, std::chrono::system_clock::time_point> last_error_printed;
126  static std::map<int, size_t> error_cnt;
127  if (error_cnt[line_number] == 0 || std::chrono::duration<double>(std::chrono::system_clock::now() - last_error_printed[line_number]).count() > 1/print_rate)
128  {
129  if (error_cnt[line_number] <= 1)
130  ROS_ERROR_STREAM(err_msg);
131  else
132  ROS_ERROR_STREAM(err_msg << " (error repeated " << error_cnt[line_number] << " times)");
133  last_error_printed[line_number] = std::chrono::system_clock::now();
134  error_cnt[line_number] = 0;
135  }
136  error_cnt[line_number] += 1;
137 }
138 
139 #define CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, bytes_to_read, module_size, name, line_number) \
140 if (((byte_required) = (byte_cnt) + (bytes_to_read)) > (module_size)) \
141 { \
142  std::stringstream err_msg; \
143  err_msg << "## ERROR CompactDataParser::ParseModuleMetaData(): module_size=" << (module_size) << ", " \
144  << (byte_required) << " bytes required to read " << (name); \
145  print_error(err_msg.str(), line_number); \
146  return (metadata); \
147 }
148 
151 {
152  std::stringstream description;
153  if (valid)
154  {
155  description << "acceleration_x:" << std::fixed << std::setprecision(1) << acceleration_x;
156  description << ", acceleration_y:" << std::fixed << std::setprecision(1) << acceleration_y;
157  description << ", acceleration_z:" << std::fixed << std::setprecision(1) << acceleration_z;
158  description << ", angular_velocity_x:" << std::fixed << std::setprecision(1) << angular_velocity_x;
159  description << ", angular_velocity_y:" << std::fixed << std::setprecision(1) << angular_velocity_y;
160  description << ", angular_velocity_z:" << std::fixed << std::setprecision(1) << angular_velocity_z;
161  description << ", orientation_w:" << std::fixed << std::setprecision(1) << orientation_w;
162  description << ", orientation_x:" << std::fixed << std::setprecision(1) << orientation_x;
163  description << ", orientation_y:" << std::fixed << std::setprecision(1) << orientation_y;
164  description << ", orientation_z:" << std::fixed << std::setprecision(1) << orientation_z;
165  }
166  return description.str();
167 }
168 
169 
172 {
173  std::stringstream description;
174  description << "commandId:" << commandId;
175  description << ", telegramVersion:" << telegramVersion;
176  description << ", timeStampTransmit:" << timeStampTransmit;
177  if (isImu())
178  {
179  description << ", IMU, " << imudata.to_string();
180  }
181  else
182  {
183  description << ", telegramCounter:" << telegramCounter;
184  description << ", sizeModule0:" << sizeModule0;
185  }
186  return description.str();
187 }
188 
191 {
192  std::stringstream description;
193  description << "SegmentCounter:" << SegmentCounter;
194  description << ", FrameNumber:" << FrameNumber;
195  description << ", SenderId:" << SenderId;
196  description << ", NumberOfLinesInModule:" << NumberOfLinesInModule;
197  description << ", NumberOfBeamsPerScan:" << NumberOfBeamsPerScan;
198  description << ", NumberOfEchosPerBeam:" << NumberOfEchosPerBeam;
199  description << ", TimeStampStart:[";
200  for(int n = 0; n < TimeStampStart.size(); n++)
201  description << (n>0?",":"") << TimeStampStart[n];
202  description << "], TimeStampStop:[";
203  for(int n = 0; n < TimeStampStop.size(); n++)
204  description << (n>0?",":"") << TimeStampStop[n];
205  description << "], Phi:[";
206  for(int n = 0; n < Phi.size(); n++)
207  description << (n>0?",":"") << Phi[n];
208  description << "], ThetaStart:[";
209  for(int n = 0; n < ThetaStart.size(); n++)
210  description << (n>0?",":"") << ThetaStart[n];
211  description << "], ThetaStop:[";
212  for(int n = 0; n < ThetaStop.size(); n++)
213  description << (n>0?",":"") << ThetaStop[n];
214  description << "], DistanceScalingFactor:" << DistanceScalingFactor;
215  description << ", NextModuleSize:" << NextModuleSize;
216  description << ", Availability:" << (int)Availability;
217  description << ", DataContentEchos:" << (int)DataContentEchos;
218  description << ", DataContentBeams:" << (int)DataContentBeams;
219  description << ", valid:" << (int)valid;
220  return description.str();
221 }
222 
225 {
226  std::stringstream description;
227  for(int group_idx = 0; group_idx < scandata.size(); group_idx++)
228  {
229  description << (group_idx > 0 ? "," : "") << "scandata[" << group_idx << "]=[";
230  const std::vector<ScanSegmentParserOutput::Scanline>& scanlines = scandata[group_idx].scanlines;
231  for(int line_idx = 0; line_idx < scanlines.size(); line_idx++)
232  {
233  description << (line_idx > 0 ? "," : "") << "scanline[" << line_idx << "]=[";
234  for(int point_idx = 0; point_idx < scanlines[line_idx].points.size(); point_idx++)
235  {
236  const ScanSegmentParserOutput::LidarPoint& point = scanlines[line_idx].points[point_idx];
237  description << (point_idx > 0 ? "," : "") << "(";
238  description << point.x << "," << point.y << "," << point.z << "," << point.i << ",";
239  description << point.range << "," << point.azimuth << "," << point.elevation << ",";
240  description << point.groupIdx << "," << point.echoIdx << "," << point.pointIdx << ")";
241  }
242  description << "]" << (scanlines.size() > 1 ? "\n": "");
243  }
244  description << "]" << (scandata.size() > 1 ? "\n": "");
245  }
246  description << ", valid:" << (int)valid;
247  return description.str();
248 }
249 
250 /*
251 * @brief parses scandata header in compact format.
252 * @param[in] scandata 32 byte scandata header
253 */
255 {
256  uint32_t byte_cnt = 0;
258  header.commandId = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // Telegram type, expected value: 1
259  if (header.commandId == 1) // scan data in compact format
260  {
261  header.telegramCounter = readUnsigned<uint64_t>(scandata + byte_cnt, &byte_cnt); // Incrementing telegram counter, starting with 1, number of telegrams since power up
262  header.timeStampTransmit = readUnsigned<uint64_t>(scandata + byte_cnt, &byte_cnt); // Sensor timestamp in microseconds since 1.1.1970 00:00 UTC
263  header.telegramVersion = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // Telegram version, expected value: 4
264  header.sizeModule0 = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // Size of first module in byte
265  }
266  else if (header.commandId == 2) // imu data in compact format, all values in little endian
267  {
268  header.telegramCounter = 0; // not transmitted
269  header.sizeModule0 = 0; // no further payload
270  header.telegramVersion = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // Telegram version, expected value: 1
271  header.imudata.acceleration_x = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float in m/s^2, acceleration along the x-axis including gravity
272  header.imudata.acceleration_y = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float in m/s^2, acceleration along the y-axis including gravity
273  header.imudata.acceleration_z = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float in m/s^2, acceleration along the z-axis including gravity
274  header.imudata.angular_velocity_x = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float in rad/s
275  header.imudata.angular_velocity_y = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float in rad/s
276  header.imudata.angular_velocity_z = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float in rad/s
277  header.imudata.orientation_w = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float, orientation quaternion w
278  header.imudata.orientation_x = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float, orientation quaternion x
279  header.imudata.orientation_y = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float, orientation quaternion y
280  header.imudata.orientation_z = readFloat32(scandata + byte_cnt, &byte_cnt); // 4 bytes float, orientation quaternion z
281  header.timeStampTransmit = readUnsigned<uint64_t>(scandata + byte_cnt, &byte_cnt); // Sensor timestamp in microseconds since 1.1.1970 00:00 UTC
282  header.imudata.valid = true;
283  }
284  else
285  {
286  ROS_WARN_STREAM("CompactDataParser::ParseHeader: header.commandId = " << header.commandId << " not supported");
287  }
288  /* Create dummy IMU data, TEST ONLY **
289  static int dummy_imu_data_cnt = 0;
290  if (((++dummy_imu_data_cnt) % 2) == 0)
291  {
292  uint64_t timeStampTransmit = header.timeStampTransmit;
293  header = sick_scansegment_xd::CompactDataHeader();
294  header.commandId = 2;
295  header.telegramVersion = 1;
296  header.imudata.acceleration_x = 1;
297  header.imudata.acceleration_y = 2;
298  header.imudata.acceleration_z = 3;
299  header.imudata.angular_velocity_x = 4;
300  header.imudata.angular_velocity_y = 5;
301  header.imudata.angular_velocity_z = 6;
302  header.imudata.orientation_w = 7;
303  header.imudata.orientation_x = 8;
304  header.imudata.orientation_y = 9;
305  header.imudata.orientation_z = 10;
306  header.timeStampTransmit = timeStampTransmit;
307  header.imudata.valid = true;
308  }
309  ** End of dummy IMU data */
310  return header;
311 }
312 
313 /*
314 * @brief Parses module meta data in compact format.
315 * @param[in] scandata received bytes
316 * @param[in] module_size Number of bytes to parse
317 * @param[in] telegramVersion compact format version, currently version 3 and 4 supported
318 * @param[out] module_metadata_size number of bytes parsed (i.e. size of meta data in bytes if successful, i.e. if metadata.valid == true)
319 */
320 sick_scansegment_xd::CompactModuleMetaData sick_scansegment_xd::CompactDataParser::ParseModuleMetaData(const uint8_t* scandata, uint32_t module_size, uint32_t telegramVersion, uint32_t& module_metadata_size)
321 {
322  uint32_t byte_cnt = 0, byte_required = 0;
324  // metadata.valid flag is false and becomes true after successful parsing
325  module_metadata_size = 0;
326  // SegmentCounter
327  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint64_t), module_size, "SegmentCounter", __LINE__);
328  metadata.SegmentCounter = readUnsigned<uint64_t>(scandata + byte_cnt, &byte_cnt); // Incrementing segment counter
329  // FrameNumber
330  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint64_t), module_size, "FrameNumber", __LINE__);
331  metadata.FrameNumber = readUnsigned<uint64_t>(scandata + byte_cnt, &byte_cnt); // Number of frames since power on
332  // SenderId
333  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint32_t), module_size, "SenderId", __LINE__);
334  metadata.SenderId = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // The serial number of the device
335  // NumberOfLinesInModule
336  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint32_t), module_size, "NumberOfLinesInModule", __LINE__);
337  metadata.NumberOfLinesInModule = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // Number of layers in this module
338  if (metadata.NumberOfLinesInModule > 16)
339  {
340  std::stringstream err_msg;
341  err_msg << "## ERROR CompactDataParser::ParseModuleMetaData(): unexpected NumberOfLinesInModule=" << metadata.NumberOfLinesInModule;
342  print_error(err_msg.str(), __LINE__);
343  }
344  // NumberOfBeamsPerScan
345  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint32_t), module_size, "NumberOfBeamsPerScan", __LINE__);
346  metadata.NumberOfBeamsPerScan = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // Number of beams per layer (all layers have the same number of beams)
347  // NumberOfEchosPerBeam
348  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint32_t), module_size, "NumberOfEchosPerBeam", __LINE__);
349  metadata.NumberOfEchosPerBeam = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // Number of echos per beams
350  // TimeStampStart
351  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, metadata.NumberOfLinesInModule * sizeof(uint64_t), module_size, "TimeStampStart", __LINE__);
352  metadata.TimeStampStart.reserve(metadata.NumberOfLinesInModule);
353  for(uint32_t cnt = 0; cnt < metadata.NumberOfLinesInModule; cnt++) // Array of timestamps of the first beam in a scan in microseconds, number of elements is numberOfLinesInModule
354  {
355  metadata.TimeStampStart.push_back(readUnsigned<uint64_t>(scandata + byte_cnt, &byte_cnt));
356  }
357  // TimeStampStop
358  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, metadata.NumberOfLinesInModule * sizeof(uint64_t), module_size, "TimeStampStop", __LINE__);
359  metadata.TimeStampStop.reserve(metadata.NumberOfLinesInModule);
360  for(uint32_t cnt = 0; cnt < metadata.NumberOfLinesInModule; cnt++) // Array of timestamps of the last beam in a scan in microseconds, number of elements is numberOfLinesInModule
361  {
362  metadata.TimeStampStop.push_back(readUnsigned<uint64_t>(scandata + byte_cnt, &byte_cnt));
363  }
364  // Phi
365  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, metadata.NumberOfLinesInModule * sizeof(float), module_size, "Phi", __LINE__);
366  metadata.Phi.reserve(metadata.NumberOfLinesInModule);
367  for(uint32_t cnt = 0; cnt < metadata.NumberOfLinesInModule; cnt++) // Array of elevation angles in radians for each layer in this module, number of elements is numberOfLinesInModule
368  {
369  float val = readFloat32(scandata + byte_cnt, &byte_cnt);
370  metadata.Phi.push_back(val);
371  }
372  // ThetaStart
373  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, metadata.NumberOfLinesInModule * sizeof(float), module_size, "ThetaStart", __LINE__);
374  metadata.ThetaStart.reserve(metadata.NumberOfLinesInModule);
375  for(uint32_t cnt = 0; cnt < metadata.NumberOfLinesInModule; cnt++) // Array of azimuth angles in radians for first beam of each layer in this module, number of elements is numberOfLinesInModule
376  {
377  float val = readFloat32(scandata + byte_cnt, &byte_cnt);
378  metadata.ThetaStart.push_back(val);
379  }
380  // ThetaStop
381  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, metadata.NumberOfLinesInModule * sizeof(float), module_size, "ThetaStop", __LINE__);
382  metadata.ThetaStop.reserve(metadata.NumberOfLinesInModule);
383  for(uint32_t cnt = 0; cnt < metadata.NumberOfLinesInModule; cnt++) // Array of azimuth angles in radians for last beam of each layer in this module, number of elements is numberOfLinesInModule
384  {
385  float val = readFloat32(scandata + byte_cnt, &byte_cnt);
386  metadata.ThetaStop.push_back(val);
387  }
388  // DistanceScalingFactor for 16-bit distance values: distance_mm = DistanceScalingFactor * distance_16bit_sensor
389  if (telegramVersion == 3)
390  {
391  metadata.DistanceScalingFactor = 1;
392  }
393  else if (telegramVersion == 4)
394  {
395  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(float), module_size, "DistanceScalingFactor", __LINE__);
396  metadata.DistanceScalingFactor = readFloat32(scandata + byte_cnt, &byte_cnt);
397  }
398  else
399  {
400  ROS_ERROR_STREAM("## ERROR CompactDataParser::ParseModuleMetaData(): telegramVersion=" << telegramVersion << " not supported");
401  return metadata;
402  }
403  // NextModuleSize
404  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint32_t), module_size, "NextModuleSize", __LINE__);
405  metadata.NextModuleSize = readUnsigned<uint32_t>(scandata + byte_cnt, &byte_cnt); // Size of the next module (or 0 if this module is the last one)
406  // Availability
407  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint8_t), module_size, "Availability", __LINE__);
408  metadata.Availability = readUnsigned<uint8_t>(scandata + byte_cnt, &byte_cnt); // Reserved for futher use (flag indication distortion in scan data)
409  // DataContentEchos
410  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint8_t), module_size, "DataContentEchos", __LINE__);
411  metadata.DataContentEchos = readUnsigned<uint8_t>(scandata + byte_cnt, &byte_cnt); // Bitarray: (DataContentEchos & 0x01) == 0x01 if distance values available, (DataContentEchos & 0x02) == 0x02 if RSSI values available, (DataContentEchos & 0x03) == 0x03: distance and RSSI values available
412  // DataContentBeams
413  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint8_t), module_size, "DataContentBeams", __LINE__);
414  metadata.DataContentBeams = readUnsigned<uint8_t>(scandata + byte_cnt, &byte_cnt); // Bitarray: (DataContentBeams & 0x01) == 0x01 if beam properties available, (DataContentBeams & 0x02) == 0x02 if azimuth angle per beeam available, (DataContentBeams & 0x03) == 0x03: both available
415  // reserved
416  CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, sizeof(uint8_t), module_size, "reserved", __LINE__);
417  metadata.reserved = readUnsigned<uint8_t>(scandata + byte_cnt, &byte_cnt); // Reserved for 32 bit alignment
418  // valid flag set true after successful parsing
419  metadata.valid = true;
420  module_metadata_size = byte_cnt;
421  return metadata;
422 }
423 
424 static std::vector<int> s_layer_elevation_table_mdeg; // Optional elevation LUT in mdeg for layers in compact format, s_layer_elevation_table_mdeg[layer_idx] := ideal elevation in mdeg
425 
426 /*
427 * @brief Sets the elevation in mdeg for layers in compact format.
428 * @param[in] layer_elevation_table_mdeg layer_elevation_table_mdeg[layer_idx] := ideal elevation in mdeg
429 */
430 void sick_scansegment_xd::CompactDataParser::SetLayerElevationTable(const std::vector<int>& layer_elevation_table_mdeg)
431 {
432  s_layer_elevation_table_mdeg = layer_elevation_table_mdeg;
433 }
434 
435 /*
436 * @brief Return a layer-id from a given elevation angle. See compact scanformat documention:
437 * The line/layer index in the figure below is not a layer id according to layer numbering for multi layer sensors.
438 * Therefore this functions returns a layer-id from the elevation angle in rad.
439 * @param[in] layer_elevation_rad layer_elevation in radians
440 * @return layer-id
441 */
442 int sick_scansegment_xd::CompactDataParser::GetLayerIDfromElevation(float layer_elevation_rad) // layer_elevation in radians
443 {
444  int layer_elevation_mdeg = (int)std::lround(layer_elevation_rad * 180000 / M_PI);
445  if (!s_layer_elevation_table_mdeg.empty())
446  {
447  int layer_idx = 0;
448  int elevation_dist = std::abs(layer_elevation_mdeg - s_layer_elevation_table_mdeg[layer_idx]);
449  for(int n = 1; n < s_layer_elevation_table_mdeg.size(); n++)
450  {
451  int dist = std::abs(layer_elevation_mdeg - s_layer_elevation_table_mdeg[n]);
452  if (elevation_dist > dist)
453  {
454  elevation_dist = dist;
455  layer_idx = n;
456  }
457  else
458  {
459  break;
460  }
461  }
462  return layer_idx;
463  }
464  else
465  {
466  static std::map<int,int> elevation_layerid_map;
467  if (elevation_layerid_map.find(layer_elevation_mdeg) == elevation_layerid_map.end())
468  {
469  elevation_layerid_map[layer_elevation_mdeg] = elevation_layerid_map.size() + 1; // Add new layer
470  int layerid = 0;
471  for(std::map<int,int>::iterator iter_layerid_map = elevation_layerid_map.begin(); iter_layerid_map != elevation_layerid_map.end(); iter_layerid_map++)
472  iter_layerid_map->second = layerid++; // Resort by ascending elevation
473  }
474  return elevation_layerid_map[layer_elevation_mdeg];
475  }
476 }
477 
478 /*
479 * @brief Return the typical (default) elevation of a given layer index
480 * @param[in] layer_idx layer index
481 * @return layer elevation in degree
482 */
484 {
485  if (layer_idx >= 0 && layer_idx < s_layer_elevation_table_mdeg.size())
486  {
487  return 0.001f * s_layer_elevation_table_mdeg[layer_idx];
488  }
489  return 0;
490 }
491 
492 
493 /*
494 * @brief Parses module measurement data in compact format.
495 * @param[in] payload binary payload
496 * @param[in] num_bytes size of binary payload in bytes
497 * @param[in] meta_data module metadata with measurement properties
498 * @param[out] measurement_data parsed and converted module measurement data
499 * @return true on success, false on error
500 */
502  const sick_scansegment_xd::CompactModuleMetaData& meta_data, float azimuth_offset, sick_scansegment_xd::CompactModuleMeasurementData& measurement_data)
503 {
505  measurement_data.valid = false;
506  if (meta_data.NumberOfLinesInModule < 1 ||
507  meta_data.NumberOfEchosPerBeam < 1 ||
508  meta_data.NumberOfBeamsPerScan < 1)
509  {
510  ROS_ERROR_STREAM("CompactDataParser::ParseModuleMeasurementData(): invalid meta_data: { " << meta_data.to_string() << " }");
511  return false;
512  }
513  bool dist_available = ((meta_data.DataContentEchos & 0x01) == 0x01);
514  bool rssi_available = ((meta_data.DataContentEchos & 0x02) == 0x02);
515  bool beam_prop_available = ((meta_data.DataContentBeams & 0x01) == 0x01);
516  bool beam_azim_available = ((meta_data.DataContentBeams & 0x02) == 0x02);
517  float dist_scale_factor = meta_data.DistanceScalingFactor;
518 
519  uint32_t num_layers = meta_data.NumberOfLinesInModule;
520  uint32_t num_echos = meta_data.NumberOfEchosPerBeam;
521  measurement_data.scandata = std::vector<ScanSegmentParserOutput::Scangroup>(num_layers);
522 
523  ROS_DEBUG_STREAM("CompactDataParser::ParseModuleMeasurementData(): num_bytes=" << num_bytes << ", num_layers=" << num_layers
524  << ", num_points=" << meta_data.NumberOfBeamsPerScan << ", num_echos=" << num_echos << ", dist_available=" << dist_available
525  << ", rssi_available=" << rssi_available << ", beam_prop_available=" << beam_prop_available
526  << ", beam_azim_available=" << beam_azim_available);
527  // Prepare output data
528  std::vector<float> lut_layer_elevation(num_layers);
529  std::vector<float> lut_layer_azimuth_start(num_layers);
530  std::vector<float> lut_layer_azimuth_stop(num_layers);
531  std::vector<float> lut_layer_azimuth_delta(num_layers);
532  std::vector<uint64_t> lut_layer_lidar_timestamp_microsec_start(num_layers);
533  std::vector<uint64_t> lut_layer_lidar_timestamp_microsec_stop(num_layers);
534  std::vector<float> lut_sin_elevation(num_layers);
535  std::vector<float> lut_cos_elevation(num_layers);
536  std::vector<int> lut_groupIdx(num_layers);
537 
538  for (uint32_t layer_idx = 0; layer_idx < num_layers; layer_idx++)
539  {
540  uint32_t layer_timeStamp_start_sec = (meta_data.TimeStampStart[layer_idx] / 1000000);
541  uint32_t layer_timeStamp_start_nsec = 1000 * (meta_data.TimeStampStart[layer_idx] % 1000000);
542  uint32_t layer_timeStamp_stop_sec = (meta_data.TimeStampStop[layer_idx] / 1000000);
543  uint32_t layer_timeStamp_stop_nsec = 1000 * (meta_data.TimeStampStop[layer_idx] % 1000000);
544  measurement_data.scandata[layer_idx].timestampStart_sec = layer_timeStamp_start_sec;
545  measurement_data.scandata[layer_idx].timestampStart_nsec = layer_timeStamp_start_nsec;
546  measurement_data.scandata[layer_idx].timestampStop_sec = layer_timeStamp_stop_sec;
547  measurement_data.scandata[layer_idx].timestampStop_nsec = layer_timeStamp_stop_nsec;
548  measurement_data.scandata[layer_idx].scanlines = std::vector<ScanSegmentParserOutput::Scanline>(num_echos);
549  for (uint32_t echo_idx = 0; echo_idx < num_echos; echo_idx++)
550  {
551  measurement_data.scandata[layer_idx].scanlines[echo_idx].points = std::vector<ScanSegmentParserOutput::LidarPoint>(meta_data.NumberOfBeamsPerScan);
552  }
553  lut_layer_elevation[layer_idx] = -meta_data.Phi[layer_idx]; // elevation must be negated, a positive pitch-angle yields negative z-coordinates (compare to MsgPackParser::Parse in msgpack_parser.cpp)
554  lut_layer_azimuth_start[layer_idx] = meta_data.ThetaStart[layer_idx];
555  lut_layer_azimuth_stop[layer_idx] = meta_data.ThetaStop[layer_idx];
556  lut_layer_azimuth_delta[layer_idx] = (lut_layer_azimuth_stop[layer_idx] - lut_layer_azimuth_start[layer_idx]) / (float)(std::max(1, (int)meta_data.NumberOfBeamsPerScan - 1));
557  lut_layer_lidar_timestamp_microsec_start[layer_idx] = meta_data.TimeStampStart[layer_idx];
558  lut_layer_lidar_timestamp_microsec_stop[layer_idx] = meta_data.TimeStampStop[layer_idx];
559  lut_sin_elevation[layer_idx] = std::sin(lut_layer_elevation[layer_idx]);
560  lut_cos_elevation[layer_idx] = std::cos(lut_layer_elevation[layer_idx]);
561  lut_groupIdx[layer_idx] = GetLayerIDfromElevation(meta_data.Phi[layer_idx]);
562  }
563  // Parse scan data
564  uint32_t byte_cnt = 0;
565  for (uint32_t point_idx = 0; point_idx < meta_data.NumberOfBeamsPerScan; point_idx++)
566  {
567  for (uint32_t layer_idx = 0; layer_idx < num_layers; layer_idx++)
568  {
569  float layer_elevation = lut_layer_elevation[layer_idx];
570  float layer_azimuth_start = lut_layer_azimuth_start[layer_idx];
571  float layer_azimuth_stop = lut_layer_azimuth_stop[layer_idx];
572  float layer_azimuth_delta = lut_layer_azimuth_delta[layer_idx];
573  uint64_t lidar_timestamp_microsec_start = lut_layer_lidar_timestamp_microsec_start[layer_idx];
574  uint64_t lidar_timestamp_microsec_stop = lut_layer_lidar_timestamp_microsec_stop[layer_idx];
575  uint64_t lidar_timestamp_microsec = ((point_idx * (lidar_timestamp_microsec_stop - lidar_timestamp_microsec_start)) / (meta_data.NumberOfBeamsPerScan - 1)) + lidar_timestamp_microsec_start;
576  float sin_elevation = lut_sin_elevation[layer_idx];
577  float cos_elevation = lut_cos_elevation[layer_idx];
578  int groupIdx = lut_groupIdx[layer_idx];
579  std::vector<ScanSegmentParserOutput::LidarPoint> points(num_echos);
580  uint8_t beam_property = 0;
581  float azimuth = 0;
582  for (uint32_t echo_idx = 0; echo_idx < num_echos; echo_idx++)
583  {
584  if (dist_available)
585  {
586  if (endOfBuffer(byte_cnt, sizeof(uint16_t), num_bytes)) // if (byte_cnt + sizeof(uint16_t) > num_bytes)
587  {
588  ROS_ERROR_STREAM("## ERROR CompactDataParser::ParseModuleMeasurementData(" << __LINE__ << "): byte_cnt=" << byte_cnt << ", num_bytes=" << num_bytes << ", layer " << layer_idx << " of " << num_layers
589  << ", point " << point_idx << " of " << meta_data.NumberOfBeamsPerScan << ", echo " << echo_idx << " of " << num_echos);
590  return false;
591  }
592  points[echo_idx].range = (dist_scale_factor * (float)readUnsigned<uint16_t>(payload + byte_cnt, &byte_cnt)) / 1000.0f;
593  }
594  if (rssi_available)
595  {
596  if (endOfBuffer(byte_cnt, sizeof(uint16_t), num_bytes)) // if (byte_cnt + sizeof(uint16_t) > num_bytes)
597  {
598  ROS_ERROR_STREAM("## ERROR CompactDataParser::ParseModuleMeasurementData(" << __LINE__ << "): byte_cnt=" << byte_cnt << ", num_bytes=" << num_bytes << ", layer " << layer_idx << " of " << num_layers
599  << ", point " << point_idx << " of " << meta_data.NumberOfBeamsPerScan << ", echo " << echo_idx << " of " << num_echos);
600  return false;
601  }
602  points[echo_idx].i = (float)readUnsigned<uint16_t>(payload + byte_cnt, &byte_cnt);
603  }
604  }
605  std::vector<ReadBeamAzimOrderEnum> azim_prop_order;
606  if (compact_header.telegramVersion == 3) // for backward compatibility only
607  {
608  azim_prop_order = { READ_BEAM_AZIM, READ_BEAM_PROP }; // telegramVersion 3: 2 byte azimuth + 1 byte property
609  }
610  else if (compact_header.telegramVersion == 4)
611  {
612  azim_prop_order = { READ_BEAM_PROP, READ_BEAM_AZIM }; // telegramVersion 4 (default): 1 byte property + 2 byte azimuth
613  }
614  else
615  {
616  ROS_ERROR_STREAM("## ERROR CompactDataParser::ParseModuleMeasurementData(" << __LINE__ << "): telegramVersion=" << compact_header.telegramVersion << " not supported");
617  return false;
618  }
619  for (int azim_prop_cnt = 0; azim_prop_cnt < 2; azim_prop_cnt++)
620  {
621  if (azim_prop_order[azim_prop_cnt] == READ_BEAM_AZIM)
622  {
623  if (beam_azim_available)
624  {
625  if (endOfBuffer(byte_cnt, sizeof(uint16_t), num_bytes)) // if (byte_cnt + sizeof(uint16_t) > num_bytes)
626  {
627  ROS_ERROR_STREAM("## ERROR CompactDataParser::ParseModuleMeasurementData(" << __LINE__ << "): byte_cnt=" << byte_cnt << ", num_bytes=" << num_bytes << ", layer " << layer_idx << " of " << num_layers
628  << ", point " << point_idx << " of " << meta_data.NumberOfBeamsPerScan);
629  return false;
630  }
631  azimuth = ((float)readUnsigned<uint16_t>(payload + byte_cnt, &byte_cnt) - 16384.0f) / 5215.0f + azimuth_offset;
632  }
633  else
634  {
635  azimuth = layer_azimuth_start + point_idx * layer_azimuth_delta + azimuth_offset;
636  }
637  }
638  if (azim_prop_order[azim_prop_cnt] == READ_BEAM_PROP)
639  {
640  if (beam_prop_available)
641  {
642  if (byte_cnt + sizeof(uint8_t) > num_bytes)
643  {
644  ROS_ERROR_STREAM("## ERROR CompactDataParser::ParseModuleMeasurementData(" << __LINE__ << "): byte_cnt=" << byte_cnt << ", num_bytes=" << num_bytes << ", layer " << layer_idx << " of " << num_layers
645  << ", point " << point_idx << " of " << meta_data.NumberOfBeamsPerScan);
646  return false;
647  }
648  beam_property = readUnsigned<uint8_t>(payload + byte_cnt, &byte_cnt);
649  }
650  }
651  }
652  float sin_azimuth = std::sin(azimuth);
653  float cos_azimuth = std::cos(azimuth);
654  // std::stringstream s;
655  // s << "Measurement[" << layer_idx << "," << point_idx << "]=(";
656  // for(uint32_t echo_idx = 0; echo_idx < num_echos; echo_idx++)
657  // s << points[echo_idx].range << "," << points[echo_idx].i << ",";
658  // s << (int)beam_property << "," << azimuth << ")";
659  // ROS_DEBUG_STREAM("" << s.str());
660  for (uint32_t echo_idx = 0; echo_idx < num_echos; echo_idx++)
661  {
662  points[echo_idx].azimuth = azimuth;
663  points[echo_idx].elevation = layer_elevation;
664  points[echo_idx].x = points[echo_idx].range * cos_azimuth * cos_elevation;
665  points[echo_idx].y = points[echo_idx].range * sin_azimuth * cos_elevation;
666  points[echo_idx].z = points[echo_idx].range * sin_elevation;
667  points[echo_idx].echoIdx = echo_idx;
668  points[echo_idx].groupIdx = groupIdx;
669  points[echo_idx].pointIdx = point_idx;
670  points[echo_idx].lidar_timestamp_microsec = lidar_timestamp_microsec;
671  points[echo_idx].reflectorbit |= (beam_property & 0x01); // reflector bit is set, if a reflector is detected on any number of echos
672  measurement_data.scandata[layer_idx].scanlines[echo_idx].points[point_idx] = points[echo_idx];
673  }
674  }
675  }
676  if (byte_cnt != num_bytes)
677  {
678  ROS_ERROR_STREAM("## ERROR CompactDataParser::ParseModuleMeasurementData(" << __LINE__ << "): byte_cnt=" << byte_cnt << ", num_bytes=" << num_bytes);
679  return false;
680  }
681  measurement_data.valid = true;
682  return measurement_data.valid;
683 }
684 
685 /*
686 * @brief Parses a scandata segment in compact format.
687 * @param[in] payload binary payload
688 * @param[in] bytes_received size of binary payload in bytes
689 * @param[out] segment_data parsed segment data (or 0 if not required)
690 * @param[out] payload_length_bytes parsed number of bytes
691 * @param[out] num_bytes_required min number of bytes required for successful parsing
692 * @param[in] azimuth_offset optional offset in case of additional coordinate transform (default: 0)
693 * @param[in] verbose > 0: print debug messages (default: 0)
694 */
696  uint32_t& payload_length_bytes, uint32_t& num_bytes_required , float azimuth_offset, int verbose)
697 {
698  // Read 32 byte compact data header
699  const uint32_t header_size_bytes = 32;
700  const std::vector<uint8_t> msg_start_seq = { 0x02, 0x02, 0x02, 0x02 };
702  if(bytes_received >= 32)
703  {
704  compact_header = sick_scansegment_xd::CompactDataParser::ParseHeader(payload + msg_start_seq.size());
705  if (verbose > 0)
706  {
707  ROS_INFO_STREAM("CompactDataParser::ParseSegment(): compact header = { " << compact_header.to_string() << " }");
708  }
709  }
710  else
711  {
712  if (verbose > 0)
713  {
714  ROS_INFO_STREAM("CompactDataParser::ParseSegment(): " << bytes_received << " bytes received (compact), at least " << (msg_start_seq.size() + 32) << " bytes required for compact data header");
715  }
716  payload_length_bytes = 0;
717  num_bytes_required = msg_start_seq.size() + 32;
718  return false;
719  }
720  if (segment_data)
721  {
722  segment_data->segmentHeader = compact_header;
723  segment_data->segmentModules.clear();
724  }
725  if (compact_header.commandId == 2) // imu data in compact format have always 64 byte payload, payload length is not coded in the header
726  {
727  payload_length_bytes = 60; // i.e. 64 byte excl. 4 byte CRC
728  num_bytes_required = 64; // 64 byte incl. 4 byte CRC
729  return compact_header.imudata.valid;
730  }
731  // Read compact data modules
732  uint32_t module_size = compact_header.sizeModule0;
733  uint32_t module_offset = header_size_bytes;
734  payload_length_bytes = header_size_bytes;
735  num_bytes_required = header_size_bytes;
736  bool success = true;
737  while (module_size > 0)
738  {
739  if (module_offset + module_size > bytes_received)
740  {
741  if (verbose > 0)
742  {
743  ROS_INFO_STREAM("CompactDataParser::ParseSegment(): " << bytes_received << " bytes received (compact), at least " << (module_offset + module_size) << " bytes required for module meta data");
744  }
745  payload_length_bytes = 0;
746  num_bytes_required = module_offset + module_size;
747  return false;
748  }
749  uint32_t module_metadata_size = 0;
750  sick_scansegment_xd::CompactModuleMetaData module_meta_data = sick_scansegment_xd::CompactDataParser::ParseModuleMetaData(payload + module_offset, module_size, compact_header.telegramVersion, module_metadata_size);
751  if (verbose > 0)
752  {
753  ROS_INFO_STREAM("CompactDataParser::ParseSegment(): module meta data = { " << module_meta_data.to_string() << " }");
754  }
755  if (module_meta_data.valid != true || module_size < module_metadata_size)
756  {
757  std::stringstream err_msg;
758  err_msg << "## ERROR CompactDataParser::ParseSegment(): " << bytes_received << " bytes received (compact), CompactDataParser::ParseModuleMetaData() failed";
759  print_error(err_msg.str(), __LINE__);
760  payload_length_bytes = 0;
761  num_bytes_required = module_offset + module_size;
762  return false;
763  }
764  if (segment_data)
765  {
767  segment_module.moduleMetadata = module_meta_data;
768  sick_scansegment_xd::CompactDataParser::ParseModuleMeasurementData(payload + module_offset + module_metadata_size, module_size - module_metadata_size, compact_header, module_meta_data, azimuth_offset, segment_module.moduleMeasurement);
769  if (verbose > 0)
770  {
771  ROS_INFO_STREAM("CompactDataParser::ParseSegment(): module measurement data = { " << segment_module.moduleMeasurement.to_string() << " }");
772  }
773  if (segment_module.moduleMeasurement.valid)
774  {
775  segment_data->segmentModules.push_back(segment_module);
776  }
777  else
778  {
779  ROS_ERROR_STREAM("## ERROR CompactDataParser::ParseSegment(): CompactDataParser::ParseModuleMeasurementData() failed, parsed data = { " << segment_module.moduleMeasurement.to_string() << " }");
780  success = false;
781  }
782  }
783  module_offset += module_size;
784  payload_length_bytes += module_size;
785  num_bytes_required = payload_length_bytes;
786  module_size = module_meta_data.NextModuleSize;
787  }
788  if (segment_data && verbose > 0)
789  {
790  ROS_INFO_STREAM("CompactDataParser: " << segment_data->segmentModules.size() << " modules");
791  for(int module_idx = 0; module_idx < segment_data->segmentModules.size(); module_idx++)
792  {
793  const std::vector<ScanSegmentParserOutput::Scangroup>& scandata = segment_data->segmentModules[module_idx].moduleMeasurement.scandata;
794  ROS_INFO_STREAM(" CompactDataParser (module " << module_idx << "): " << scandata.size() << " groups");
795  for(int group_idx = 0; group_idx < scandata.size(); group_idx++)
796  {
797  ROS_INFO_STREAM(" CompactDataParser (module " << module_idx << ", group " << group_idx << "): " << scandata[group_idx].scanlines.size() << " lines");
798  for(int line_idx = 0; line_idx < scandata[group_idx].scanlines.size(); line_idx++)
799  {
800  const std::vector<ScanSegmentParserOutput::LidarPoint>& points = scandata[group_idx].scanlines[line_idx].points;
801  ROS_INFO_STREAM(" CompactDataParser (module " << module_idx << ", group " << group_idx << ", line " << line_idx << "): " << points.size() << " points");
802  for(int point_idx = 0; point_idx < points.size(); point_idx++)
803  {
804  ROS_INFO_STREAM(" [" << points[point_idx].x << "," << points[point_idx].y << "," << points[point_idx].z << "," << points[point_idx].range << "," << points[point_idx].azimuth << "," << points[point_idx].elevation << "," << points[point_idx].groupIdx << "," << points[point_idx].pointIdx << "]");
805  }
806  }
807  }
808  }
809  }
810  return success;
811 }
812 
813 #define EXPORT_MEASUREMENT_AZIMUTH_ACCELERATION_CSV 0 // Measurement of IMU latency (development only): Export ticks (imu resp. lidar timestamp in micro seconds), imu acceleration and lidar max azimuth of board cube
814 #if EXPORT_MEASUREMENT_AZIMUTH_ACCELERATION_CSV
815 /*
816 * @brief Returns the max azimuth aperture (i.e. max - min azimuth) of all scan points within a cube, i.e. the max azimuth aperture of all points p
817 * with x_min <= p.x <= x_max && y_min <= p.y <= y_max && z_min <= p.z <= z_max && p.azimuth >= azimuth_min && p.azimuth <= azimuth_max
818 */
819 static bool getMaxAzimuthApertureWithinCube(const std::vector<sick_scansegment_xd::ScanSegmentParserOutput::Scangroup>& scandata,
820  float x_min, float x_max, float y_min, float y_max, float z_min, float z_max, float azimuth_min, float azimuth_max,
821  double& azimuth_aperture, uint64_t& timestamp_microsec)
822 {
823  bool success = false;
824  azimuth_aperture = 0;
825  timestamp_microsec = 0;
826  for(int group_idx = 0; group_idx < scandata.size(); group_idx++)
827  {
828  for(int line_idx = 0; line_idx < scandata[group_idx].scanlines.size(); line_idx++)
829  {
830  uint64_t timestampStart_microsec = (uint64_t)scandata[group_idx].timestampStart_sec * 1000000UL + (uint64_t)scandata[group_idx].timestampStart_nsec / 1000;
831  uint64_t timestampStop_microsec = (uint64_t)scandata[group_idx].timestampStop_sec * 1000000UL + (uint64_t)scandata[group_idx].timestampStop_nsec / 1000;
832  double point_azi_min = FLT_MAX, point_azi_max = -FLT_MAX;
833  for(int point_idx = 0; point_idx < scandata[group_idx].scanlines[line_idx].points.size(); point_idx++)
834  {
835  const sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint& p = scandata[group_idx].scanlines[line_idx].points[point_idx];
836  if (p.x >= x_min && p.x <= x_max && p.y >= y_min && p.y <= y_max && p.z >= z_min && p.z<= z_max && p.azimuth >= azimuth_min && p.azimuth <= azimuth_max)
837  {
838  point_azi_min = std::min(point_azi_min, (double)p.azimuth);
839  point_azi_max = std::max(point_azi_max, (double)p.azimuth);
840  }
841  }
842  if (point_azi_max > point_azi_min && azimuth_aperture < point_azi_max) // (point_azi_max - point_azi_min))
843  {
844  azimuth_aperture = point_azi_max; // - point_azi_min;
845  timestamp_microsec = timestampStart_microsec;
846  success = true;
847  }
848  }
849  }
850  return success;
851 }
852 #endif // EXPORT_MEASUREMENT_AZIMUTH_ACCELERATION_CSV
853 
854 /*
855 * @brief Parses a scandata segment in compact format.
856 * @param[in] parser_config configuration and settings for multiScan and picoScan parser
857 * @param[in] segment_data binary segment data in compact format
858 * @param[in] system_timestamp receive timestamp of segment_data (system time)
859 * @param[in] add_transform_xyz_rpy Optionally apply an additional transform to the cartesian pointcloud, default: "0,0,0,0,0,0" (i.e. no transform)
860 * @param[out] result scandata converted to ScanSegmentParserOutput
861 * @param[in] use_software_pll true (default): result timestamp from sensor ticks by software pll, false: result timestamp from msg receiving
862 * @param[in] verbose true: enable debug output, false: quiet mode
863 */
864 bool sick_scansegment_xd::CompactDataParser::Parse(const ScanSegmentParserConfig& parser_config, const std::vector<uint8_t>& payload, fifo_timestamp system_timestamp, sick_scan_xd::SickCloudTransform& add_transform_xyz_rpy,
865  ScanSegmentParserOutput& result, bool use_software_pll, bool verbose)
866 {
867  // Parse segment data
869  uint32_t payload_length_bytes = 0, num_bytes_required = 0;
870  if (!sick_scansegment_xd::CompactDataParser::ParseSegment(payload.data(), payload.size(), &segment_data, payload_length_bytes, num_bytes_required , add_transform_xyz_rpy.azimuthOffset()))
871  {
872  ROS_ERROR_STREAM("## ERROR CompactDataParser::Parse(): CompactDataParser::ParseSegment() failed, payload = " << sick_scansegment_xd::UdpReceiver::ToHexString(payload, payload.size()));
873  return false;
874  }
875  // Convert segment data to ScanSegmentParserOutput
876  sick_scansegment_xd::CompactDataHeader& segmentHeader = segment_data.segmentHeader;
877  result.scandata.clear();
878  result.imudata = segment_data.segmentHeader.imudata;
879  result.segmentIndex = 0;
880  result.telegramCnt = segmentHeader.telegramCounter;
881  for (int module_idx = 0; module_idx < segment_data.segmentModules.size(); module_idx++)
882  {
883  sick_scansegment_xd::CompactModuleMetaData& moduleMetadata = segment_data.segmentModules[module_idx].moduleMetadata;
884  sick_scansegment_xd::CompactModuleMeasurementData& moduleMeasurement = segment_data.segmentModules[module_idx].moduleMeasurement;
885  if (!moduleMeasurement.valid)
886  {
887  ROS_ERROR_STREAM("## ERROR CompactDataParser::Parse(): invalid moduleMeasurement (ignored)");
888  continue;
889  }
890  for (int measurement_idx = 0; measurement_idx < moduleMeasurement.scandata.size(); measurement_idx++)
891  {
892  ScanSegmentParserOutput::Scangroup& scandata = moduleMeasurement.scandata[measurement_idx];
893  // Apply optional range filter and optional transform
894  for(int line_idx = 0; line_idx < scandata.scanlines.size(); line_idx++)
895  {
896  std::vector<ScanSegmentParserOutput::LidarPoint>& points_in = scandata.scanlines[line_idx].points;
897  std::vector<ScanSegmentParserOutput::LidarPoint> points_out;
898  points_out.reserve(points_in.size());
899  for(int point_idx = 0; point_idx < points_in.size(); point_idx++)
900  {
901  add_transform_xyz_rpy.applyTransform(points_in[point_idx].x, points_in[point_idx].y, points_in[point_idx].z);
902  points_out.push_back(points_in[point_idx]);
903  }
904  scandata.scanlines[line_idx].points = points_out;
905  }
906  // result.scandata.push_back(scandata);
907  // Reorder lidar points by layer id (groupIdx) and echoIdx (identical to the msgpack scandata)
908  // result.scandata[groupIdx] = all scandata of layer <groupIdx> appended to one scanline
909  for(int line_idx = 0; line_idx < scandata.scanlines.size(); line_idx++)
910  {
911  std::vector<ScanSegmentParserOutput::LidarPoint>& points = scandata.scanlines[line_idx].points;
912  for(int point_idx = 0; point_idx < points.size(); point_idx++)
913  {
914  ScanSegmentParserOutput::LidarPoint& point = points[point_idx];
915  int groupIdx = point.groupIdx;
916  int echoIdx = point.echoIdx;
917  while(result.scandata.size() <= groupIdx)
918  {
919  result.scandata.push_back(ScanSegmentParserOutput::Scangroup());
920  }
921  if (result.scandata[groupIdx].scanlines.empty())
922  {
923  result.scandata[groupIdx].timestampStart_sec = scandata.timestampStart_sec;
924  result.scandata[groupIdx].timestampStart_nsec = scandata.timestampStart_nsec;
925  result.scandata[groupIdx].timestampStop_sec = scandata.timestampStop_sec;
926  result.scandata[groupIdx].timestampStop_nsec = scandata.timestampStop_nsec;
927  }
928  while(result.scandata[groupIdx].scanlines.size() <= echoIdx)
929  {
930  result.scandata[groupIdx].scanlines.push_back(ScanSegmentParserOutput::Scanline());
931  }
932  if (result.scandata[groupIdx].scanlines[echoIdx].points.empty())
933  {
934  result.scandata[groupIdx].scanlines[echoIdx].points.reserve(points.size());
935  }
936  result.scandata[groupIdx].scanlines[echoIdx].points.push_back(point);
937  }
938  }
939  }
940  if (module_idx == 0)
941  {
942  result.segmentIndex = moduleMetadata.SegmentCounter;
943  }
944  else if (result.segmentIndex != moduleMetadata.SegmentCounter)
945  {
946  ROS_WARN_STREAM("## WARNING CompactDataParser::Parse(): different SegmentCounter in module 0 and module " << module_idx << " currently not supported,"
947  << " scandata of segment " << moduleMetadata.SegmentCounter << " appended to segment " << result.segmentIndex);
948  }
949  }
950  if (result.scandata.empty() && !result.imudata.valid)
951  {
952  ROS_ERROR_STREAM("## ERROR CompactDataParser::Parse(): CompactDataParser::ParseSegment() failed (no scandata found)");
953  return false;
954  }
955  // Convert timestamp from sensor time to system time
956  uint64_t sensor_timeStamp = segmentHeader.timeStampTransmit; // Sensor timestamp in microseconds since 1.1.1970 00:00 UTC
957  if (result.imudata.valid)
958  sensor_timeStamp -= parser_config.imu_latency_microsec;
959  if (!result.scandata.empty())
960  sensor_timeStamp = (uint64_t)result.scandata[0].timestampStart_sec * 1000000UL + (uint64_t)result.scandata[0].timestampStart_nsec / 1000; // i.e. start of scan in microseconds
961  result.timestamp_sec = (sensor_timeStamp / 1000000);
962  result.timestamp_nsec= 1000 * (sensor_timeStamp % 1000000);
963  if (use_software_pll)
964  {
965  SoftwarePLL& software_pll = SoftwarePLL::instance();
966  int64_t systemtime_nanoseconds = system_timestamp.time_since_epoch().count();
967  uint32_t systemtime_sec = (uint32_t)(systemtime_nanoseconds / 1000000000); // seconds part of system timestamp
968  uint32_t systemtime_nsec = (uint32_t)(systemtime_nanoseconds % 1000000000); // nanoseconds part of system timestamp
969  software_pll.updatePLL(systemtime_sec, systemtime_nsec, sensor_timeStamp);
970  if (software_pll.IsInitialized())
971  {
972  uint32_t pll_sec = 0, pll_nsec = 0;
973  software_pll.getCorrectedTimeStamp(pll_sec, pll_nsec, sensor_timeStamp);
974  result.timestamp_sec = pll_sec;
975  result.timestamp_nsec = pll_nsec;
976  }
977  }
979 
980 #if EXPORT_MEASUREMENT_AZIMUTH_ACCELERATION_CSV // Measurement of IMU latency (development only): Export ticks (imu resp. lidar timestamp in micro seconds), imu acceleration and lidar max azimuth of board cube
981  ROS_INFO_STREAM("CompactDataParser::Parse(): header = " << segmentHeader.to_string() << ", system timestamp = " << result.timestamp);
982  if (result.imudata.valid) // Export imu acceleration
983  {
984  std::ofstream csv_ostream("/tmp/imu_latency.csv", std::ios::app);
985  csv_ostream << segmentHeader.timeStampTransmit << ";" << ";" << std::fixed << std::setprecision(3) << (result.imudata.acceleration_z) << "\n";
986  }
987  else // Export lidar azimuth of calibration board
988  {
989  uint64_t timestamp_microsec_azi = 0;
990  double azimuth_aperture = 0;
991  if (getMaxAzimuthApertureWithinCube(result.scandata, 0.5f, 1.5f, -1.0f, +1.0f, -1.0f, +1.0f, -M_PI, +M_PI, azimuth_aperture, timestamp_microsec_azi))
992  {
993  std::ofstream csv_ostream("/tmp/imu_latency.csv", std::ios::app);
994  csv_ostream << timestamp_microsec_azi << ";" << std::fixed << std::setprecision(3) << (azimuth_aperture * 180 / M_PI) << ";" << "\n";
995  }
996  }
997 #endif // EXPORT_MEASUREMENT_AZIMUTH_ACCELERATION_CSV
998 
999  return true;
1000 }
sick_scansegment_xd::CompactModuleMetaData::NumberOfEchosPerBeam
uint32_t NumberOfEchosPerBeam
Definition: compact_parser.h:96
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::range
float range
Definition: scansegment_parser_output.h:139
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::elevation
float elevation
Definition: scansegment_parser_output.h:141
sick_scansegment_xd::ScanSegmentParserOutput::imudata
CompactImuData imudata
Definition: scansegment_parser_output.h:180
sick_scansegment_xd::Timestamp
std::string Timestamp(uint32_t sec, uint32_t nsec)
Definition: scansegment_parser_output.cpp:100
sick_scansegment_xd::CompactModuleMetaData::FrameNumber
uint64_t FrameNumber
Definition: compact_parser.h:92
SoftwarePLL
class SoftwarePLL implements synchronisation between ticks and timestamp. See https://github....
Definition: softwarePLL.h:21
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::echoIdx
int echoIdx
Definition: scansegment_parser_output.h:143
sick_scansegment_xd::CompactImuData::acceleration_z
float acceleration_z
Definition: scansegment_parser_output.h:107
COMPACT_8BYTE_UNION::f64_val
double f64_val[1]
Definition: compact_parser.cpp:75
SoftwarePLL::updatePLL
bool updatePLL(uint32_t sec, uint32_t nanoSec, uint32_t curtick)
Definition: softwarePLL.cpp:191
sick_scansegment_xd::CompactModuleMetaData::SenderId
uint32_t SenderId
Definition: compact_parser.h:93
sick_scansegment_xd::CompactDataHeader::imudata
CompactImuData imudata
Definition: compact_parser.h:80
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::z
float z
Definition: scansegment_parser_output.h:137
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::azimuth
float azimuth
Definition: scansegment_parser_output.h:140
sick_scansegment_xd::CompactDataHeader::timeStampTransmit
uint64_t timeStampTransmit
Definition: compact_parser.h:77
sick_scansegment_xd::CompactDataParser::SetLayerElevationTable
static void SetLayerElevationTable(const std::vector< int > &layer_elevation_table_mdeg)
Definition: compact_parser.cpp:430
sick_scan_xd::SickCloudTransform
Definition: sick_cloud_transform.h:85
sick_scansegment_xd::CompactImuData::orientation_z
float orientation_z
Definition: scansegment_parser_output.h:114
sick_scansegment_xd::ScanSegmentParserOutput::timestamp_nsec
uint32_t timestamp_nsec
Definition: scansegment_parser_output.h:187
sick_scansegment_xd::UdpReceiver::ToHexString
static std::string ToHexString(const std::vector< uint8_t > &payload, size_t bytes_received)
Definition: udp_receiver.cpp:383
sick_scansegment_xd::CompactDataHeader::telegramCounter
uint64_t telegramCounter
Definition: compact_parser.h:76
readUnsigned
static T readUnsigned(const uint8_t *scandata, uint32_t *byte_cnt)
Definition: compact_parser.cpp:84
sick_scansegment_xd::CompactDataHeader::sizeModule0
uint32_t sizeModule0
Definition: compact_parser.h:79
sick_scansegment_xd::CompactModuleMetaData::TimeStampStop
std::vector< uint64_t > TimeStampStop
Definition: compact_parser.h:98
sick_scansegment_xd::CompactModuleMetaData::DistanceScalingFactor
float DistanceScalingFactor
Definition: compact_parser.h:102
description
description
sick_scansegment_xd::CompactModuleMetaData::NextModuleSize
uint32_t NextModuleSize
Definition: compact_parser.h:103
COMPACT_8BYTE_UNION::u64_bytes
uint64_t u64_bytes[1]
Definition: compact_parser.cpp:73
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::pointIdx
int pointIdx
Definition: scansegment_parser_output.h:144
SoftwarePLL::instance
static SoftwarePLL & instance()
Definition: softwarePLL.h:24
sick_scansegment_xd::CompactImuData::angular_velocity_x
float angular_velocity_x
Definition: scansegment_parser_output.h:108
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::groupIdx
int groupIdx
Definition: scansegment_parser_output.h:142
sick_scansegment_xd::CompactModuleMetaData::reserved
uint8_t reserved
Definition: compact_parser.h:107
sick_scansegment_xd::ScanSegmentParserOutput::Scangroup::timestampStop_sec
uint32_t timestampStop_sec
Definition: scansegment_parser_output.h:167
sick_scansegment_xd::CompactModuleMeasurementData::scandata
std::vector< ScanSegmentParserOutput::Scangroup > scandata
Definition: compact_parser.h:118
sick_scansegment_xd::CompactModuleMetaData::NumberOfBeamsPerScan
uint32_t NumberOfBeamsPerScan
Definition: compact_parser.h:95
sick_scansegment_xd::CompactModuleData
Definition: compact_parser.h:126
ReadBeamAzimOrder
enum ReadBeamAzimOrderEnum ReadBeamAzimOrder
sick_scansegment_xd::CompactSegmentData::segmentHeader
CompactDataHeader segmentHeader
Definition: compact_parser.h:139
SoftwarePLL::getCorrectedTimeStamp
bool getCorrectedTimeStamp(uint32_t &sec, uint32_t &nanoSec, uint32_t tick)
Definition: softwarePLL.cpp:226
udp_receiver.h
sick_scansegment_xd::CompactImuData::to_string
std::string to_string() const
Definition: compact_parser.cpp:150
COMPACT_4BYTE_UNION::u32_bytes
uint32_t u32_bytes[1]
Definition: compact_parser.cpp:64
endOfBuffer
static bool endOfBuffer(uint32_t byte_cnt, size_t bytes_to_read, uint32_t num_bytes)
Definition: compact_parser.cpp:118
READ_BEAM_AZIM
@ READ_BEAM_AZIM
Definition: compact_parser.cpp:80
sick_scansegment_xd::ScanSegmentParserOutput::Scangroup::timestampStart_sec
uint32_t timestampStart_sec
Definition: scansegment_parser_output.h:165
sick_scansegment_xd::CompactSegmentData
Definition: compact_parser.h:136
COMPACT_8BYTE_UNION
Definition: compact_parser.cpp:68
f
f
sick_scansegment_xd::CompactImuData::angular_velocity_y
float angular_velocity_y
Definition: scansegment_parser_output.h:109
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::CompactImuData::acceleration_x
float acceleration_x
Definition: scansegment_parser_output.h:105
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::y
float y
Definition: scansegment_parser_output.h:136
ROS::now
ROS::Time now(void)
Definition: ros_wrapper.cpp:116
sick_scansegment_xd::CompactModuleMetaData::DataContentBeams
uint8_t DataContentBeams
Definition: compact_parser.h:106
sick_scansegment_xd::CompactModuleMetaData::to_string
std::string to_string() const
Definition: compact_parser.cpp:190
compact_parser.h
readFloat32
static float readFloat32(const uint8_t *scandata, uint32_t *byte_cnt)
Definition: compact_parser.cpp:101
pcap_json_converter.payload
string payload
Definition: pcap_json_converter.py:130
COMPACT_8BYTE_UNION::u8_bytes
uint8_t u8_bytes[8]
Definition: compact_parser.cpp:70
sick_scansegment_xd::ScanSegmentParserConfig::imu_latency_microsec
int imu_latency_microsec
Definition: scansegment_parser_output.h:94
sick_scansegment_xd::CompactSegmentData::segmentModules
std::vector< CompactModuleData > segmentModules
Definition: compact_parser.h:140
s_layer_elevation_table_mdeg
static std::vector< int > s_layer_elevation_table_mdeg
Definition: compact_parser.cpp:424
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
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::CompactDataParser::ParseModuleMeasurementData
static bool ParseModuleMeasurementData(const uint8_t *payload, uint32_t num_bytes, const sick_scansegment_xd::CompactDataHeader &compact_header, const sick_scansegment_xd::CompactModuleMetaData &meta_data, float azimuth_offset, sick_scansegment_xd::CompactModuleMeasurementData &measurement_data)
Definition: compact_parser.cpp:501
sick_scansegment_xd::CompactModuleMeasurementData
Definition: compact_parser.h:115
sick_scansegment_xd::CompactModuleMetaData::valid
bool valid
Definition: compact_parser.h:108
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint
Definition: scansegment_parser_output.h:129
sick_scansegment_xd::CompactDataHeader::telegramVersion
uint32_t telegramVersion
Definition: compact_parser.h:78
COMPACT_4BYTE_UNION
Definition: compact_parser.cpp:60
sick_scansegment_xd::CompactModuleMetaData::ThetaStop
std::vector< float > ThetaStop
Definition: compact_parser.h:101
sick_scansegment_xd::ScanSegmentParserOutput::Scangroup::timestampStop_nsec
uint32_t timestampStop_nsec
Definition: scansegment_parser_output.h:168
ROS_DEBUG_STREAM
#define ROS_DEBUG_STREAM(args)
Definition: sick_scan_logging.h:113
sick_scansegment_xd::CompactModuleMetaData::ThetaStart
std::vector< float > ThetaStart
Definition: compact_parser.h:100
sick_scansegment_xd::CompactDataParser::Parse
static bool Parse(const ScanSegmentParserConfig &parser_config, const std::vector< uint8_t > &payload, fifo_timestamp system_timestamp, sick_scan_xd::SickCloudTransform &add_transform_xyz_rpy, ScanSegmentParserOutput &result, bool use_software_pll=true, bool verbose=false)
Definition: compact_parser.cpp:864
fifo_timestamp
std::chrono::time_point< std::chrono::system_clock > fifo_timestamp
Definition: fifo.h:68
sick_scansegment_xd::CompactModuleMetaData::SegmentCounter
uint64_t SegmentCounter
Definition: compact_parser.h:91
sick_scansegment_xd::ScanSegmentParserOutput::LidarPoint::i
float i
Definition: scansegment_parser_output.h:138
sick_scansegment_xd::CompactModuleData::moduleMetadata
CompactModuleMetaData moduleMetadata
Definition: compact_parser.h:129
COMPACT_4BYTE_UNION::u16_bytes
uint16_t u16_bytes[2]
Definition: compact_parser.cpp:63
sick_scansegment_xd::CompactImuData::angular_velocity_z
float angular_velocity_z
Definition: scansegment_parser_output.h:110
CHECK_MODULE_SIZE
#define CHECK_MODULE_SIZE(metadata, byte_required, byte_cnt, bytes_to_read, module_size, name, line_number)
Definition: compact_parser.cpp:139
sick_scansegment_xd::CompactDataParser::ParseSegment
static bool ParseSegment(const uint8_t *payload, size_t bytes_received, sick_scansegment_xd::CompactSegmentData *segment_data, uint32_t &payload_length_bytes, uint32_t &num_bytes_required, float azimuth_offset=0, int verbose=0)
Definition: compact_parser.cpp:695
COMPACT_8BYTE_UNION::u32_bytes
uint32_t u32_bytes[2]
Definition: compact_parser.cpp:72
sick_scansegment_xd::CompactModuleMeasurementData::valid
bool valid
Definition: compact_parser.h:119
READ_BEAM_PROP
@ READ_BEAM_PROP
Definition: compact_parser.cpp:81
sick_scansegment_xd::ScanSegmentParserOutput::Scangroup::scanlines
std::vector< Scanline > scanlines
Definition: scansegment_parser_output.h:169
sick_scansegment_xd::CompactModuleMeasurementData::to_string
std::string to_string() const
Definition: compact_parser.cpp:224
sick_scansegment_xd::CompactModuleMetaData::TimeStampStart
std::vector< uint64_t > TimeStampStart
Definition: compact_parser.h:97
sick_scansegment_xd::CompactModuleMetaData::DataContentEchos
uint8_t DataContentEchos
Definition: compact_parser.h:105
COMPACT_8BYTE_UNION::f32_val
float f32_val[2]
Definition: compact_parser.cpp:74
sick_scansegment_xd::ScanSegmentParserOutput::timestamp_sec
uint32_t timestamp_sec
Definition: scansegment_parser_output.h:186
COMPACT_4BYTE_UNION::f32_val
float f32_val[1]
Definition: compact_parser.cpp:65
sick_scansegment_xd::CompactDataParser::ParseHeader
static CompactDataHeader ParseHeader(const uint8_t *scandata)
Definition: compact_parser.cpp:254
sick_scansegment_xd::CompactDataParser::GetElevationDegFromLayerIdx
static float GetElevationDegFromLayerIdx(int layer_idx)
Definition: compact_parser.cpp:483
SoftwarePLL::IsInitialized
bool IsInitialized() const
Definition: softwarePLL.h:47
sick_scansegment_xd::CompactDataHeader::commandId
uint32_t commandId
Definition: compact_parser.h:75
COMPACT_8BYTE_UNION::u16_bytes
uint16_t u16_bytes[4]
Definition: compact_parser.cpp:71
sick_scansegment_xd::CompactImuData::orientation_x
float orientation_x
Definition: scansegment_parser_output.h:112
sick_scansegment_xd::CompactModuleMetaData::Phi
std::vector< float > Phi
Definition: compact_parser.h:99
config.h
sick_scansegment_xd::CompactDataHeader
Definition: compact_parser.h:72
sick_scansegment_xd::ScanSegmentParserOutput::scandata
std::vector< Scangroup > scandata
Definition: scansegment_parser_output.h:175
sick_scansegment_xd::CompactDataParser::ParseModuleMetaData
static CompactModuleMetaData ParseModuleMetaData(const uint8_t *scandata, uint32_t module_size, uint32_t telegramVersion, uint32_t &module_metadata_size)
Definition: compact_parser.cpp:320
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
sick_scansegment_xd::CompactImuData::acceleration_y
float acceleration_y
Definition: scansegment_parser_output.h:106
sick_scansegment_xd::CompactImuData::orientation_w
float orientation_w
Definition: scansegment_parser_output.h:111
sick_scansegment_xd::ScanSegmentParserConfig
Definition: scansegment_parser_output.h:91
COMPACT_4BYTE_UNION::u8_bytes
uint8_t u8_bytes[4]
Definition: compact_parser.cpp:62
roswrap::message_traits::header
std_msgs::Header * header(M &m)
returns Header<M>::pointer(m);
Definition: message_traits.h:281
sick_scansegment_xd::CompactModuleMetaData
Definition: compact_parser.h:88
print_error
static void print_error(const std::string &err_msg, int line_number, double print_rate=1)
Definition: compact_parser.cpp:123
sick_scansegment_xd::CompactModuleMetaData::NumberOfLinesInModule
uint32_t NumberOfLinesInModule
Definition: compact_parser.h:94
sick_scansegment_xd::ScanSegmentParserOutput::Scangroup::timestampStart_nsec
uint32_t timestampStart_nsec
Definition: scansegment_parser_output.h:166
sick_scansegment_xd::CompactModuleMetaData::Availability
uint8_t Availability
Definition: compact_parser.h:104
sick_scansegment_xd::CompactDataHeader::to_string
std::string to_string() const
Definition: compact_parser.cpp:171
sick_scan_xd::SickCloudTransform::azimuthOffset
float azimuthOffset(void) const
Definition: sick_cloud_transform.h:133
sick_scansegment_xd::CompactImuData::orientation_y
float orientation_y
Definition: scansegment_parser_output.h:113
multiscan_perftest_player.num_echos
int num_echos
Definition: multiscan_perftest_player.py:22
sick_scansegment_xd::CompactImuData::valid
bool valid
Definition: scansegment_parser_output.h:104
sick_scansegment_xd::ScanSegmentParserOutput::Scangroup
Definition: scansegment_parser_output.h:161
sick_scansegment_xd::CompactModuleData::moduleMeasurement
CompactModuleMeasurementData moduleMeasurement
Definition: compact_parser.h:130
sick_scansegment_xd::CompactDataParser::GetLayerIDfromElevation
static int GetLayerIDfromElevation(float layer_elevation_rad)
Definition: compact_parser.cpp:442
sick_scansegment_xd::ScanSegmentParserOutput::timestamp
std::string timestamp
Definition: scansegment_parser_output.h:185
ReadBeamAzimOrderEnum
ReadBeamAzimOrderEnum
Definition: compact_parser.cpp:78


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