From e6063c4b7b6586258376400e0b8b442f0afd24fb Mon Sep 17 00:00:00 2001 From: "Yi-Hsiang Fang (Vivid)" <146902905+vividf@users.noreply.github.com> Date: Fri, 6 Sep 2024 02:03:47 +0900 Subject: [PATCH] fix(autoware_image_projection_based_fusion): resolve issue with segmentation pointcloud fusion node failing with multiple mask inputs (#8769) --- .../segmentation_pointcloud_fusion/node.hpp | 2 + .../segmentation_pointcloud_fusion/node.cpp | 53 ++++++++++--------- 2 files changed, 30 insertions(+), 25 deletions(-) diff --git a/perception/image_projection_based_fusion/include/image_projection_based_fusion/segmentation_pointcloud_fusion/node.hpp b/perception/image_projection_based_fusion/include/image_projection_based_fusion/segmentation_pointcloud_fusion/node.hpp index bf729fe0e6f76..cf6a7bb6ca873 100644 --- a/perception/image_projection_based_fusion/include/image_projection_based_fusion/segmentation_pointcloud_fusion/node.hpp +++ b/perception/image_projection_based_fusion/include/image_projection_based_fusion/segmentation_pointcloud_fusion/node.hpp @@ -20,6 +20,7 @@ #include #include +#include #include #include @@ -46,6 +47,7 @@ class SegmentPointCloudFusionNode : public FusionNode filter_global_offset_set_; public: explicit SegmentPointCloudFusionNode(const rclcpp::NodeOptions & options); diff --git a/perception/image_projection_based_fusion/src/segmentation_pointcloud_fusion/node.cpp b/perception/image_projection_based_fusion/src/segmentation_pointcloud_fusion/node.cpp index 86544e3490c99..44481a97fe560 100644 --- a/perception/image_projection_based_fusion/src/segmentation_pointcloud_fusion/node.cpp +++ b/perception/image_projection_based_fusion/src/segmentation_pointcloud_fusion/node.cpp @@ -49,10 +49,31 @@ void SegmentPointCloudFusionNode::preprocess(__attribute__((unused)) PointCloud2 return; } -void SegmentPointCloudFusionNode::postprocess(__attribute__((unused)) PointCloud2 & pointcloud_msg) +void SegmentPointCloudFusionNode::postprocess(PointCloud2 & pointcloud_msg) { + auto original_cloud = std::make_shared(pointcloud_msg); + + int point_step = original_cloud->point_step; + size_t output_pointcloud_size = 0; + pointcloud_msg.data.clear(); + pointcloud_msg.data.resize(original_cloud->data.size()); + + for (size_t global_offset = 0; global_offset < original_cloud->data.size(); + global_offset += point_step) { + if (filter_global_offset_set_.find(global_offset) == filter_global_offset_set_.end()) { + copyPointCloud( + *original_cloud, point_step, global_offset, pointcloud_msg, output_pointcloud_size); + } + } + + pointcloud_msg.data.resize(output_pointcloud_size); + pointcloud_msg.row_step = output_pointcloud_size / pointcloud_msg.height; + pointcloud_msg.width = output_pointcloud_size / pointcloud_msg.point_step / pointcloud_msg.height; + + filter_global_offset_set_.clear(); return; } + void SegmentPointCloudFusionNode::fuseOnSingleImage( const PointCloud2 & input_pointcloud_msg, __attribute__((unused)) const std::size_t image_id, [[maybe_unused]] const Image & input_mask, __attribute__((unused)) const CameraInfo & camera_info, @@ -102,15 +123,7 @@ void SegmentPointCloudFusionNode::fuseOnSingleImage( int x_offset = input_pointcloud_msg.fields[pcl::getFieldIndex(input_pointcloud_msg, "x")].offset; int y_offset = input_pointcloud_msg.fields[pcl::getFieldIndex(input_pointcloud_msg, "y")].offset; int z_offset = input_pointcloud_msg.fields[pcl::getFieldIndex(input_pointcloud_msg, "z")].offset; - size_t output_pointcloud_size = 0; - output_cloud.data.clear(); - output_cloud.data.resize(input_pointcloud_msg.data.size()); - output_cloud.fields = input_pointcloud_msg.fields; - output_cloud.header = input_pointcloud_msg.header; - output_cloud.height = input_pointcloud_msg.height; - output_cloud.point_step = input_pointcloud_msg.point_step; - output_cloud.is_bigendian = input_pointcloud_msg.is_bigendian; - output_cloud.is_dense = input_pointcloud_msg.is_dense; + for (size_t global_offset = 0; global_offset < transformed_cloud.data.size(); global_offset += point_step) { float transformed_x = @@ -121,8 +134,6 @@ void SegmentPointCloudFusionNode::fuseOnSingleImage( *reinterpret_cast(&transformed_cloud.data[global_offset + z_offset]); // skip filtering pointcloud behind the camera or too far from camera if (transformed_z <= 0.0 || transformed_z > filter_distance_threshold_) { - copyPointCloud( - input_pointcloud_msg, point_step, global_offset, output_cloud, output_pointcloud_size); continue; } @@ -135,8 +146,6 @@ void SegmentPointCloudFusionNode::fuseOnSingleImage( normalized_projected_point.x() > 0 && normalized_projected_point.x() < camera_info.width && normalized_projected_point.y() > 0 && normalized_projected_point.y() < camera_info.height; if (!is_inside_image) { - copyPointCloud( - input_pointcloud_msg, point_step, global_offset, output_cloud, output_pointcloud_size); continue; } @@ -144,20 +153,14 @@ void SegmentPointCloudFusionNode::fuseOnSingleImage( uint8_t semantic_id = mask.at( static_cast(normalized_projected_point.y()), static_cast(normalized_projected_point.x())); - if (static_cast(semantic_id) >= filter_semantic_label_target_list_.size()) { - copyPointCloud( - input_pointcloud_msg, point_step, global_offset, output_cloud, output_pointcloud_size); + if ( + static_cast(semantic_id) >= filter_semantic_label_target_list_.size() || + !filter_semantic_label_target_list_.at(semantic_id).second) { continue; } - if (!filter_semantic_label_target_list_.at(semantic_id).second) { - copyPointCloud( - input_pointcloud_msg, point_step, global_offset, output_cloud, output_pointcloud_size); - } - } - output_cloud.data.resize(output_pointcloud_size); - output_cloud.row_step = output_pointcloud_size / output_cloud.height; - output_cloud.width = output_pointcloud_size / output_cloud.point_step / output_cloud.height; + filter_global_offset_set_.insert(global_offset); + } } bool SegmentPointCloudFusionNode::out_of_scope(__attribute__((unused))