From 15ec6fff807c7e6c7311b9b76d2ead25232c4fba Mon Sep 17 00:00:00 2001 From: Takeshi Miura <57553950+1222-takeshi@users.noreply.github.com> Date: Thu, 2 Dec 2021 12:10:37 +0900 Subject: [PATCH] feat: add dummy perception publisher (#90) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * release v0.4.0 * add use_object_recognition flag in dummy_perception_publisher (#696) * remove ROS1 packages temporarily Signed-off-by: mitsudome-r * add sample ros2 packages Signed-off-by: mitsudome-r * remove ROS1 packages Signed-off-by: mitsudome-r * Revert "remove ROS1 packages temporarily" This reverts commit 2e9822586a3539a32653e6bcd378715674b519ca. Signed-off-by: mitsudome-r * add COLCON_IGNORE to ros1 packages Signed-off-by: mitsudome-r * Rename launch files to launch.xml (#28) * Port dummy_perception_publisher to ROS2 (#90) * Port dummy_perception_publisher to ROS2 * Uncrustify * Lint * Copyright * Period * Further ament_cpplint fixes * Convert calls of Duration to Duration::from_seconds where appropriate (#131) * Use quotes for includes where appropriate (#144) * Use quotes for includes where appropriate * Fix lint tests * Make tests pass hopefully * adding linters to dummy_perception_publisher (#220) * [dummy_perception_publisher] fix launch file and installation (#215) * [dummy_perception_publisher] fix launch file and installation Signed-off-by: mitsudome-r * Apply suggestions from code review Co-authored-by: Takamasa Horibe Co-authored-by: Takamasa Horibe * reduce terminal ouput for better error message visibility (#200) * reduce terminal ouput for better error message visibility Signed-off-by: mitsudome-r * [costmap_generator] fix waiting for first transform Signed-off-by: mitsudome-r * fix tests Signed-off-by: mitsudome-r * fix test Signed-off-by: mitsudome-r * modify launch file for making psim work (#238) * modify launch file for making psim work Signed-off-by: kosuke murakami * remove unnecesary ns Signed-off-by: kosuke murakami * Ros2 v0.8.0 dummy perception publisher (#286) * Remove "/" in frame_id (#406) Signed-off-by: Kenji Miyake * Fix transform (#420) * Replace rclcpp::Time(0) by tf2::TimePointZero() in lookupTransform Signed-off-by: Kenji Miyake * Fix canTransform Signed-off-by: Kenji Miyake * Fix test Signed-off-by: Kenji Miyake * add use_sim-time option (#454) * Remove use_sim_time for set_parameter (#1260) Signed-off-by: wep21 * Diable dummy_perception_publisher if argument 'scenario_simulation' i… (#1275) * Diable dummy_perception_publisher if argument 'scenario_simulation' is true * Rename argument to 'disable_dummy_perception_publisher_node' from 'scenario_simulation' * change theta step for obj point cloud (#1280) * Revert changes of PR #1275 (#1377) * Add pre-commit (#1560) * add pre-commit * add pre-commit-config * add additional settings for private repository * use default pre-commit-config * update pre-commit setting * Ignore whitespace for line breaks in markdown * Update .github/workflows/pre-commit.yml Co-authored-by: Kazuki Miyahara * exclude svg * remove pretty-format-json * add double-quote-string-fixer * consider COLCON_IGNORE file when seaching modified package * format file * pre-commit fixes * Update pre-commit.yml * Update .pre-commit-config.yaml Co-authored-by: Kazuki Miyahara Co-authored-by: pre-commit Co-authored-by: Kenji Miyake <31987104+kenji-miyake@users.noreply.github.com> * Fix dependency type of rosidl_default_generators (#1801) * Fix dependency type of rosidl_default_generators Signed-off-by: Kenji Miyake * Remove unnecessary depends Signed-off-by: Kenji Miyake * Use ament_cmake_auto Signed-off-by: Kenji Miyake * Fix -Wunused-parameter (#1836) * Fix -Wunused-parameter Signed-off-by: Kenji Miyake * Fix mistake Signed-off-by: Kenji Miyake * fix spell * Fix lint issues Signed-off-by: Kenji Miyake * Ignore flake8 warnings Signed-off-by: Kenji Miyake Co-authored-by: Hiroki OTA * fix topic namespace (#2054) Signed-off-by: Azumi Suzuki * add sort-package-xml hook in pre-commit (#1881) * add sort xml hook in pre-commit * change retval to exit_status * rename * add prettier plugin-xml * use early return * add license note * add tier4 license * restore prettier * change license order * move local hooks to public repo * move prettier-xml to pre-commit-hooks-ros * update version for bug-fix * apply pre-commit * Feature/porting occlusion spot (#1740) * Feature/occlusion_spot safety planner public road (#1594) * add blind spot safety planner public road * remove duplicated procesing * remove unused private param * renaming fix typo add comments * fix spell check * velocity -> relative velocity * calc2d, To param, simplify search, To original * add the num possible collision gurd * computational cost reduction * Cosmetic change for PossibleCollisionInfo * add interpolation to possible collision value * refactor codes * fix details * invalid point gurd * To Param * refacotor to occlusion spot util * cosmetic change * clean up blindspot * To Occlusion Spot * revise readme * refactor drawing * for better explanation * fix velocity profile * clean up details * cosmetic change for debug marker * use max velocity in obstacle info instead * add gtest for Too Many Possible Collision case * change case * refactor readme * minor fix * add more occlusion spot explanation * replace svg * add gtest build path lanelet * hotfix lateral distance and searching method * update g test for lateral distance * use faster search * set more realistic param * add lanelet subtype highway * cosmetic change of reviews * add occlusion spot module in private area (#1640) * add occlusion spot in private * For debugging change * add spline interpolation to path * add review changes * adding minor change * cosmetic change * Vector to retval * Blindspot To OcclusionSpot1 * To Occlusion Spot 2 * To Occlusions spot3 * update gtest with unified anchor * remove anchor * add test slice * simplify interpolation * Too Occlusion spot4 * add test buffer * To Occlusion spot * namespace gurd * correct slice and add interpolation first * handle self crossing with check for invation * to ros debug stream * removed unused interpolation * add readme to plant uml * cosmetic change * minor change * update readme * review change * change occlusion spot text color * To Offset no Throw * better debug marker * catch only inversion error * skip return in case of inversion * for better grid * simplify path lanelet at first * remove std::cout * for better path * limit ego path inside target lanelet location * remove last points * cosmetic change for markers * apply module for limited scene * fix interpolation gurd * for better params * do not includes path behind * remove dummy perception publisher * Revert "remove dummy perception publisher" This reverts commit 4acad985fe31dd9befaa21a16631495de6c3a117. * replace hard coded occupancy grid option in psim * remove respawn * add arg to params and remove redundunt launch * fix spell check * fix default value Co-authored-by: tkimura4 * Feature/occlusion spot private slice size param (#1703) * add min slice size * for a bit narrow lateral distance * Update planning/scenario_planning/lane_driving/behavior_planning/behavior_velocity_planner/config/occlusion_spot_param.yaml Co-authored-by: Maxime CLEMENT <78338830+maxime-clem@users.noreply.github.com> Co-authored-by: Maxime CLEMENT <78338830+maxime-clem@users.noreply.github.com> * Rename files Signed-off-by: wep21 * Porting to ros2 Signed-off-by: wep21 * pre-commit fixes Signed-off-by: wep21 * Fix typo Signed-off-by: wep21 * Fix launch namespace Co-authored-by: tkimura4 * Fix parameter type Co-authored-by: tkimura4 * Update planning/scenario_planning/lane_driving/behavior_planning/behavior_velocity_planner/src/scene_module/occlusion_spot/scene_occlusion_spot_in_private_road.cpp Co-authored-by: tkimura4 Co-authored-by: taikitanaka3 <65527974+taikitanaka3@users.noreply.github.com> Co-authored-by: tkimura4 Co-authored-by: Maxime CLEMENT <78338830+maxime-clem@users.noreply.github.com> * Change formatter to clang-format and black (#2332) * Revert "Temporarily comment out pre-commit hooks" This reverts commit 748e9cdb145ce12f8b520bcbd97f5ff899fc28a3. * Replace ament_lint_common with autoware_lint_common Signed-off-by: Kenji Miyake * Remove ament_cmake_uncrustify and ament_clang_format Signed-off-by: Kenji Miyake * Apply Black Signed-off-by: Kenji Miyake * Apply clang-format Signed-off-by: Kenji Miyake * Fix build errors Signed-off-by: Kenji Miyake * Fix for cpplint * Fix include double quotes to angle brackets Signed-off-by: Kenji Miyake * Apply clang-format Signed-off-by: Kenji Miyake * Fix build errors Signed-off-by: Kenji Miyake * Add COLCON_IGNORE (#500) Signed-off-by: Kenji Miyake * port dummy perception publisher to auto (#562) * port dummy perception publisher to auto * autoware_perception_msgs/DynamicObjectWithFeatureArray convert to autoware_perception_msgs/DetectedObjectsWithFeature * change tracked objects to PREDICTED objects * separate pub type with real * Add README.md to dummy perception publisher (#641) * Added readme for dummy_perception_pub * README update * README update * Fix pre-commit * fix typo * Update README.md * Update README.md * Update README.md * Modified node.cpp * Modified README.md * change parameter name * Update README.md * [shape_estimation]change type (#663) * change output type of shape_estimation * remove unused function * add dynamic_object_converter * rename * fix typo * fix dummy_perception_publisher * update readme * fix copyright * rename package * add readme * fix launch name * remove unused variable * fix readme * fix convert function * change topic name of DynamicObjectsWithFeature * Fix no ground pointcloud topic name (#733) Signed-off-by: j4tfwm6z Co-authored-by: j4tfwm6z * auto/fix occupancy grid name space in dummy perception publisher (#739) * fix name space * change namespace: object_segmentation -> obstacle_segmentation * feat: add use_traffic_light status Co-authored-by: mitsudome-r Co-authored-by: Taichi Higashide Co-authored-by: Nikolai Morin Co-authored-by: nik-tier4 <71747268+nik-tier4@users.noreply.github.com> Co-authored-by: Ryohsuke Mitsudome <43976834+mitsudome-r@users.noreply.github.com> Co-authored-by: Takamasa Horibe Co-authored-by: Kosuke Murakami Co-authored-by: Daisuke Nishimatsu <42202095+wep21@users.noreply.github.com> Co-authored-by: Kenji Miyake <31987104+kenji-miyake@users.noreply.github.com> Co-authored-by: tkimura4 Co-authored-by: Tatsuya Yamasaki Co-authored-by: taikitanaka3 <65527974+taikitanaka3@users.noreply.github.com> Co-authored-by: Keisuke Shima <19993104+KeisukeShima@users.noreply.github.com> Co-authored-by: Kazuki Miyahara Co-authored-by: pre-commit Co-authored-by: Hiroki OTA Co-authored-by: Azumi Suzuki <38061530+s-azumi@users.noreply.github.com> Co-authored-by: Maxime CLEMENT <78338830+maxime-clem@users.noreply.github.com> Co-authored-by: Yohei Mishina <66298900+YoheiMishina@users.noreply.github.com> Co-authored-by: j4tfwm6z Co-authored-by: Satoshi OTA <44889564+satoshi-ota@users.noreply.github.com> --- .../dummy_perception_publisher/CMakeLists.txt | 87 ++++ .../dummy_perception_publisher/README.md | 44 ++ .../dummy_perception_publisher/node.hpp | 69 +++ .../dummy_perception_publisher.launch.xml | 97 ++++ .../msg/InitialState.msg | 2 + .../dummy_perception_publisher/msg/Object.msg | 11 + .../dummy_perception_publisher/package.xml | 38 ++ .../src/empty_objects_publisher.cpp | 59 +++ .../dummy_perception_publisher/src/main.cpp | 28 ++ .../dummy_perception_publisher/src/node.cpp | 418 ++++++++++++++++++ 10 files changed, 853 insertions(+) create mode 100644 simulator/dummy_perception_publisher/CMakeLists.txt create mode 100644 simulator/dummy_perception_publisher/README.md create mode 100644 simulator/dummy_perception_publisher/include/dummy_perception_publisher/node.hpp create mode 100644 simulator/dummy_perception_publisher/launch/dummy_perception_publisher.launch.xml create mode 100644 simulator/dummy_perception_publisher/msg/InitialState.msg create mode 100644 simulator/dummy_perception_publisher/msg/Object.msg create mode 100644 simulator/dummy_perception_publisher/package.xml create mode 100644 simulator/dummy_perception_publisher/src/empty_objects_publisher.cpp create mode 100644 simulator/dummy_perception_publisher/src/main.cpp create mode 100644 simulator/dummy_perception_publisher/src/node.cpp diff --git a/simulator/dummy_perception_publisher/CMakeLists.txt b/simulator/dummy_perception_publisher/CMakeLists.txt new file mode 100644 index 0000000000000..3e9a7f5a2feb9 --- /dev/null +++ b/simulator/dummy_perception_publisher/CMakeLists.txt @@ -0,0 +1,87 @@ +cmake_minimum_required(VERSION 3.5) +project(dummy_perception_publisher) + +### Compile options +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() + +# Dependencies for messages +find_package(ament_cmake REQUIRED) +find_package(rosidl_default_generators REQUIRED) +find_package(autoware_auto_perception_msgs REQUIRED) +find_package(autoware_perception_msgs REQUIRED) +find_package(geometry_msgs REQUIRED) +find_package(std_msgs REQUIRED) +find_package(unique_identifier_msgs REQUIRED) + +rosidl_generate_interfaces(${PROJECT_NAME} + "msg/InitialState.msg" + "msg/Object.msg" + DEPENDENCIES autoware_auto_perception_msgs autoware_perception_msgs geometry_msgs std_msgs unique_identifier_msgs +) + +# See ndt_omp package for documentation on why PCL is special +find_package(PCL REQUIRED COMPONENTS common filters) +find_package(ament_cmake_auto REQUIRED) +ament_auto_find_build_dependencies() + +set(${PROJECT_NAME}_DEPENDENCIES + autoware_auto_perception_msgs + autoware_perception_msgs + pcl_conversions + rclcpp + sensor_msgs + std_msgs + tf2 + tf2_geometry_msgs + tf2_ros +) + +ament_auto_add_executable(dummy_perception_publisher_node + src/main.cpp + src/node.cpp +) + +ament_target_dependencies(dummy_perception_publisher_node ${${PROJECT_NAME}_DEPENDENCIES}) + +target_include_directories(dummy_perception_publisher_node + PUBLIC + $ + $) + +# For using message definitions from the same package +rosidl_target_interfaces(dummy_perception_publisher_node + ${PROJECT_NAME} "rosidl_typesupport_cpp") + +# PCL dependencies – `ament_target_dependencies` doesn't respect the +# components/modules selected above and only links in `common` ,so we need +# to do this manually. +target_compile_definitions(dummy_perception_publisher_node PRIVATE ${PCL_DEFINITIONS}) +target_include_directories(dummy_perception_publisher_node PRIVATE ${PCL_INCLUDE_DIRS}) +# Unfortunately, this one can't be PRIVATE because only the plain or only the +# keyword (PRIVATE) signature of target_link_libraries can be used for one +# target, not both. The plain signature is already used inside +# `ament_target_dependencies` and possibly rosidl_target_interfaces. +target_link_libraries(dummy_perception_publisher_node ${PCL_LIBRARIES}) +target_link_directories(dummy_perception_publisher_node PRIVATE ${PCL_LIBRARY_DIRS}) + + +ament_auto_add_executable(empty_objects_publisher + src/empty_objects_publisher.cpp +) + +if(BUILD_TESTING) + find_package(ament_lint_auto REQUIRED) + ament_lint_auto_find_test_dependencies() +endif() + +ament_auto_package( + INSTALL_TO_SHARE + launch +) diff --git a/simulator/dummy_perception_publisher/README.md b/simulator/dummy_perception_publisher/README.md new file mode 100644 index 0000000000000..281932fa1caaa --- /dev/null +++ b/simulator/dummy_perception_publisher/README.md @@ -0,0 +1,44 @@ +# dummy_perception_publisher + +## Purpose + +This node publishes the result of the dummy detection with the type of perception. + +## Inner-workings / Algorithms + +## Inputs / Outputs + +### Input + +| Name | Type | Description | +| -------------- | ----------------------------------------- | ----------------------- | +| `/tf` | `tf2_msgs/TFMessage` | TF (self-pose) | +| `input/object` | `dummy_perception_publisher::msg::Object` | dummy detection objects | + +### Output + +| Name | Type | Description | +| ----------------------- | ----------------------------------------------------------- | ---------------------- | +| `output/dynamic_object` | `autoware_perception_msgs::msg::DetectedObjectsWithFeature` | Publishes objects | +| `output/points_raw` | `sensor_msgs::msg::PointCloud2` | point cloud of objects | + +## Parameters + +| Name | Type | Default Value | Explanation | +| --------------------------- | ------ | ------------- | ------------------------------------------- | +| `visible_range` | double | 100.0 | sensor visible range [m] | +| `detection_successful_rate` | double | 0.8 | sensor detection rate. (min) 0.0 - 1.0(max) | +| `enable_ray_tracing` | bool | true | if True, use ray tracking | +| `use_object_recognition` | bool | true | if True, publish objects topic | + +### Node Parameters + +None. + +### Core Parameters + +None. + +## Assumptions / Known limits + +TBD. diff --git a/simulator/dummy_perception_publisher/include/dummy_perception_publisher/node.hpp b/simulator/dummy_perception_publisher/include/dummy_perception_publisher/node.hpp new file mode 100644 index 0000000000000..b51887bd09fe0 --- /dev/null +++ b/simulator/dummy_perception_publisher/include/dummy_perception_publisher/node.hpp @@ -0,0 +1,69 @@ +// Copyright 2020 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 DUMMY_PERCEPTION_PUBLISHER__NODE_HPP_ +#define DUMMY_PERCEPTION_PUBLISHER__NODE_HPP_ + +#include "dummy_perception_publisher/msg/object.hpp" + +#include + +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +class DummyPerceptionPublisherNode : public rclcpp::Node +{ +private: + rclcpp::Publisher::SharedPtr pointcloud_pub_; + rclcpp::Publisher::SharedPtr + detected_object_with_feature_pub_; + rclcpp::Subscription::SharedPtr object_sub_; + rclcpp::TimerBase::SharedPtr timer_; + tf2_ros::Buffer tf_buffer_; + tf2_ros::TransformListener tf_listener_; + std::vector objects_; + double visible_range_; + double detection_successful_rate_; + bool enable_ray_tracing_; + bool use_object_recognition_; + bool use_real_param_; + std::mt19937 random_generator_; + void timerCallback(); + void createObjectPointcloud( + const double length, const double width, const double height, const double std_dev_x, + const double std_dev_y, const double std_dev_z, + const tf2::Transform & tf_base_link2moved_object, + pcl::PointCloud::Ptr & pointcloud_ptr); + void objectCallback(const dummy_perception_publisher::msg::Object::ConstSharedPtr msg); + +public: + DummyPerceptionPublisherNode(); + ~DummyPerceptionPublisherNode() {} +}; + +#endif // DUMMY_PERCEPTION_PUBLISHER__NODE_HPP_ diff --git a/simulator/dummy_perception_publisher/launch/dummy_perception_publisher.launch.xml b/simulator/dummy_perception_publisher/launch/dummy_perception_publisher.launch.xml new file mode 100644 index 0000000000000..421bd978020d3 --- /dev/null +++ b/simulator/dummy_perception_publisher/launch/dummy_perception_publisher.launch.xml @@ -0,0 +1,97 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/simulator/dummy_perception_publisher/msg/InitialState.msg b/simulator/dummy_perception_publisher/msg/InitialState.msg new file mode 100644 index 0000000000000..1b73ce57e8a88 --- /dev/null +++ b/simulator/dummy_perception_publisher/msg/InitialState.msg @@ -0,0 +1,2 @@ +geometry_msgs/PoseWithCovariance pose_covariance +geometry_msgs/TwistWithCovariance twist_covariance diff --git a/simulator/dummy_perception_publisher/msg/Object.msg b/simulator/dummy_perception_publisher/msg/Object.msg new file mode 100644 index 0000000000000..199a48b949bcb --- /dev/null +++ b/simulator/dummy_perception_publisher/msg/Object.msg @@ -0,0 +1,11 @@ +std_msgs/Header header +unique_identifier_msgs/UUID id +dummy_perception_publisher/InitialState initial_state +autoware_auto_perception_msgs/ObjectClassification classification +autoware_auto_perception_msgs/Shape shape + +uint8 ADD=0 +uint8 MODIFY=1 +uint8 DELETE=2 +uint8 DELETEALL=3 +int32 action diff --git a/simulator/dummy_perception_publisher/package.xml b/simulator/dummy_perception_publisher/package.xml new file mode 100644 index 0000000000000..104a179442513 --- /dev/null +++ b/simulator/dummy_perception_publisher/package.xml @@ -0,0 +1,38 @@ + + + + dummy_perception_publisher + 0.1.0 + The dummy_perception_publisher package + + Yukihiro Saito + Apache License 2.0 + + ament_cmake_auto + + rosidl_default_generators + + rosidl_default_runtime + + ament_lint_auto + autoware_lint_common + + autoware_auto_perception_msgs + autoware_perception_msgs + geometry_msgs + libpcl-all-dev + pcl_conversions + rclcpp + sensor_msgs + std_msgs + tf2 + tf2_geometry_msgs + tf2_ros + unique_identifier_msgs + + rosidl_interface_packages + + + ament_cmake + + diff --git a/simulator/dummy_perception_publisher/src/empty_objects_publisher.cpp b/simulator/dummy_perception_publisher/src/empty_objects_publisher.cpp new file mode 100644 index 0000000000000..fe1df627f143d --- /dev/null +++ b/simulator/dummy_perception_publisher/src/empty_objects_publisher.cpp @@ -0,0 +1,59 @@ +// Copyright 2020 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 + +#include + +#include +#include + +class EmptyObjectsPublisher : public rclcpp::Node +{ +public: + EmptyObjectsPublisher() : Node("empty_objects_publisher") + { + empty_objects_pub_ = + this->create_publisher( + "~/output/objects", 1); + + auto timer_callback = std::bind(&EmptyObjectsPublisher::timerCallback, this); + const auto period = std::chrono::milliseconds(100); + timer_ = std::make_shared>( + this->get_clock(), period, std::move(timer_callback), + this->get_node_base_interface()->get_context()); + this->get_node_timers_interface()->add_timer(timer_, nullptr); + } + +private: + rclcpp::Publisher::SharedPtr + empty_objects_pub_; + rclcpp::TimerBase::SharedPtr timer_; + + void timerCallback() + { + autoware_auto_perception_msgs::msg::PredictedObjects empty_objects; + empty_objects.header.frame_id = "map"; + empty_objects.header.stamp = this->now(); + empty_objects_pub_->publish(empty_objects); + } +}; + +int main(int argc, char ** argv) +{ + rclcpp::init(argc, argv); + rclcpp::spin(std::make_shared()); + rclcpp::shutdown(); + return 0; +} diff --git a/simulator/dummy_perception_publisher/src/main.cpp b/simulator/dummy_perception_publisher/src/main.cpp new file mode 100644 index 0000000000000..5e627f2cbc8ed --- /dev/null +++ b/simulator/dummy_perception_publisher/src/main.cpp @@ -0,0 +1,28 @@ +// Copyright 2020 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 "dummy_perception_publisher/node.hpp" + +#include + +#include + +int main(int argc, char ** argv) +{ + rclcpp::init(argc, argv); + rclcpp::spin(std::make_shared()); + rclcpp::shutdown(); + + return 0; +} diff --git a/simulator/dummy_perception_publisher/src/node.cpp b/simulator/dummy_perception_publisher/src/node.cpp new file mode 100644 index 0000000000000..0adf8912c507a --- /dev/null +++ b/simulator/dummy_perception_publisher/src/node.cpp @@ -0,0 +1,418 @@ +// Copyright 2020 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 "dummy_perception_publisher/node.hpp" + +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +DummyPerceptionPublisherNode::DummyPerceptionPublisherNode() +: Node("dummy_perception_publisher"), tf_buffer_(this->get_clock()), tf_listener_(tf_buffer_) +{ + visible_range_ = this->declare_parameter("visible_range", 100.0); + detection_successful_rate_ = this->declare_parameter("detection_successful_rate", 0.8); + enable_ray_tracing_ = this->declare_parameter("enable_ray_tracing", true); + use_object_recognition_ = this->declare_parameter("use_object_recognition", true); + + std::random_device seed_gen; + random_generator_.seed(seed_gen()); + + rclcpp::QoS qos{1}; + qos.transient_local(); + detected_object_with_feature_pub_ = + this->create_publisher( + "output/dynamic_object", qos); + pointcloud_pub_ = this->create_publisher("output/points_raw", qos); + object_sub_ = this->create_subscription( + "input/object", 100, + std::bind(&DummyPerceptionPublisherNode::objectCallback, this, std::placeholders::_1)); + + auto timer_callback = std::bind(&DummyPerceptionPublisherNode::timerCallback, this); + auto period = std::chrono::milliseconds(100); + timer_ = std::make_shared>( + this->get_clock(), period, std::move(timer_callback), + this->get_node_base_interface()->get_context()); + this->get_node_timers_interface()->add_timer(timer_, nullptr); +} + +void DummyPerceptionPublisherNode::timerCallback() +{ + // output msgs + autoware_perception_msgs::msg::DetectedObjectsWithFeature output_dynamic_object_msg; + geometry_msgs::msg::PoseStamped output_moved_object_pose; + sensor_msgs::msg::PointCloud2 output_pointcloud_msg; + std_msgs::msg::Header header; + rclcpp::Time current_time = this->now(); + + // avoid terminal contamination. + static rclcpp::Time failed_tf_time = rclcpp::Time(0, 0, RCL_ROS_TIME); + if ((this->now() - failed_tf_time).seconds() < 5.0) { + return; + } + + std::string error; + if (!tf_buffer_.canTransform("base_link", /*src*/ "map", tf2::TimePointZero, &error)) { + failed_tf_time = this->now(); + RCLCPP_WARN_THROTTLE(get_logger(), *get_clock(), 5000, "map->base_link is not available yet"); + return; + } + + tf2::Transform tf_base_link2map; + try { + geometry_msgs::msg::TransformStamped ros_base_link2map; + ros_base_link2map = tf_buffer_.lookupTransform( + /*target*/ "base_link", /*src*/ "map", current_time, rclcpp::Duration::from_seconds(0.5)); + tf2::fromMsg(ros_base_link2map.transform, tf_base_link2map); + } catch (tf2::TransformException & ex) { + RCLCPP_WARN_THROTTLE(get_logger(), *get_clock(), 5000, "%s", ex.what()); + return; + } + + std::vector::Ptr> v_pointcloud; + std::vector delete_idxs; + static std::uniform_real_distribution<> detection_successful_random(0.0, 1.0); + for (size_t i = 0; i < objects_.size(); ++i) { + if (detection_successful_rate_ < detection_successful_random(random_generator_)) { + continue; + } + const double std_dev_x = std::sqrt(objects_.at(i).initial_state.pose_covariance.covariance[0]); + const double std_dev_y = std::sqrt(objects_.at(i).initial_state.pose_covariance.covariance[7]); + const double std_dev_z = std::sqrt(objects_.at(i).initial_state.pose_covariance.covariance[14]); + const double std_dev_yaw = + std::sqrt(objects_.at(i).initial_state.pose_covariance.covariance[35]); + const double move_distance = + objects_.at(i).initial_state.twist_covariance.twist.linear.x * + (current_time.seconds() - rclcpp::Time(objects_.at(i).header.stamp).seconds()); + tf2::Transform tf_object_origin2moved_object; + tf2::Transform tf_map2object_origin; + tf2::Transform tf_map2moved_object; + { + geometry_msgs::msg::Transform ros_object_origin2moved_object; + ros_object_origin2moved_object.translation.x = move_distance; + ros_object_origin2moved_object.rotation.x = 0; + ros_object_origin2moved_object.rotation.y = 0; + ros_object_origin2moved_object.rotation.z = 0; + ros_object_origin2moved_object.rotation.w = 1; + tf2::fromMsg(ros_object_origin2moved_object, tf_object_origin2moved_object); + } + tf2::fromMsg(objects_.at(i).initial_state.pose_covariance.pose, tf_map2object_origin); + tf_map2moved_object = tf_map2object_origin * tf_object_origin2moved_object; + tf2::toMsg(tf_map2moved_object, output_moved_object_pose.pose); + + // pointcloud + pcl::PointCloud::Ptr pointcloud_ptr(new pcl::PointCloud); + createObjectPointcloud( + objects_.at(i).shape.dimensions.x, objects_.at(i).shape.dimensions.y, + objects_.at(i).shape.dimensions.z, std_dev_x, std_dev_y, std_dev_z, + tf_base_link2map * tf_map2moved_object, pointcloud_ptr); + v_pointcloud.push_back(pointcloud_ptr); + + // dynamic object + std::normal_distribution<> x_random(0.0, std_dev_x); + std::normal_distribution<> y_random(0.0, std_dev_y); + std::normal_distribution<> yaw_random(0.0, std_dev_yaw); + tf2::Quaternion noised_quat; + noised_quat.setRPY(0, 0, yaw_random(random_generator_)); + tf2::Transform tf_moved_object2noised_moved_object( + noised_quat, tf2::Vector3(x_random(random_generator_), y_random(random_generator_), 0.0)); + tf2::Transform tf_base_link2noised_moved_object; + tf_base_link2noised_moved_object = + tf_base_link2map * tf_map2moved_object * tf_moved_object2noised_moved_object; + autoware_perception_msgs::msg::DetectedObjectWithFeature feature_object; + feature_object.object.classification.push_back(objects_.at(i).classification); + feature_object.object.kinematics.pose_with_covariance = + objects_.at(i).initial_state.pose_covariance; + feature_object.object.kinematics.twist_with_covariance = + objects_.at(i).initial_state.twist_covariance; + feature_object.object.kinematics.orientation_availability = + autoware_auto_perception_msgs::msg::DetectedObjectKinematics::UNAVAILABLE; + feature_object.object.kinematics.has_twist = false; + tf2::toMsg( + tf_base_link2noised_moved_object, feature_object.object.kinematics.pose_with_covariance.pose); + feature_object.object.shape = objects_.at(i).shape; + pcl::toROSMsg(*pointcloud_ptr, feature_object.feature.cluster); + output_dynamic_object_msg.feature_objects.push_back(feature_object); + + // check delete idx + tf2::Transform tf_base_link2moved_object; + tf_base_link2moved_object = tf_base_link2map * tf_map2moved_object; + double dist = std::sqrt( + tf_base_link2moved_object.getOrigin().x() * tf_base_link2moved_object.getOrigin().x() + + tf_base_link2moved_object.getOrigin().y() * tf_base_link2moved_object.getOrigin().y()); + if (visible_range_ < dist) { + delete_idxs.push_back(i); + } + } + // delete + for (int delete_idx = delete_idxs.size() - 1; 0 <= delete_idx; --delete_idx) { + objects_.erase(objects_.begin() + delete_idxs.at(delete_idx)); + } + + // merge all pointcloud + pcl::PointCloud::Ptr merged_pointcloud_ptr(new pcl::PointCloud); + for (size_t i = 0; i < v_pointcloud.size(); ++i) { + for (size_t j = 0; j < v_pointcloud.at(i)->size(); ++j) { + merged_pointcloud_ptr->push_back(v_pointcloud.at(i)->at(j)); + } + } + // no ground + pcl::toROSMsg(*merged_pointcloud_ptr, output_pointcloud_msg); + + // ray tracing + if (enable_ray_tracing_) { + pcl::PointCloud::Ptr ray_traced_merged_pointcloud_ptr( + new pcl::PointCloud); + pcl::VoxelGridOcclusionEstimation ray_tracing_filter; + ray_tracing_filter.setInputCloud(merged_pointcloud_ptr); + ray_tracing_filter.setLeafSize(0.25, 0.25, 0.25); + ray_tracing_filter.initializeVoxelGrid(); + for (size_t i = 0; i < v_pointcloud.size(); ++i) { + pcl::PointCloud::Ptr ray_traced_pointcloud_ptr( + new pcl::PointCloud); + for (size_t j = 0; j < v_pointcloud.at(i)->size(); ++j) { + Eigen::Vector3i grid_coordinates = ray_tracing_filter.getGridCoordinates( + v_pointcloud.at(i)->at(j).x, v_pointcloud.at(i)->at(j).y, v_pointcloud.at(i)->at(j).z); + int grid_state; + if (ray_tracing_filter.occlusionEstimation(grid_state, grid_coordinates) != 0) { + RCLCPP_ERROR(get_logger(), "ray tracing failed"); + } + if (grid_state == 1) { // occluded + continue; + } else { // not occluded + ray_traced_pointcloud_ptr->push_back(v_pointcloud.at(i)->at(j)); + ray_traced_merged_pointcloud_ptr->push_back(v_pointcloud.at(i)->at(j)); + } + } + pcl::toROSMsg( + *ray_traced_pointcloud_ptr, + output_dynamic_object_msg.feature_objects.at(i).feature.cluster); + output_dynamic_object_msg.feature_objects.at(i).feature.cluster.header.frame_id = "base_link"; + output_dynamic_object_msg.feature_objects.at(i).feature.cluster.header.stamp = current_time; + } + pcl::toROSMsg(*ray_traced_merged_pointcloud_ptr, output_pointcloud_msg); + } + + // create output header + output_moved_object_pose.header.frame_id = "map"; + output_moved_object_pose.header.stamp = current_time; + output_dynamic_object_msg.header.frame_id = "base_link"; + output_dynamic_object_msg.header.stamp = current_time; + output_pointcloud_msg.header.frame_id = "base_link"; + output_pointcloud_msg.header.stamp = current_time; + + // publish + pointcloud_pub_->publish(output_pointcloud_msg); + if (use_object_recognition_) { + detected_object_with_feature_pub_->publish(output_dynamic_object_msg); + } +} + +void DummyPerceptionPublisherNode::createObjectPointcloud( + const double length, const double width, const double height, const double std_dev_x, + const double std_dev_y, const double std_dev_z, const tf2::Transform & tf_base_link2moved_object, + pcl::PointCloud::Ptr & pointcloud_ptr) +{ + std::normal_distribution<> x_random(0.0, std_dev_x); + std::normal_distribution<> y_random(0.0, std_dev_y); + std::normal_distribution<> z_random(0.0, std_dev_z); + auto getBaseLinkTo2DPoint = [tf_base_link2moved_object](double x, double y) -> pcl::PointXYZ { + tf2::Transform tf_moved_object2point; + tf2::Transform tf_base_link2point; + geometry_msgs::msg::Transform ros_moved_object2point; + ros_moved_object2point.translation.x = x; + ros_moved_object2point.translation.y = y; + ros_moved_object2point.translation.z = 0.0; + ros_moved_object2point.rotation.x = 0; + ros_moved_object2point.rotation.y = 0; + ros_moved_object2point.rotation.z = 0; + ros_moved_object2point.rotation.w = 1; + tf2::fromMsg(ros_moved_object2point, tf_moved_object2point); + tf_base_link2point = tf_base_link2moved_object * tf_moved_object2point; + pcl::PointXYZ point; + point.x = tf_base_link2point.getOrigin().x(); + point.y = tf_base_link2point.getOrigin().y(); + point.z = tf_base_link2point.getOrigin().z(); + return point; + }; + const double epsilon = 0.001; + const double step = 0.05; + const double vertical_theta_step = (1.0 / 180.0) * M_PI; + const double vertical_min_theta = (-15.0 / 180.0) * M_PI; + const double vertical_max_theta = (15.0 / 180.0) * M_PI; + const double horizontal_theta_step = (0.1 / 180.0) * M_PI; + const double horizontal_min_theta = (-180.0 / 180.0) * M_PI; + const double horizontal_max_theta = (180.0 / 180.0) * M_PI; + + const double min_z = -1.0 * (height / 2.0) + tf_base_link2moved_object.getOrigin().z(); + const double max_z = 1.0 * (height / 2.0) + tf_base_link2moved_object.getOrigin().z(); + pcl::PointCloud horizontal_candidate_pointcloud; + pcl::PointCloud horizontal_pointcloud; + { + const double y = -1.0 * (width / 2.0); + for (double x = -1.0 * (length / 2.0); x <= ((length / 2.0) + epsilon); x += step) { + horizontal_candidate_pointcloud.push_back(getBaseLinkTo2DPoint(x, y)); + } + } + { + const double y = 1.0 * (width / 2.0); + for (double x = -1.0 * (length / 2.0); x <= ((length / 2.0) + epsilon); x += step) { + horizontal_candidate_pointcloud.push_back(getBaseLinkTo2DPoint(x, y)); + } + } + { + const double x = -1.0 * (length / 2.0); + for (double y = -1.0 * (width / 2.0); y <= ((width / 2.0) + epsilon); y += step) { + horizontal_candidate_pointcloud.push_back(getBaseLinkTo2DPoint(x, y)); + } + } + { + const double x = 1.0 * (length / 2.0); + for (double y = -1.0 * (width / 2.0); y <= ((width / 2.0) + epsilon); y += step) { + horizontal_candidate_pointcloud.push_back(getBaseLinkTo2DPoint(x, y)); + } + } + // 2D ray tracing + size_t ranges_size = + std::ceil((horizontal_max_theta - horizontal_min_theta) / horizontal_theta_step); + std::vector horizontal_ray_traced_2d_pointcloud; + horizontal_ray_traced_2d_pointcloud.assign(ranges_size, std::numeric_limits::infinity()); + const int no_data = -1; + std::vector horizontal_ray_traced_pointcloud_indices; + horizontal_ray_traced_pointcloud_indices.assign(ranges_size, no_data); + for (size_t i = 0; i < horizontal_candidate_pointcloud.points.size(); ++i) { + double angle = + std::atan2(horizontal_candidate_pointcloud.at(i).y, horizontal_candidate_pointcloud.at(i).x); + double range = + std::hypot(horizontal_candidate_pointcloud.at(i).y, horizontal_candidate_pointcloud.at(i).x); + if (angle < horizontal_min_theta || angle > horizontal_max_theta) { + continue; + } + int index = (angle - horizontal_min_theta) / horizontal_theta_step; + if (range < horizontal_ray_traced_2d_pointcloud[index]) { + horizontal_ray_traced_2d_pointcloud[index] = range; + horizontal_ray_traced_pointcloud_indices.at(index) = i; + } + } + for (const auto & pointcloud_index : horizontal_ray_traced_pointcloud_indices) { + if (pointcloud_index != no_data) { + // generate vertical point + horizontal_pointcloud.push_back(horizontal_candidate_pointcloud.at(pointcloud_index)); + const double distance = std::hypot( + horizontal_candidate_pointcloud.at(pointcloud_index).x, + horizontal_candidate_pointcloud.at(pointcloud_index).y); + for (double vertical_theta = vertical_min_theta; + vertical_theta <= vertical_max_theta + epsilon; vertical_theta += vertical_theta_step) { + const double z = distance * std::tan(vertical_theta); + if (min_z <= z && z <= max_z + epsilon) { + pcl::PointXYZ point; + point.x = + horizontal_candidate_pointcloud.at(pointcloud_index).x + x_random(random_generator_); + point.y = + horizontal_candidate_pointcloud.at(pointcloud_index).y + y_random(random_generator_); + point.z = z + z_random(random_generator_); + pointcloud_ptr->push_back(point); + } + } + } + } +} + +void DummyPerceptionPublisherNode::objectCallback( + const dummy_perception_publisher::msg::Object::ConstSharedPtr msg) +{ + switch (msg->action) { + case dummy_perception_publisher::msg::Object::ADD: { + tf2::Transform tf_input2map; + tf2::Transform tf_input2object_origin; + tf2::Transform tf_map2object_origin; + try { + geometry_msgs::msg::TransformStamped ros_input2map; + ros_input2map = tf_buffer_.lookupTransform( + /*target*/ msg->header.frame_id, /*src*/ "map", msg->header.stamp, + rclcpp::Duration::from_seconds(0.5)); + tf2::fromMsg(ros_input2map.transform, tf_input2map); + } catch (tf2::TransformException & ex) { + RCLCPP_WARN_THROTTLE(get_logger(), *get_clock(), 5000, "%s", ex.what()); + return; + } + tf2::fromMsg(msg->initial_state.pose_covariance.pose, tf_input2object_origin); + tf_map2object_origin = tf_input2map.inverse() * tf_input2object_origin; + dummy_perception_publisher::msg::Object object; + object = *msg; + tf2::toMsg(tf_map2object_origin, object.initial_state.pose_covariance.pose); + + // Use base_link Z + geometry_msgs::msg::TransformStamped ros_map2base_link; + try { + ros_map2base_link = tf_buffer_.lookupTransform( + "map", "base_link", rclcpp::Time(0), rclcpp::Duration::from_seconds(0.5)); + object.initial_state.pose_covariance.pose.position.z = + ros_map2base_link.transform.translation.z; + } catch (tf2::TransformException & ex) { + RCLCPP_WARN_SKIPFIRST_THROTTLE(get_logger(), *get_clock(), 5000, "%s", ex.what()); + return; + } + + objects_.push_back(object); + break; + } + case dummy_perception_publisher::msg::Object::DELETE: { + for (size_t i = 0; i < objects_.size(); ++i) { + if (objects_.at(i).id.uuid == msg->id.uuid) { + objects_.erase(objects_.begin() + i); + break; + } + } + break; + } + case dummy_perception_publisher::msg::Object::MODIFY: { + for (size_t i = 0; i < objects_.size(); ++i) { + if (objects_.at(i).id.uuid == msg->id.uuid) { + tf2::Transform tf_input2map; + tf2::Transform tf_input2object_origin; + tf2::Transform tf_map2object_origin; + try { + geometry_msgs::msg::TransformStamped ros_input2map; + ros_input2map = tf_buffer_.lookupTransform( + /*target*/ msg->header.frame_id, /*src*/ "map", msg->header.stamp, + rclcpp::Duration::from_seconds(0.5)); + tf2::fromMsg(ros_input2map.transform, tf_input2map); + } catch (tf2::TransformException & ex) { + RCLCPP_WARN_THROTTLE(get_logger(), *get_clock(), 5000, "%s", ex.what()); + return; + } + tf2::fromMsg(msg->initial_state.pose_covariance.pose, tf_input2object_origin); + tf_map2object_origin = tf_input2map.inverse() * tf_input2object_origin; + dummy_perception_publisher::msg::Object object; + objects_.at(i) = *msg; + tf2::toMsg(tf_map2object_origin, objects_.at(i).initial_state.pose_covariance.pose); + break; + } + } + break; + } + case dummy_perception_publisher::msg::Object::DELETEALL: { + objects_.clear(); + break; + } + } +}