diff --git a/common/tier4_autoware_utils/include/tier4_autoware_utils/ros/published_time_publisher.hpp b/common/tier4_autoware_utils/include/tier4_autoware_utils/ros/published_time_publisher.hpp new file mode 100644 index 0000000000000..12c1f4c8dcd08 --- /dev/null +++ b/common/tier4_autoware_utils/include/tier4_autoware_utils/ros/published_time_publisher.hpp @@ -0,0 +1,106 @@ +// 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 TIER4_AUTOWARE_UTILS__ROS__PUBLISHED_TIME_PUBLISHER_HPP_ +#define TIER4_AUTOWARE_UTILS__ROS__PUBLISHED_TIME_PUBLISHER_HPP_ + +#include + +#include + +#include +#include +#include + +namespace tier4_autoware_utils +{ +using autoware_optional_msgs::msg::PublishedTime; + +struct GidHash +{ + size_t operator()(const rmw_gid_t & gid) const noexcept + { + // Hashing function that computes a hash value for the GID + std::hash hasher; + return hasher(std::string(reinterpret_cast(gid.data), RMW_GID_STORAGE_SIZE)); + } +}; + +struct GidEqual +{ + bool operator()(const rmw_gid_t & lhs, const rmw_gid_t & rhs) const noexcept + { + return std::memcmp(lhs.data, rhs.data, RMW_GID_STORAGE_SIZE) == 0; + } +}; + +class PublishedTimePublisher +{ +public: + static std::unique_ptr create( + rclcpp::Node * node, const std::string & end_name = "/debug/published_time", + const rclcpp::QoS & qos = rclcpp::QoS(1)) + { + const bool use_published_time = node->declare_parameter("use_published_time", false); + if (use_published_time) { + return std::unique_ptr( + new PublishedTimePublisher(node, end_name, qos)); + } else { + return nullptr; + } + } + + void publish( + const rclcpp::PublisherBase::ConstSharedPtr & publisher, const rclcpp::Time & header_stamp) + { + const auto & gid = publisher->get_gid(); + const auto & topic_name = publisher->get_topic_name(); + + // if the publisher is not in the map, create a new publisher for published time + if (publishers_.find(gid) == publishers_.end()) { + publishers_[gid] = node_->create_publisher( + static_cast(topic_name) + end_name_, qos_); + } + + const auto & pub_published_time_ = publishers_[gid]; + + // Check if there are any subscribers, otherwise don't do anything + if (pub_published_time_->get_subscription_count() > 0) { + PublishedTime published_time; + + published_time.header_stamp = header_stamp; + published_time.published_stamp = rclcpp::Clock().now(); + + pub_published_time_->publish(published_time); + } + } + +private: + explicit PublishedTimePublisher( + rclcpp::Node * node, const std::string & end_name, const rclcpp::QoS & qos) + : node_(node), end_name_(end_name), qos_(qos) + { + } + + rclcpp::Node * node_; + std::string end_name_; + rclcpp::QoS qos_; + + // store them for each different publisher of the node + std::unordered_map::SharedPtr, GidHash, GidEqual> + publishers_; +}; +} // namespace tier4_autoware_utils + +#endif // TIER4_AUTOWARE_UTILS__ROS__PUBLISHED_TIME_PUBLISHER_HPP_ \ No newline at end of file diff --git a/common/tier4_autoware_utils/package.xml b/common/tier4_autoware_utils/package.xml index 04a07b88edf4e..d968451b1605b 100644 --- a/common/tier4_autoware_utils/package.xml +++ b/common/tier4_autoware_utils/package.xml @@ -15,6 +15,8 @@ autoware_auto_perception_msgs autoware_auto_planning_msgs autoware_auto_vehicle_msgs + autoware_common_msgs + autoware_optional_msgs builtin_interfaces diagnostic_msgs geometry_msgs