diff --git a/.gitmodules b/.gitmodules index d025b6b..50965c3 100644 --- a/.gitmodules +++ b/.gitmodules @@ -1,6 +1,3 @@ [submodule "rmw_zenoh"] path = rmw_zenoh url = git@github.com:ros2/rmw_zenoh.git -[submodule "ouster-ros"] - path = ouster-ros - url = git@github.com:ouster-lidar/ouster-ros.git diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..56cf85c --- /dev/null +++ b/Dockerfile @@ -0,0 +1,42 @@ +FROM ros:jazzy-ros-base + +ENV ROS_WS=/ros2 +WORKDIR $ROS_WS + +COPY rmw_zenoh src/rmw_zenoh +COPY meta src/meta + +RUN apt-get update &&\ + rosdep update &&\ + rosdep install --from-paths src --ignore-src -y &&\ + rm -rf /var/lib/apt/lists/* + +SHELL ["/bin/bash", "-c"] + +RUN source /opt/ros/$ROS_DISTRO/setup.bash &&\ + colcon build \ + --event-handlers console_direct+ \ + --parallel-workers $(nproc) \ + --cmake-args -DCMAKE_BUILD_TYPE=Release + +ARG INSTALL_DESKTOP=false + +RUN if [ "$INSTALL_DESKTOP" = "true" ]; then \ + apt-get update &&\ + apt-get install -y ros-${ROS_DISTRO}-desktop &&\ + rm -rf /var/lib/apt/lists/*; \ + fi + +COPY ouster_simulator src/ouster_simulator + +RUN source /opt/ros/$ROS_DISTRO/setup.bash &&\ + colcon build \ + --event-handlers console_direct+ \ + --parallel-workers $(nproc) \ + --cmake-args -DCMAKE_BUILD_TYPE=Release + +ENV ROS_DOMAIN_ID=10 +ENV RMW_IMPLEMENTATION=rmw_zenoh_cpp +COPY entrypoint.sh . +ENTRYPOINT ["./entrypoint.sh"] + diff --git a/Dockerfile.jazzy.ouster b/Dockerfile.jazzy.ouster deleted file mode 100644 index bfe0a3a..0000000 --- a/Dockerfile.jazzy.ouster +++ /dev/null @@ -1,21 +0,0 @@ -FROM jazzy:zenoh - -ENV RMW_IMPLEMENTATION=rmw_zenoh_cpp -WORKDIR $ROS_WS - -COPY ouster-ros src/ouster-ros - -RUN apt-get update &&\ - rosdep update &&\ - rosdep install --from-paths src --ignore-src -y &&\ - rm -rf /var/lib/apt/lists/* - -SHELL ["/bin/bash", "-c"] -RUN source install/setup.bash &&\ - colcon build \ - --event-handlers console_direct+ \ - --parallel-workers $(nproc) \ - --cmake-args -DCMAKE_BUILD_TYPE=Release -DBUILD_VIZ=OFF -DBUILD_OSF=OFF -DBUILD_EXAMPLES=OFF - -COPY entrypoint.ouster.sh entrypoint.sh -ENTRYPOINT ["./entrypoint.sh"] diff --git a/Dockerfile.jazzy.zenoh b/Dockerfile.jazzy.zenoh deleted file mode 100644 index 878c259..0000000 --- a/Dockerfile.jazzy.zenoh +++ /dev/null @@ -1,23 +0,0 @@ -FROM ros:jazzy-ros-base - -ENV ROS_WS=/ros2 -WORKDIR $ROS_WS - -COPY rmw_zenoh src/rmw_zenoh -COPY meta src/meta - -RUN apt-get update &&\ - rosdep update &&\ - rosdep install --from-paths src --ignore-src -y &&\ - rm -rf /var/lib/apt/lists/* - -SHELL ["/bin/bash", "-c"] -RUN source /opt/ros/$ROS_DISTRO/setup.bash &&\ - colcon build \ - --event-handlers console_direct+ \ - --parallel-workers $(nproc) \ - --cmake-args -DCMAKE_BUILD_TYPE=Release - -COPY entrypoint.ros2.sh entrypoint.sh -ENTRYPOINT ["./entrypoint.sh"] - diff --git a/entrypoint.ouster.sh b/entrypoint.ouster.sh deleted file mode 100755 index f3277c8..0000000 --- a/entrypoint.ouster.sh +++ /dev/null @@ -1,64 +0,0 @@ -#!/bin/bash -set -e - -# setup ros2 environment -source "/opt/ros/${ROS_DISTRO}/setup.bash" -[[ -f "${ROS_WS}/install/setup.bash" ]] && source "${ROS_WS}/install/setup.bash" - -# ROS information -echo 'ROS distro: ' $ROS_DISTRO - -if [ -n "$RMW_IMPLEMENTATION" ]; then - echo 'DDS middleware: ' $RMW_IMPLEMENTATION; -fi - -if [ -n "$ROS_DOMAIN_ID" ]; then - echo 'ROS Domain ID: ' $ROS_DOMAIN_ID; -fi - -echo "---------------------" -echo "Checking network settings" - -# Define the expected values -EXPECTED_RMEM_MAX=2147483647 -EXPECTED_IPFRAG_TIME=3 -EXPECTED_IPFRAG_HIGH_THRESH=1342177280 - -# Initialize a flag to track if any parameter is incorrectly set -ALL_VALID=true - -# Function to check and report parameter -check_parameter() { - PARAM_NAME=$1 - EXPECTED_VALUE=$2 - CURRENT_VALUE=$(sysctl -n $PARAM_NAME) - - if [ "$CURRENT_VALUE" -eq "$EXPECTED_VALUE" ]; then - echo "✔ $PARAM_NAME = $EXPECTED_VALUE" - else - echo "✘ $PARAM_NAME is incorrectly set to $CURRENT_VALUE (expected $EXPECTED_VALUE)" - echo "To set it correctly, run:" - echo " sudo sysctl -w $PARAM_NAME=$EXPECTED_VALUE" - echo "To make this change permanent, add the following line to /etc/sysctl.conf or a file in /etc/sysctl.d/:" - echo " $PARAM_NAME=$EXPECTED_VALUE" - echo - ALL_VALID=false - fi -} - -# Check each parameter -check_parameter net.core.rmem_max $EXPECTED_RMEM_MAX -check_parameter net.ipv4.ipfrag_time $EXPECTED_IPFRAG_TIME -check_parameter net.ipv4.ipfrag_high_thresh $EXPECTED_IPFRAG_HIGH_THRESH - -# Exit with an error code if any parameter is incorrect -if [ "$ALL_VALID" = false ]; then - echo "Error: One or more network parameters are incorrectly set." - exit 1 -fi - -echo "---------------------" - - -exec "$@" - diff --git a/entrypoint.ros2.sh b/entrypoint.sh similarity index 100% rename from entrypoint.ros2.sh rename to entrypoint.sh diff --git a/meta/package.xml b/meta/package.xml index 485e3c3..441d16d 100644 --- a/meta/package.xml +++ b/meta/package.xml @@ -11,7 +11,6 @@ ament_cmake demo_nodes_cpp - desktop foxglove_bridge diff --git a/ouster-ros b/ouster-ros deleted file mode 160000 index e4dd8bf..0000000 --- a/ouster-ros +++ /dev/null @@ -1 +0,0 @@ -Subproject commit e4dd8bf4bb1d66ec6272344cf0582d0c3f9d340e diff --git a/ouster_simulator/CMakeLists.txt b/ouster_simulator/CMakeLists.txt new file mode 100644 index 0000000..5ad36e1 --- /dev/null +++ b/ouster_simulator/CMakeLists.txt @@ -0,0 +1,30 @@ +cmake_minimum_required(VERSION 3.8) +project(ouster_simulator) + +set(CMAKE_CXX_STANDARD 17) + +find_package(ament_cmake REQUIRED) +find_package(rclcpp REQUIRED) +find_package(sensor_msgs REQUIRED) + +add_executable(ouster_publisher_node src/ouster_publisher_node.cpp) +ament_target_dependencies( + ouster_publisher_node + rclcpp + sensor_msgs +) + +add_executable(ouster_subscriber_node src/ouster_subscriber_node.cpp) +ament_target_dependencies( + ouster_subscriber_node + rclcpp + sensor_msgs +) + +install( + TARGETS ouster_publisher_node ouster_subscriber_node + DESTINATION lib/${PROJECT_NAME} +) + +ament_package() + diff --git a/ouster_simulator/package.xml b/ouster_simulator/package.xml new file mode 100644 index 0000000..3ae6d41 --- /dev/null +++ b/ouster_simulator/package.xml @@ -0,0 +1,17 @@ + + + ouster_simulator + 0.0.1 + Example package that publishes/subscibes pointclouds in ouster format + Your Name + MIT + ament_cmake + rclcpp + sensor_msgs + rclcpp + sensor_msgs + + + ament_cmake + + diff --git a/ouster_simulator/src/ouster_publisher_node.cpp b/ouster_simulator/src/ouster_publisher_node.cpp new file mode 100644 index 0000000..517ba34 --- /dev/null +++ b/ouster_simulator/src/ouster_publisher_node.cpp @@ -0,0 +1,198 @@ +#include +#include +#include +#include +#include + +// ROS 2 +#include "rclcpp/rclcpp.hpp" +#include "sensor_msgs/msg/point_cloud2.hpp" +#include "sensor_msgs/msg/point_field.hpp" +#include "sensor_msgs/point_cloud2_iterator.hpp" + +using namespace std::chrono_literals; + +/** + * @brief A ROS 2 node that publishes a 128×1024 point cloud using + * a custom layout matching the 'ouster_ros::Point' fields: + * + * x (float), y (float), z (float), + * intensity (float), + * t (uint32_t), + * reflectivity (uint16_t), + * ring (uint16_t), + * ambient (uint16_t), + * range (uint32_t). + * + * see https://github.com/ouster-lidar/ouster-ros/blob/ros2/ouster-ros/include/ouster_ros/os_point.h + */ +class OusterPublisher : public rclcpp::Node +{ +public: + OusterPublisher() + : Node("ouster_publisher") + { + publisher_ = this->create_publisher( + "/data", 10); + + timer_ = this->create_wall_timer( + 50ms, std::bind(&OusterPublisher::onTimer, this)); + + RCLCPP_INFO(this->get_logger(), "OusterPublisher node started."); + } + +private: + void onTimer() + { + auto cloud_msg = generatePointCloud(); + publisher_->publish(cloud_msg); + RCLCPP_INFO(this->get_logger(), "Published."); + } + + sensor_msgs::msg::PointCloud2 generatePointCloud() + { + // Dimensions + static constexpr uint32_t HEIGHT = 128; + static constexpr uint32_t WIDTH = 1024; + + // We have 9 fields, totalling 32 bytes per point: + // offset | name | type | bytes + // -------+---------------+-----------+------ + // 0 | x | float32 | 4 + // 4 | y | float32 | 4 + // 8 | z | float32 | 4 + // 12 | intensity | float32 | 4 + // 16 | t | uint32 | 4 + // 20 | reflectivity | uint16 | 2 + // 22 | ring | uint16 | 2 + // 24 | ambient | uint16 | 2 + // 26 | (padding) | uint16 | 2 (for alignment) + // 28 | range | uint32 | 4 + // + // total per point: 32 bytes + // + // Because we need alignment for the 'range' field, we introduce 2 bytes + // of padding after 'ambient' to make 'range' start at a multiple of 4. + + static float total_iterations = 0.0; + + sensor_msgs::msg::PointCloud2 cloud_msg; + cloud_msg.header.stamp = now(); + cloud_msg.header.frame_id = "os_lidar"; + cloud_msg.height = HEIGHT; + cloud_msg.width = WIDTH; + cloud_msg.is_bigendian = false; + cloud_msg.is_dense = false; + + cloud_msg.fields.resize(9); + + cloud_msg.fields[0].name = "x"; + cloud_msg.fields[0].offset = 0; + cloud_msg.fields[0].datatype = sensor_msgs::msg::PointField::FLOAT32; + cloud_msg.fields[0].count = 1; + + cloud_msg.fields[1].name = "y"; + cloud_msg.fields[1].offset = 4; + cloud_msg.fields[1].datatype = sensor_msgs::msg::PointField::FLOAT32; + cloud_msg.fields[1].count = 1; + + cloud_msg.fields[2].name = "z"; + cloud_msg.fields[2].offset = 8; + cloud_msg.fields[2].datatype = sensor_msgs::msg::PointField::FLOAT32; + cloud_msg.fields[2].count = 1; + + cloud_msg.fields[3].name = "intensity"; + cloud_msg.fields[3].offset = 12; + cloud_msg.fields[3].datatype = sensor_msgs::msg::PointField::FLOAT32; + cloud_msg.fields[3].count = 1; + + cloud_msg.fields[4].name = "t"; + cloud_msg.fields[4].offset = 16; + cloud_msg.fields[4].datatype = sensor_msgs::msg::PointField::UINT32; + cloud_msg.fields[4].count = 1; + + cloud_msg.fields[5].name = "reflectivity"; + cloud_msg.fields[5].offset = 20; + cloud_msg.fields[5].datatype = sensor_msgs::msg::PointField::UINT16; + cloud_msg.fields[5].count = 1; + + cloud_msg.fields[6].name = "ring"; + cloud_msg.fields[6].offset = 22; + cloud_msg.fields[6].datatype = sensor_msgs::msg::PointField::UINT16; + cloud_msg.fields[6].count = 1; + + cloud_msg.fields[7].name = "ambient"; + cloud_msg.fields[7].offset = 24; + cloud_msg.fields[7].datatype = sensor_msgs::msg::PointField::UINT16; + cloud_msg.fields[7].count = 1; + + cloud_msg.fields[8].name = "range"; + cloud_msg.fields[8].offset = 28; + cloud_msg.fields[8].datatype = sensor_msgs::msg::PointField::UINT32; + cloud_msg.fields[8].count = 1; + + cloud_msg.point_step = 32; + cloud_msg.row_step = cloud_msg.point_step * cloud_msg.width; + cloud_msg.data.resize(cloud_msg.row_step * cloud_msg.height); + + sensor_msgs::PointCloud2Iterator iter_x(cloud_msg, "x"); + sensor_msgs::PointCloud2Iterator iter_y(cloud_msg, "y"); + sensor_msgs::PointCloud2Iterator iter_z(cloud_msg, "z"); + sensor_msgs::PointCloud2Iterator iter_intensity(cloud_msg, "intensity"); + sensor_msgs::PointCloud2Iterator iter_t(cloud_msg, "t"); + sensor_msgs::PointCloud2Iterator iter_reflectivity(cloud_msg, "reflectivity"); + sensor_msgs::PointCloud2Iterator iter_ring(cloud_msg, "ring"); + sensor_msgs::PointCloud2Iterator iter_ambient(cloud_msg, "ambient"); + sensor_msgs::PointCloud2Iterator iter_range(cloud_msg, "range"); + + // Fill the point cloud in row-major order + // Simple pattern: + // x = (float) col + // y = sin(col / 50.0 + row) + // z = (float) row + // intensity = row + col + // t = synthetic timestamp + // reflectivity = row + // ring = col + // ambient = row*col mod 65535 + // range = (col * 100 + row) + for (uint32_t row = 0; row < HEIGHT; ++row) + { + for (uint32_t col = 0; col < WIDTH; ++col) + { + *iter_x = static_cast(col); + *iter_y = std::sin(static_cast(col) / 50.0f + row); + *iter_z = static_cast(row) + 0.00001f * total_iterations; // static offset to slowly move cloud towards the sky + + *iter_intensity = static_cast(row + col); + + *iter_t = static_cast((row * 100000) + col); // pseudorandom + + *iter_reflectivity = static_cast(row); + *iter_ring = static_cast(col % 65535); + *iter_ambient = static_cast((row * col) % 65535); + + *iter_range = static_cast((col * 100) + row); + + ++iter_x; ++iter_y; ++iter_z; ++iter_intensity; + ++iter_t; + ++iter_reflectivity; ++iter_ring; ++iter_ambient; + ++iter_range; + total_iterations++; + } + } + + return cloud_msg; + } + + rclcpp::Publisher::SharedPtr publisher_; + rclcpp::TimerBase::SharedPtr timer_; +}; + +int main(int argc, char** argv) +{ + rclcpp::init(argc, argv); + rclcpp::spin(std::make_shared()); + rclcpp::shutdown(); + return 0; +} diff --git a/ouster_simulator/src/ouster_subscriber_node.cpp b/ouster_simulator/src/ouster_subscriber_node.cpp new file mode 100644 index 0000000..fe6bedb --- /dev/null +++ b/ouster_simulator/src/ouster_subscriber_node.cpp @@ -0,0 +1,46 @@ +#include +#include +#include +#include + +class OusterSubscriber : public rclcpp::Node +{ +public: + OusterSubscriber() + : Node("ouster_subscriber"), first_message_received_(false) + { + subscription_ = this->create_subscription("/data", 10, std::bind(&OusterSubscriber::topicCallback, this, std::placeholders::_1)); + } + +private: + void topicCallback(const sensor_msgs::msg::PointCloud2::SharedPtr msg) + { + rclcpp::Time current_time = this->now(); + + if (first_message_received_) { + double delta_seconds = (current_time - prev_time_).seconds(); + if (delta_seconds > 0.0) { + double frequency = 1.0 / delta_seconds; + RCLCPP_INFO(this->get_logger(), "Got data at frequency: %.2f Hz", frequency); + } + } else { + first_message_received_ = true; + } + + prev_time_ = current_time; + } + + rclcpp::Subscription::SharedPtr subscription_; + rclcpp::Time prev_time_; + bool first_message_received_; +}; + +int main(int argc, char ** argv) +{ + rclcpp::init(argc, argv); + auto node = std::make_shared(); + rclcpp::spin(node); + rclcpp::shutdown(); + return 0; +} +