From 1e52a0069b006b5ac99d9c2fbfa8eb2efeb2ae6d Mon Sep 17 00:00:00 2001 From: Autumn60 Date: Sat, 13 Jul 2024 14:35:36 +0900 Subject: [PATCH 1/5] create files and define PointXYZIR for velodyne_points Signed-off-by: Autumn60 --- .../velodyne_cloud_separator/CMakeLists.txt | 34 ++++++++++++ .../include/point_types.hpp | 18 +++++++ .../velodyne_cloud_separator.hpp | 40 ++++++++++++++ .../velodyne_cloud_separator/package.xml | 24 +++++++++ .../src/velodyne_cloud_separator.cpp | 54 +++++++++++++++++++ 5 files changed, 170 insertions(+) create mode 100644 src/perception/velodyne_cloud_separator/CMakeLists.txt create mode 100644 src/perception/velodyne_cloud_separator/include/point_types.hpp create mode 100644 src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp create mode 100644 src/perception/velodyne_cloud_separator/package.xml create mode 100644 src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp diff --git a/src/perception/velodyne_cloud_separator/CMakeLists.txt b/src/perception/velodyne_cloud_separator/CMakeLists.txt new file mode 100644 index 0000000..e6bcaf2 --- /dev/null +++ b/src/perception/velodyne_cloud_separator/CMakeLists.txt @@ -0,0 +1,34 @@ +cmake_minimum_required(VERSION 3.5) +project(velodyne_cloud_separator) + +if(NOT CMAKE_CXX_STANDARD) + set(CMAKE_CXX_STANDARD 14) + set(CMAKE_CXX_STANDARD_REQUIRED ON) + set(CMAKE_CXX_EXTENSIONS OFF) +endif() + +if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang") + add_compile_options(-Wall -Wextra -Wpedantic) +endif() + +find_package(ament_cmake_auto REQUIRED) +ament_auto_find_build_dependencies() + +ament_auto_add_library(velodyne_cloud_separator SHARED + src/velodyne_cloud_separator.cpp +) + +rclcpp_components_register_node(velodyne_cloud_separator + PLUGIN "velodyne_cloud_separator::VelodyneCloudSeparator" + EXECUTABLE velodyne_cloud_separator_node +) + +if(BUILD_TESTING) + find_package(ament_lint_auto REQUIRED) + ament_lint_auto_find_test_dependencies() +endif() + +ament_auto_package(INSTALL_TO_SHARE + # config + # launch +) diff --git a/src/perception/velodyne_cloud_separator/include/point_types.hpp b/src/perception/velodyne_cloud_separator/include/point_types.hpp new file mode 100644 index 0000000..407de86 --- /dev/null +++ b/src/perception/velodyne_cloud_separator/include/point_types.hpp @@ -0,0 +1,18 @@ +#ifndef POINT_TYPES_HPP_ +#define POINT_TYPES_HPP_ + +#include + +struct PointXYZIR +{ + PCL_ADD_POINT4D; + float intensity; + uint16_t ring; + EIGEN_MAKE_ALIGNED_OPERATOR_NEW +} EIGEN_ALIGN16; + +POINT_CLOUD_REGISTER_POINT_STRUCT( + PointXYZIR, + (float, x, x)(float, y, y)(float, z, z)(float, intensity, intensity)(uint16_t, ring, ring)) + +#endif // POINT_TYPES_HPP_ diff --git a/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp b/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp new file mode 100644 index 0000000..4aeb44d --- /dev/null +++ b/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp @@ -0,0 +1,40 @@ +#ifndef VELODYNE_CLOUD_SEPARATOR__VELODYNE_CLOUD_SEPARATOR_HPP_ +#define VELODYNE_CLOUD_SEPARATOR__VELODYNE_CLOUD_SEPARATOR_HPP_ + +#include "point_types.hpp" + +#include +#include + +#include + +#include + +namespace velodyne_cloud_separator +{ + +using PointCloud2 = sensor_msgs::msg::PointCloud2; + +using PointCloud2Subscription = wheel_stuck_utils::ros::NoCallbackSubscription; +using PointCloud2Publisher = rclcpp::Publisher; + +class VelodyneCloudSeparator : public rclcpp::Node +{ +public: + explicit VelodyneCloudSeparator(const rclcpp::NodeOptions & options); + void update(); + + rclcpp::TimerBase::SharedPtr update_timer_; + +private: + bool try_subscribe_pc(); + + PointCloud2Subscription::SharedPtr pc_sub_; + PointCloud2Publisher::SharedPtr pc_ground_pub_; + PointCloud2Publisher::SharedPtr pc_obstacle_pub_; + + pcl::PointCloud::Ptr pc_; +}; +} // namespace velodyne_cloud_separator + +#endif // VELODYNE_CLOUD_SEPARATOR__VELODYNE_CLOUD_SEPARATOR_HPP_ diff --git a/src/perception/velodyne_cloud_separator/package.xml b/src/perception/velodyne_cloud_separator/package.xml new file mode 100644 index 0000000..29c6b74 --- /dev/null +++ b/src/perception/velodyne_cloud_separator/package.xml @@ -0,0 +1,24 @@ + + + + velodyne_cloud_separator + 0.0.0 + The velodyne_cloud_separator package + Akiro Harada + + Apache 2 + + ament_cmake_auto + + pcl_conversions + pcl_ros + rclcpp + rclcpp_components + sensor_msgs + wheel_stuck_utils + ament_lint_auto + + + ament_cmake + + diff --git a/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp b/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp new file mode 100644 index 0000000..2588c46 --- /dev/null +++ b/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp @@ -0,0 +1,54 @@ +#include "velodyne_cloud_separator/velodyne_cloud_separator.hpp" + +#include + +namespace velodyne_cloud_separator +{ +VelodyneCloudSeparator::VelodyneCloudSeparator(const rclcpp::NodeOptions & options) +: Node("velodyne_cloud_separator", options) +{ + // Declare Parameters + double update_rate_hz = this->declare_parameter("update_rate_hz", 20.0); + + // Declare Subscriptions + { + pc_sub_ = PointCloud2Subscription::create_subscription(this, "~/input/points"); + } + + // Declare Publishers + { + pc_ground_pub_ = this->create_publisher("~/output/ground_points", 1); + pc_obstacle_pub_ = this->create_publisher("~/output/obstacle_points", 1); + } + + // Declare timer for update function + { + auto update_callback = [this]() { this->update(); }; + auto period = std::chrono::duration_cast( + std::chrono::duration(1.0 / update_rate_hz)); + update_timer_ = std::make_shared>( + this->get_clock(), period, std::move(update_callback), + this->get_node_base_interface()->get_context()); + this->get_node_timers_interface()->add_timer(update_timer_, nullptr); + } +} + +void VelodyneCloudSeparator::update() +{ + if (!try_subscribe_pc()) return; +} + +bool VelodyneCloudSeparator::try_subscribe_pc() +{ + auto pc_msg = pc_sub_->getData(); + if (!pc_msg) return false; + try { + pc_ = pcl::PointCloud::Ptr(new pcl::PointCloud); + pcl::fromROSMsg(*pc_msg, *pc_); + } catch (const std::exception & e) { + RCLCPP_ERROR(get_logger(), "Failed to convert PointCloud2 to pcl::PointCloud: %s", e.what()); + return false; + } + return true; +} +} // namespace velodyne_cloud_separator From 294d0214c81fda5f7c4f7bc169f39a4449a5ff08 Mon Sep 17 00:00:00 2001 From: Autumn60 Date: Sat, 13 Jul 2024 14:43:14 +0900 Subject: [PATCH 2/5] fix member scope Signed-off-by: Autumn60 --- .../velodyne_cloud_separator/velodyne_cloud_separator.hpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp b/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp index 4aeb44d..828db74 100644 --- a/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp +++ b/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp @@ -22,13 +22,13 @@ class VelodyneCloudSeparator : public rclcpp::Node { public: explicit VelodyneCloudSeparator(const rclcpp::NodeOptions & options); - void update(); - - rclcpp::TimerBase::SharedPtr update_timer_; private: + void update(); bool try_subscribe_pc(); + rclcpp::TimerBase::SharedPtr update_timer_; + PointCloud2Subscription::SharedPtr pc_sub_; PointCloud2Publisher::SharedPtr pc_ground_pub_; PointCloud2Publisher::SharedPtr pc_obstacle_pub_; From 6b0c51ac9c392ba6017d2a3327ea97fe972819ad Mon Sep 17 00:00:00 2001 From: Autumn60 Date: Sat, 13 Jul 2024 19:36:46 +0900 Subject: [PATCH 3/5] implement separate function Signed-off-by: Autumn60 --- .../velodyne_cloud_separator.hpp | 17 ++ .../src/velodyne_cloud_separator.cpp | 159 ++++++++++++++++++ 2 files changed, 176 insertions(+) diff --git a/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp b/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp index 828db74..56cd8d9 100644 --- a/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp +++ b/src/perception/velodyne_cloud_separator/include/velodyne_cloud_separator/velodyne_cloud_separator.hpp @@ -10,6 +10,10 @@ #include +#ifndef LASERS_NUM +#define LASERS_NUM 32 +#endif + namespace velodyne_cloud_separator { @@ -20,10 +24,15 @@ using PointCloud2Publisher = rclcpp::Publisher; class VelodyneCloudSeparator : public rclcpp::Node { +private: + enum class PointType { UNKNOWN = 0, GROUND = 1, OBSTACLE = 2 }; + public: explicit VelodyneCloudSeparator(const rclcpp::NodeOptions & options); private: + void init( + const double sensor_height, const double radius_coef_close, const double radius_coef_far); void update(); bool try_subscribe_pc(); @@ -34,6 +43,14 @@ class VelodyneCloudSeparator : public rclcpp::Node PointCloud2Publisher::SharedPtr pc_obstacle_pub_; pcl::PointCloud::Ptr pc_; + + static const int height_ = LASERS_NUM; + + double radius_array_[LASERS_NUM]; + + double limiting_ratio_; + double gap_threshold_; + int min_points_num_; }; } // namespace velodyne_cloud_separator diff --git a/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp b/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp index 2588c46..969526f 100644 --- a/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp +++ b/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp @@ -2,6 +2,9 @@ #include +#include +#include + namespace velodyne_cloud_separator { VelodyneCloudSeparator::VelodyneCloudSeparator(const rclcpp::NodeOptions & options) @@ -33,9 +36,165 @@ VelodyneCloudSeparator::VelodyneCloudSeparator(const rclcpp::NodeOptions & optio } } +void VelodyneCloudSeparator::init( + const double sensor_height, const double radius_coef_close, const double radius_coef_far) +{ + // HDL-32E unique parameters + double angle_start = 92.0 / 3.0 * M_PI / 180.0; + double angle_diff = 4.0 / 3.0 * M_PI / 180.0; + + radius_array_[0] = std::numeric_limits::max(); + for (int i = 1; i < height_; i++) { + if (i > 20) { + radius_array_[i] = radius_array_[20]; + continue; + } + + double angle = angle_start - angle_diff * i; + radius_array_[i] = sensor_height * (1.0 / tan(angle) - 1.0 / tan(angle + angle_diff)); + radius_array_[i] *= (i <= 12 ? radius_coef_close : radius_coef_far); + } +} + void VelodyneCloudSeparator::update() { if (!try_subscribe_pc()) return; + + int points_size = pc_->points.size(); + + const double width_double = points_size * 1.8 / height_; + const int width = static_cast(width_double); + + // Create index map + std::vector> index_map(height_, std::vector(width, -1)); + for (int i = 0; i < points_size; i++) { + const auto & point = pc_->points[i]; + float angle = atan2(point.y, point.x) * 180.0 / M_PI; + if (angle < 0.0) angle += 360.0; + int row = height_ - 1 - point.ring; + int col = width - 1 - static_cast(width_double * angle / 360.0); + index_map[row][col] = i; + } + + // Separate points + pcl::PointCloud::Ptr ground_points(new pcl::PointCloud); + pcl::PointCloud::Ptr obstacle_points(new pcl::PointCloud); + for (int i = 0; i > width; i++) { + PointType point_types[height_] = {PointType::UNKNOWN}; + int point_indices[height_] = {0}; + int point_indices_size = 0; + int unknown_indices[height_] = {0}; + int unknown_indices_size = 0; + double z_ref = 0.0; + double r_ref = 0.0; + + for (int j = 0; j < height_; j++) { + auto & point = pc_->points[index_map[j][i]]; + if (index_map[j][i] == -1 && point_types[j] == PointType::UNKNOWN) { + const float x0 = point.x; + const float y0 = point.y; + const float z0 = point.z; + const float r0 = sqrt(x0 * x0 + y0 * y0); + const float r_diff = r0 - r_ref; + const float z_diff = z0 - z_ref; + const float angle = z_diff / r_diff; + + if ( + !(angle > 0 && angle < limiting_ratio_ && z_diff < gap_threshold_) || + point_indices_size != 0) { + for (int k = 0; k < point_indices_size; k++) { + int index = index_map[point_indices[k]][i]; + if (point_indices_size > min_points_num_) { + auto & p = pc_->points[index]; + pcl::PointXYZ p_xyz; + p_xyz.x = p.x; + p_xyz.y = p.y; + p_xyz.z = p.z; + ground_points->push_back(p_xyz); + point_types[point_indices[k]] = PointType::GROUND; + } else { + unknown_indices[unknown_indices_size++] = index; + } + } + point_indices_size = 0; + } + point_indices[point_indices_size++] = j; + z_ref = z0; + r_ref = r0; + } + if (j != 0) continue; + + if (point_indices != 0) { + for (int k = 0; k < point_indices_size; k++) { + int index = index_map[point_indices[k]][i]; + if (point_indices_size > min_points_num_) { + auto & p = pc_->points[index]; + pcl::PointXYZ p_xyz; + p_xyz.x = p.x; + p_xyz.y = p.y; + p_xyz.z = p.z; + ground_points->push_back(p_xyz); + point_types[point_indices[k]] = PointType::GROUND; + } else { + unknown_indices[unknown_indices_size++] = index; + } + } + point_indices_size = 0; + } + + double centroid = 0; + int centroid_ring = 0; + int cluster_indices[height_] = {0}; + int cluster_indices_size = 0; + for (int k = unknown_indices_size - 1; k >= 0; k--) { + float x0 = pc_->points[unknown_indices[k]].x; + float y0 = pc_->points[unknown_indices[k]].y; + float r0 = sqrt(x0 * x0 + y0 * y0); + float r_diff = fabs(r0 - centroid); + float dist = radius_array_[centroid_ring]; + if (r_diff >= dist && cluster_indices_size != 0) { + for (int l = 0; l < cluster_indices_size; l++) { + auto & p = pc_->points[cluster_indices[l]]; + pcl::PointXYZ p_xyz; + p_xyz.x = p.x; + p_xyz.y = p.y; + p_xyz.z = p.z; + (cluster_indices_size > 1 ? obstacle_points : ground_points)->push_back(p_xyz); + } + cluster_indices_size = 0; + } + cluster_indices[cluster_indices_size++] = unknown_indices[k]; + centroid = r0; + centroid_ring = pc_->points[unknown_indices[k]].ring; + if (k != 0) continue; + for (int l = 0; l < cluster_indices_size; l++) { + auto & p = pc_->points[cluster_indices[l]]; + pcl::PointXYZ p_xyz; + p_xyz.x = p.x; + p_xyz.y = p.y; + p_xyz.z = p.z; + (cluster_indices_size > 1 ? obstacle_points : ground_points)->push_back(p_xyz); + } + cluster_indices_size = 0; + } + } + } + + // Publish + { + sensor_msgs::msg::PointCloud2 pc_ground_msg; + pcl::toROSMsg(*ground_points, pc_ground_msg); + pc_ground_msg.header.frame_id = pc_->header.frame_id; + pc_ground_msg.header.stamp = this->now(); + pc_ground_pub_->publish(pc_ground_msg); + } + { + sensor_msgs::msg::PointCloud2 pc_obstacle_msg; + pcl::toROSMsg(*obstacle_points, pc_obstacle_msg); + pc_obstacle_msg.header.frame_id = pc_->header.frame_id; + pc_obstacle_msg.header.stamp = this->now(); + pc_obstacle_pub_->publish(pc_obstacle_msg); + } } bool VelodyneCloudSeparator::try_subscribe_pc() From 0b876cbf32ed243cb7df0ce9ce3963151d4fc8af Mon Sep 17 00:00:00 2001 From: Autumn60 Date: Sat, 13 Jul 2024 19:38:54 +0900 Subject: [PATCH 4/5] chore: Initialize parameters and perform initialization in VelodyneCloudSeparator constructor Signed-off-by: Autumn60 --- .../src/velodyne_cloud_separator.cpp | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp b/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp index 969526f..b0a5eec 100644 --- a/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp +++ b/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp @@ -12,6 +12,12 @@ VelodyneCloudSeparator::VelodyneCloudSeparator(const rclcpp::NodeOptions & optio { // Declare Parameters double update_rate_hz = this->declare_parameter("update_rate_hz", 20.0); + double sensor_height = this->declare_parameter("sensor_height", 2.0); + double radius_coef_close = this->declare_parameter("radius_coef_close", 1.0); + double radius_coef_far = this->declare_parameter("radius_coef_far", 1.0); + limiting_ratio_ = this->declare_parameter("limiting_ratio", 0.1); + gap_threshold_ = this->declare_parameter("gap_threshold", 0.1); + min_points_num_ = this->declare_parameter("min_points_num", 3); // Declare Subscriptions { @@ -34,6 +40,9 @@ VelodyneCloudSeparator::VelodyneCloudSeparator(const rclcpp::NodeOptions & optio this->get_node_base_interface()->get_context()); this->get_node_timers_interface()->add_timer(update_timer_, nullptr); } + + // Initialize + init(sensor_height, radius_coef_close, radius_coef_far); } void VelodyneCloudSeparator::init( From 70f12ac3c48011e3df5ea2774035c10b706d0a15 Mon Sep 17 00:00:00 2001 From: Autumn60 Date: Sun, 14 Jul 2024 17:12:08 +0900 Subject: [PATCH 5/5] create launch and config Signed-off-by: Autumn60 --- .../velodyne_cloud_separator/CMakeLists.txt | 4 +-- .../velodyne_cloud_separator.param.yaml | 9 ++++++ .../velodyne_cloud_separator.launch.xml | 14 +++++++++ .../src/velodyne_cloud_separator.cpp | 31 ++++++++++--------- 4 files changed, 42 insertions(+), 16 deletions(-) create mode 100644 src/perception/velodyne_cloud_separator/config/velodyne_cloud_separator.param.yaml create mode 100644 src/perception/velodyne_cloud_separator/launch/velodyne_cloud_separator.launch.xml diff --git a/src/perception/velodyne_cloud_separator/CMakeLists.txt b/src/perception/velodyne_cloud_separator/CMakeLists.txt index e6bcaf2..eb0ff3e 100644 --- a/src/perception/velodyne_cloud_separator/CMakeLists.txt +++ b/src/perception/velodyne_cloud_separator/CMakeLists.txt @@ -29,6 +29,6 @@ if(BUILD_TESTING) endif() ament_auto_package(INSTALL_TO_SHARE - # config - # launch + config + launch ) diff --git a/src/perception/velodyne_cloud_separator/config/velodyne_cloud_separator.param.yaml b/src/perception/velodyne_cloud_separator/config/velodyne_cloud_separator.param.yaml new file mode 100644 index 0000000..b0b8399 --- /dev/null +++ b/src/perception/velodyne_cloud_separator/config/velodyne_cloud_separator.param.yaml @@ -0,0 +1,9 @@ +/**: + ros__parameters: + update_rate_hz: 20.0 + sensor_height_m: 0.68 + radius_coef_close: 0.2 + radius_coef_far: 0.7 + max_slope_deg: 10.0 + gap_threshold_m: 0.2 + min_points_per_cluster: 2 diff --git a/src/perception/velodyne_cloud_separator/launch/velodyne_cloud_separator.launch.xml b/src/perception/velodyne_cloud_separator/launch/velodyne_cloud_separator.launch.xml new file mode 100644 index 0000000..f3c89b5 --- /dev/null +++ b/src/perception/velodyne_cloud_separator/launch/velodyne_cloud_separator.launch.xml @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + diff --git a/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp b/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp index b0a5eec..cbea368 100644 --- a/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp +++ b/src/perception/velodyne_cloud_separator/src/velodyne_cloud_separator.cpp @@ -12,12 +12,13 @@ VelodyneCloudSeparator::VelodyneCloudSeparator(const rclcpp::NodeOptions & optio { // Declare Parameters double update_rate_hz = this->declare_parameter("update_rate_hz", 20.0); - double sensor_height = this->declare_parameter("sensor_height", 2.0); - double radius_coef_close = this->declare_parameter("radius_coef_close", 1.0); - double radius_coef_far = this->declare_parameter("radius_coef_far", 1.0); - limiting_ratio_ = this->declare_parameter("limiting_ratio", 0.1); - gap_threshold_ = this->declare_parameter("gap_threshold", 0.1); - min_points_num_ = this->declare_parameter("min_points_num", 3); + double sensor_height = this->declare_parameter("sensor_height_m", 1.0); + double radius_coef_close = this->declare_parameter("radius_coef_close", 0.2); + double radius_coef_far = this->declare_parameter("radius_coef_far", 0.7); + float max_slope_deg = this->declare_parameter("max_slope_deg", 15.0); + limiting_ratio_ = tan(max_slope_deg * M_PI / 180.0); + gap_threshold_ = this->declare_parameter("gap_threshold_m", 0.1); + min_points_num_ = this->declare_parameter("min_points_per_cluster", 2); // Declare Subscriptions { @@ -71,8 +72,7 @@ void VelodyneCloudSeparator::update() int points_size = pc_->points.size(); - const double width_double = points_size * 1.8 / height_; - const int width = static_cast(width_double); + const int width = static_cast(points_size * 1.8 / height_); // Create index map std::vector> index_map(height_, std::vector(width, -1)); @@ -81,14 +81,14 @@ void VelodyneCloudSeparator::update() float angle = atan2(point.y, point.x) * 180.0 / M_PI; if (angle < 0.0) angle += 360.0; int row = height_ - 1 - point.ring; - int col = width - 1 - static_cast(width_double * angle / 360.0); + int col = width - 1 - static_cast(static_cast(width) * angle / 360.0); index_map[row][col] = i; } // Separate points pcl::PointCloud::Ptr ground_points(new pcl::PointCloud); pcl::PointCloud::Ptr obstacle_points(new pcl::PointCloud); - for (int i = 0; i > width; i++) { + for (int i = 0; i < width; i++) { PointType point_types[height_] = {PointType::UNKNOWN}; int point_indices[height_] = {0}; int point_indices_size = 0; @@ -97,15 +97,15 @@ void VelodyneCloudSeparator::update() double z_ref = 0.0; double r_ref = 0.0; - for (int j = 0; j < height_; j++) { + for (int j = height_ - 1; j >= 0; j--) { auto & point = pc_->points[index_map[j][i]]; - if (index_map[j][i] == -1 && point_types[j] == PointType::UNKNOWN) { + if (index_map[j][i] > -1 && point_types[j] == PointType::UNKNOWN) { const float x0 = point.x; const float y0 = point.y; const float z0 = point.z; const float r0 = sqrt(x0 * x0 + y0 * y0); const float r_diff = r0 - r_ref; - const float z_diff = z0 - z_ref; + const float z_diff = fabs(z0 - z_ref); const float angle = z_diff / r_diff; if ( @@ -133,7 +133,7 @@ void VelodyneCloudSeparator::update() } if (j != 0) continue; - if (point_indices != 0) { + if (point_indices_size != 0) { for (int k = 0; k < point_indices_size; k++) { int index = index_map[point_indices[k]][i]; if (point_indices_size > min_points_num_) { @@ -220,3 +220,6 @@ bool VelodyneCloudSeparator::try_subscribe_pc() return true; } } // namespace velodyne_cloud_separator + +#include "rclcpp_components/register_node_macro.hpp" +RCLCPP_COMPONENTS_REGISTER_NODE(velodyne_cloud_separator::VelodyneCloudSeparator)