-
Notifications
You must be signed in to change notification settings - Fork 39
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: port autoware_point_types from autoware.universe (#151)
* feat: port autoware_point_types from universe Signed-off-by: liu cui <[email protected]> * chore: remove change log and reset version to 0.0.0 Signed-off-by: liu cui <[email protected]> * add myself as maintainer Signed-off-by: liu cui <[email protected]> * docs: finish README.md Signed-off-by: liu cui <[email protected]> * style(pre-commit): autofix * fix: fix pre-commit.ci error Signed-off-by: liu cui <[email protected]> --------- Signed-off-by: liu cui <[email protected]> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Yutaka Kondo <[email protected]>
- Loading branch information
1 parent
d473117
commit 871106e
Showing
5 changed files
with
382 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
cmake_minimum_required(VERSION 3.14) | ||
project(autoware_point_types) | ||
|
||
find_package(autoware_cmake REQUIRED) | ||
autoware_package() | ||
|
||
include_directories( | ||
include | ||
SYSTEM | ||
${PCL_INCLUDE_DIRS} | ||
) | ||
|
||
if(BUILD_TESTING) | ||
ament_add_ros_isolated_gtest(test_autoware_point_types | ||
test/test_point_types.cpp | ||
) | ||
target_include_directories(test_autoware_point_types | ||
PRIVATE include | ||
) | ||
ament_target_dependencies(test_autoware_point_types | ||
point_cloud_msg_wrapper | ||
) | ||
endif() | ||
|
||
ament_auto_package() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,68 @@ | ||
# Autoware Point Types | ||
|
||
## Overview | ||
|
||
This package provides a variety of structures to represent different types of point cloud data, mainly used for point cloud processing and analysis. | ||
|
||
## Design | ||
|
||
### Point cloud data type definition | ||
|
||
`autoware_point_types` defines multiple structures (such as PointXYZI, PointXYZIRC, PointXYZIRADRT, PointXYZIRCAEDT), each structure contains different attributes to adapt to different application scenarios. | ||
|
||
- `autoware::point_types::PointXYZI`: Point type with intensity information. | ||
- `autoware::point_types::PointXYZIRC`: Extended PointXYZI, adds return_type and channel information. | ||
- `autoware::point_types::PointXYZIRADRT`: Extended PointXYZI, adds ring, azimuth, distance, return_type and time_stamp information. | ||
- `autoware::point_types::PointXYZIRCAEDT`: Similar to PointXYZIRADRT, but adds elevation information and uses `std::uint32_t` as the data type for time_stamp. | ||
|
||
### Operator overload | ||
|
||
Each structure overloads the `==` operator, allowing users to easily compare whether two points are equal, which is very useful for deduplication and matching of point cloud data. | ||
|
||
### Field generators | ||
|
||
The field generator is implemented using macro definitions and std::tuple, which simplifies the serialization and deserialization process of point cloud messages and improves the reusability and readability of the code. | ||
|
||
### Registration mechanism | ||
|
||
Register custom point cloud structures into the PCL library through the macro `POINT_CLOUD_REGISTER_POINT_STRUCT`, so that these structures can be directly integrated with other functions of the PCL library. | ||
|
||
## Usage | ||
|
||
- Create a point cloud object of PointXYZIRC type | ||
|
||
```cpp | ||
#include "autoware/point_types/types.hpp" | ||
|
||
int main(){ | ||
pcl::PointCloud<autoware::point_types::PointXYZIRC>::Ptr cloud(new pcl::PointCloud<autoware::point_types::PointXYZIRC>()); | ||
|
||
for (int i = 0; i < 5; ++i) { | ||
autoware::point_types::PointXYZIRC point; | ||
point.x = static_cast<float>(i * 0.1); | ||
point.y = static_cast<float>(i * 0.2); | ||
point.z = static_cast<float>(i * 0.3); | ||
point.intensity = static_cast<std::uint8_t>(i * 10); | ||
point.return_type = autoware::point_types::ReturnType::SINGLE_STRONGEST; | ||
point.channel = static_cast<std::uint16_t>(i); | ||
|
||
cloud->points.push_back(point); | ||
} | ||
cloud->width = cloud->points.size(); | ||
cloud->height = 1; | ||
|
||
return 0; | ||
} | ||
``` | ||
|
||
- Convert ROS message to point cloud of PointXYZIRC type | ||
|
||
```cpp | ||
ExampleNode::points_callback(const PointCloud2::ConstSharedPtr & points_msg_ptr) | ||
{ | ||
pcl::PointCloud<autoware::point_types::PointXYZIRC>::Ptr points_ptr( | ||
new pcl::PointCloud<autoware::point_types::PointXYZIRC>); | ||
|
||
pcl::fromROSMsg(*points_msg_ptr, *points_ptr); | ||
} | ||
``` |
188 changes: 188 additions & 0 deletions
188
common/autoware_point_types/include/autoware/point_types/types.hpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,188 @@ | ||
// Copyright 2021 Tier IV, Inc. | ||
// | ||
// Licensed under the Apache License, Version 2.0 (the "License"); | ||
// you may not use this file except in compliance with the License. | ||
// You may obtain a copy of the License at | ||
// | ||
// http://www.apache.org/licenses/LICENSE-2.0 | ||
// | ||
// Unless required by applicable law or agreed to in writing, software | ||
// distributed under the License is distributed on an "AS IS" BASIS, | ||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
// See the License for the specific language governing permissions and | ||
// limitations under the License. | ||
|
||
#ifndef AUTOWARE__POINT_TYPES__TYPES_HPP_ | ||
#define AUTOWARE__POINT_TYPES__TYPES_HPP_ | ||
|
||
#include <point_cloud_msg_wrapper/point_cloud_msg_wrapper.hpp> | ||
|
||
#include <pcl/point_types.h> | ||
|
||
#include <cmath> | ||
#include <tuple> | ||
|
||
namespace autoware::point_types | ||
{ | ||
template <class T> | ||
bool float_eq(const T a, const T b, const T eps = 10e-6) | ||
{ | ||
return std::fabs(a - b) < eps; | ||
} | ||
|
||
struct PointXYZI | ||
{ | ||
float x{0.0F}; | ||
float y{0.0F}; | ||
float z{0.0F}; | ||
float intensity{0.0F}; | ||
friend bool operator==(const PointXYZI & p1, const PointXYZI & p2) noexcept | ||
{ | ||
return float_eq<float>(p1.x, p2.x) && float_eq<float>(p1.y, p2.y) && | ||
float_eq<float>(p1.z, p2.z) && float_eq<float>(p1.intensity, p2.intensity); | ||
} | ||
}; | ||
|
||
enum ReturnType : uint8_t { | ||
INVALID = 0, | ||
SINGLE_STRONGEST, | ||
SINGLE_LAST, | ||
DUAL_STRONGEST_FIRST, | ||
DUAL_STRONGEST_LAST, | ||
DUAL_WEAK_FIRST, | ||
DUAL_WEAK_LAST, | ||
DUAL_ONLY, | ||
}; | ||
|
||
struct PointXYZIRC | ||
{ | ||
float x{0.0F}; | ||
float y{0.0F}; | ||
float z{0.0F}; | ||
std::uint8_t intensity{0U}; | ||
std::uint8_t return_type{0U}; | ||
std::uint16_t channel{0U}; | ||
|
||
friend bool operator==(const PointXYZIRC & p1, const PointXYZIRC & p2) noexcept | ||
{ | ||
return float_eq<float>(p1.x, p2.x) && float_eq<float>(p1.y, p2.y) && | ||
float_eq<float>(p1.z, p2.z) && p1.intensity == p2.intensity && | ||
p1.return_type == p2.return_type && p1.channel == p2.channel; | ||
} | ||
}; | ||
|
||
struct PointXYZIRADRT | ||
{ | ||
float x{0.0F}; | ||
float y{0.0F}; | ||
float z{0.0F}; | ||
float intensity{0.0F}; | ||
uint16_t ring{0U}; | ||
float azimuth{0.0F}; | ||
float distance{0.0F}; | ||
uint8_t return_type{0U}; | ||
double time_stamp{0.0}; | ||
friend bool operator==(const PointXYZIRADRT & p1, const PointXYZIRADRT & p2) noexcept | ||
{ | ||
return float_eq<float>(p1.x, p2.x) && float_eq<float>(p1.y, p2.y) && | ||
float_eq<float>(p1.z, p2.z) && float_eq<float>(p1.intensity, p2.intensity) && | ||
p1.ring == p2.ring && float_eq<float>(p1.azimuth, p2.azimuth) && | ||
float_eq<float>(p1.distance, p2.distance) && p1.return_type == p2.return_type && | ||
float_eq<float>(p1.time_stamp, p2.time_stamp); | ||
} | ||
}; | ||
|
||
struct PointXYZIRCAEDT | ||
{ | ||
float x{0.0F}; | ||
float y{0.0F}; | ||
float z{0.0F}; | ||
std::uint8_t intensity{0U}; | ||
std::uint8_t return_type{0U}; | ||
std::uint16_t channel{0U}; | ||
float azimuth{0.0F}; | ||
float elevation{0.0F}; | ||
float distance{0.0F}; | ||
std::uint32_t time_stamp{0U}; | ||
|
||
friend bool operator==(const PointXYZIRCAEDT & p1, const PointXYZIRCAEDT & p2) noexcept | ||
{ | ||
return float_eq<float>(p1.x, p2.x) && float_eq<float>(p1.y, p2.y) && | ||
float_eq<float>(p1.z, p2.z) && p1.intensity == p2.intensity && | ||
p1.return_type == p2.return_type && p1.channel == p2.channel && | ||
float_eq<float>(p1.azimuth, p2.azimuth) && float_eq<float>(p1.distance, p2.distance) && | ||
p1.time_stamp == p2.time_stamp; | ||
} | ||
}; | ||
|
||
enum class PointXYZIIndex { X, Y, Z, Intensity }; | ||
enum class PointXYZIRCIndex { X, Y, Z, Intensity, ReturnType, Channel }; | ||
enum class PointXYZIRADRTIndex { | ||
X, | ||
Y, | ||
Z, | ||
Intensity, | ||
Ring, | ||
Azimuth, | ||
Distance, | ||
ReturnType, | ||
TimeStamp | ||
}; | ||
enum class PointXYZIRCAEDTIndex { | ||
X, | ||
Y, | ||
Z, | ||
Intensity, | ||
ReturnType, | ||
Channel, | ||
Azimuth, | ||
Elevation, | ||
Distance, | ||
TimeStamp | ||
}; | ||
|
||
LIDAR_UTILS__DEFINE_FIELD_GENERATOR_FOR_MEMBER(azimuth); | ||
LIDAR_UTILS__DEFINE_FIELD_GENERATOR_FOR_MEMBER(elevation); | ||
LIDAR_UTILS__DEFINE_FIELD_GENERATOR_FOR_MEMBER(distance); | ||
LIDAR_UTILS__DEFINE_FIELD_GENERATOR_FOR_MEMBER(return_type); | ||
LIDAR_UTILS__DEFINE_FIELD_GENERATOR_FOR_MEMBER(time_stamp); | ||
|
||
LIDAR_UTILS__DEFINE_FIELD_GENERATOR_FOR_MEMBER(channel); | ||
|
||
using PointXYZIRCGenerator = std::tuple< | ||
point_cloud_msg_wrapper::field_x_generator, point_cloud_msg_wrapper::field_y_generator, | ||
point_cloud_msg_wrapper::field_z_generator, point_cloud_msg_wrapper::field_intensity_generator, | ||
field_return_type_generator, field_channel_generator>; | ||
|
||
using PointXYZIRADRTGenerator = std::tuple< | ||
point_cloud_msg_wrapper::field_x_generator, point_cloud_msg_wrapper::field_y_generator, | ||
point_cloud_msg_wrapper::field_z_generator, point_cloud_msg_wrapper::field_intensity_generator, | ||
point_cloud_msg_wrapper::field_ring_generator, field_azimuth_generator, field_distance_generator, | ||
field_return_type_generator, field_time_stamp_generator>; | ||
|
||
using PointXYZIRCAEDTGenerator = std::tuple< | ||
point_cloud_msg_wrapper::field_x_generator, point_cloud_msg_wrapper::field_y_generator, | ||
point_cloud_msg_wrapper::field_z_generator, point_cloud_msg_wrapper::field_intensity_generator, | ||
field_return_type_generator, field_channel_generator, field_azimuth_generator, | ||
field_elevation_generator, field_distance_generator, field_time_stamp_generator>; | ||
|
||
} // namespace autoware::point_types | ||
|
||
POINT_CLOUD_REGISTER_POINT_STRUCT( | ||
autoware::point_types::PointXYZIRC, | ||
(float, x, x)(float, y, y)(float, z, z)(std::uint8_t, intensity, intensity)( | ||
std::uint8_t, return_type, return_type)(std::uint16_t, channel, channel)) | ||
|
||
POINT_CLOUD_REGISTER_POINT_STRUCT( | ||
autoware::point_types::PointXYZIRADRT, | ||
(float, x, x)(float, y, y)(float, z, z)(float, intensity, intensity)(std::uint16_t, ring, ring)( | ||
float, azimuth, azimuth)(float, distance, distance)(std::uint8_t, return_type, return_type)( | ||
double, time_stamp, time_stamp)) | ||
|
||
POINT_CLOUD_REGISTER_POINT_STRUCT( | ||
autoware::point_types::PointXYZIRCAEDT, | ||
(float, x, x)(float, y, y)(float, z, z)(std::uint8_t, intensity, intensity)( | ||
std::uint8_t, return_type, | ||
return_type)(std::uint16_t, channel, channel)(float, azimuth, azimuth)( | ||
float, elevation, elevation)(float, distance, distance)(std::uint32_t, time_stamp, time_stamp)) | ||
#endif // AUTOWARE__POINT_TYPES__TYPES_HPP_ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
<?xml version="1.0"?> | ||
<?xml-model href="http://download.ros.org/schema/package_format3.xsd" schematypens="http://www.w3.org/2001/XMLSchema"?> | ||
<package format="3"> | ||
<name>autoware_point_types</name> | ||
<version>0.0.0</version> | ||
<description>The point types definition to use point_cloud_msg_wrapper</description> | ||
<maintainer email="[email protected]">David Wong</maintainer> | ||
<maintainer email="[email protected]">Max Schmeller</maintainer> | ||
<maintainer email="[email protected]">Cynthia Liu</maintainer> | ||
<license>Apache License 2.0</license> | ||
|
||
<buildtool_depend>ament_cmake_core</buildtool_depend> | ||
<buildtool_depend>ament_cmake_export_dependencies</buildtool_depend> | ||
<buildtool_depend>ament_cmake_test</buildtool_depend> | ||
<buildtool_depend>autoware_cmake</buildtool_depend> | ||
|
||
<buildtool_export_depend>ament_cmake_core</buildtool_export_depend> | ||
<buildtool_export_depend>ament_cmake_test</buildtool_export_depend> | ||
|
||
<depend>ament_cmake_copyright</depend> | ||
<depend>ament_cmake_cppcheck</depend> | ||
<depend>ament_cmake_lint_cmake</depend> | ||
<depend>ament_cmake_xmllint</depend> | ||
<depend>pcl_ros</depend> | ||
<depend>point_cloud_msg_wrapper</depend> | ||
|
||
<test_depend>ament_cmake_ros</test_depend> | ||
<test_depend>ament_lint_auto</test_depend> | ||
<test_depend>autoware_lint_common</test_depend> | ||
<test_depend>point_cloud_msg_wrapper</test_depend> | ||
|
||
<export> | ||
<build_type>ament_cmake</build_type> | ||
</export> | ||
</package> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,66 @@ | ||
// Copyright 2021 Tier IV, Inc. | ||
// | ||
// Licensed under the Apache License, Version 2.0 (the "License"); | ||
// you may not use this file except in compliance with the License. | ||
// You may obtain a copy of the License at | ||
// | ||
// http://www.apache.org/licenses/LICENSE-2.0 | ||
// | ||
// Unless required by applicable law or agreed to in writing, software | ||
// distributed under the License is distributed on an "AS IS" BASIS, | ||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
// See the License for the specific language governing permissions and | ||
// limitations under the License. | ||
|
||
#include "autoware/point_types/types.hpp" | ||
|
||
#include <gtest/gtest.h> | ||
|
||
#include <limits> | ||
|
||
TEST(PointEquality, PointXYZI) | ||
{ | ||
using autoware::point_types::PointXYZI; | ||
|
||
PointXYZI pt0{0, 1, 2, 3}; | ||
PointXYZI pt1{0, 1, 2, 3}; | ||
EXPECT_EQ(pt0, pt1); | ||
} | ||
|
||
TEST(PointEquality, PointXYZIRADRT) | ||
{ | ||
using autoware::point_types::PointXYZIRADRT; | ||
|
||
PointXYZIRADRT pt0{0, 1, 2, 3, 4, 5, 6, 7, 8}; | ||
PointXYZIRADRT pt1{0, 1, 2, 3, 4, 5, 6, 7, 8}; | ||
EXPECT_EQ(pt0, pt1); | ||
} | ||
|
||
TEST(PointEquality, PointXYZIRCAEDT) | ||
{ | ||
using autoware::point_types::PointXYZIRCAEDT; | ||
|
||
PointXYZIRCAEDT pt0{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}; | ||
PointXYZIRCAEDT pt1{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}; | ||
EXPECT_EQ(pt0, pt1); | ||
} | ||
|
||
TEST(PointEquality, FloatEq) | ||
{ | ||
// test template | ||
EXPECT_TRUE(autoware::point_types::float_eq<float>(1, 1)); | ||
EXPECT_TRUE(autoware::point_types::float_eq<double>(1, 1)); | ||
|
||
// test floating point error | ||
EXPECT_TRUE(autoware::point_types::float_eq<float>(1, 1 + std::numeric_limits<float>::epsilon())); | ||
|
||
// test difference of sign | ||
EXPECT_FALSE(autoware::point_types::float_eq<float>(2, -2)); | ||
EXPECT_FALSE(autoware::point_types::float_eq<float>(-2, 2)); | ||
|
||
// small value difference | ||
EXPECT_FALSE(autoware::point_types::float_eq<float>(2, 2 + 10e-6)); | ||
|
||
// expect same value if epsilon is larger than difference | ||
EXPECT_TRUE(autoware::point_types::float_eq<float>(2, 2 + 10e-6, 10e-5)); | ||
} |