From c812ce9ada615be4d2d104cfe8f8ac746119afe8 Mon Sep 17 00:00:00 2001 From: Dzmitry Maladzenkau Date: Sat, 31 May 2025 17:27:49 +0200 Subject: [PATCH] refactored gstreamer pipeline --- README.md | 76 ++++++++++++++----------- config/parameters.yaml | 51 +++++++++++------ include/image2rtsp.hpp | 11 ++-- src/image2rtsp.cpp | 122 ++++++++++++++++++++++++++++------------- src/video.cpp | 10 ++-- 5 files changed, 173 insertions(+), 97 deletions(-) diff --git a/README.md b/README.md index 1f9e84a..a57dff1 100644 --- a/README.md +++ b/README.md @@ -2,17 +2,17 @@ This project is a migration from ROS1 to ROS2. The original code was developed by [CircusMonkey](https://github.com/CircusMonkey/ros_rtsp/tree/master). I would like to express my gratitude for his contribution. ## image2rtsp -This project enables the conversion of a selected ROS2 topic of type `sensor_msgs::msg::Image` into an `RTSP` stream, with an anticipated delay of approximately 0,3-0,4s. It also supports usb camera as a direct source. The generated stream can be utilized for various purposes such as remote control, object detection tasks, monitoring and more. +This project enables the conversion of a selected ROS2 topic of type `sensor_msgs::msg::Image` or `sensor_msgs::msg::CompressedImage` into an `RTSP` stream, with an anticipated delay of approximately 0,3-0,4s. It also supports usb camera as a direct source. The generated stream can be utilized for various purposes such as remote control, object detection tasks, monitoring and more. -Currently supported `sensor_msgs::msg::Image` formats: "**rgb8**", "**rgba8**", "**rgb16**", "**rgba16**", "**bgr8**", "**bgra8**", "**bgr16**", "**bgra16**", "**mono8**", "**mono16**", "**yuv422_yuy2**". -**Compressed images** are supported as well. -(if you need some specific unsupported format, create an issue and i will try to add it as soon as possible) +Currently supported and tested `sensor_msgs::msg::Image` formats: "**rgb8**", "**rgba8**", "**rgb16**", "**rgba16**", "**bgr8**", "**bgra8**", "**bgr16**", "**bgra16**", "**mono8**", "**mono16**", "**yuv422_yuy2**". -The development is being carried out on Ubuntu 20.04 with ROS2 Foxy, also tested on Ubuntu 22.04 with ROS2 Humble. +Supported and tested `sensor_msgs::msg::CompressedImage` formats: "**rgb8; jpeg compressed bgr8**". Other formats may work as well with some color scheme deviations. Please open an issue in this case and attach your **ros2 bag**, so i can fix it. If you need some specific unsupported format, create an issue and i will try to add it as soon as possible, but normally it takes pretty long, so dont hesitate to create a PR. + +The development is being carried out on Ubuntu 22.04 with ROS2 Humble. Tested with Intel RealSense d435i. You are reading now the README for a **default** ROS2 package. If you want to use this package written as a ROS2 component, checkout `ros2_component` branch. ## Dependencies -- ROS2 Foxy/Humble +- ROS2 Humble - gstreamer libs: ```bash @@ -29,36 +29,49 @@ sudo apt-get install libgstreamer-plugins-base1.0-dev libgstreamer-plugins-good1 ```bashrc git clone https://github.com/maladzenkau/image2rtsp.git --single-branch ``` - - Check the framerate of the topic to be subscribed: - ```bashrc - ros2 topic hz /someTopic - ``` - Adjust `parameters.yaml` according to your needs: ```bashrc gedit ~/ros2_ws/src/image2rtsp/config/parameters.yaml ``` # Example ROS2 Image topic stream + # If the source is a ros2 topic (default case) + compressed: False + topic: "color/image_raw" + default_pipeline: | + ( appsrc name=imagesrc do-timestamp=true min-latency=0 + max-latency=0 max-bytes=1000 is-live=true ! + videoconvert ! + videoscale ! + video/x-raw, framerate=30/1, width=640, height=480 ! + x264enc tune=zerolatency bitrate=500 key-int-max=30 ! + video/x-h264, profile=baseline ! + rtph264pay name=pay0 pt=96 ) - # If camera serves as a source (switched off by default) - camera: False - source: "v4l2src device=/dev/video0" - - topic: "/color/image_raw" # The ROS2 topic to subscribe to. Dont change, if you use a camera + # Notice: The framerate setting does not affect the RTSP stream — it entirely depends on the ros2 topic frequency. + # It is included in the pipeline and code for structural reasons. You can likely remove it from the pipeline without impacting the package's behavior. + + + # If camera serves as a source + camera: False + camera_pipeline: | + ( v4l2src device=/dev/video0 ! + videoconvert ! + videoscale ! + video/x-raw, framerate=30/1, width=640, height=480 ! + x264enc tune=zerolatency bitrate=500 key-int-max=30 ! + video/x-h264, profile=baseline ! + rtph264pay name=pay0 pt=96 ) + + # Notice: Here the framerate might be set to the camera framerate, otherwise "503 Service Unavailable" error will appear. + + # RTSP setup + mountpoint: "/back" + port: "8554" + local_only: True # True = rtsp://127.0.0.1:portAndMountpoint (The stream is accessible only from the local machine) + # False = rtsp://0.0.0.0:portAndMountpoint (The stream is accessible from the outside) + # For example, to access the stream running on the machine with IP = 192.168.20.20, + # use rtsp://192.186.20.20:portAndMountpoint - # Parameters for both cases - mountpoint: "/back" # Choose the mountpoint for the rtsp stream. - # This will be able to be accessed from rtsp:///portAndMountpoint - bitrate: "500" - framerate: "30" # Make sure that your framerate corresponds to the frequency of a topic you are subscribing to - caps_1: "video/x-raw,framerate=" - capr_2: "/1,width=640,height=480" - # Set the caps to be applied after getting the ROS2 Image and before the x265 encoder. Ignore - # framerate setting here. - port: "8554" - local_only: True # True = rtsp://127.0.0.1:portAndMountpoint (The stream is accessible only from the local machine) - # False = rtsp://0.0.0.0:portAndMountpoint (The stream is accessible from the outside) - # For example, to access the stream running on the machine with IP = 192.168.20.20, - # use rtsp://192.186.20.20:portAndMountpoint - Save your configuration and navigate to `ros2_ws` colcon root, source and build the package: ```bashrc cd ~/ros2_ws/ @@ -83,5 +96,6 @@ cd ~/ros2_ws/ source install/setup.bash ros2 launch image2rtsp rtsp.launch.py ``` -## To Do -- Add compressed images formats (jpeg, png) +## Note + +- The YAML configuration allows you to fully customize the pipeline according to your needs. This package does not provide any built-in acceleration. As its stability has not been validated across a wide range of Linux systems using advanced hardware or software techniques, support for such configurations is left to the user. There are no plans to update the package to support GPU/CPU acceleration. Please do not open issues related to software/hardware acceleration if they are directly related to the GStreamer pipeline itself. diff --git a/config/parameters.yaml b/config/parameters.yaml index 8020cb1..2ca4cb0 100644 --- a/config/parameters.yaml +++ b/config/parameters.yaml @@ -1,22 +1,39 @@ /image2rtsp: ros__parameters: - # If camera serves as a source - camera: False - source: "v4l2src device=/dev/video0" - # If the source is a ros2 topic (default case) - compressed: True - topic: "/image_raw/compressed" + compressed: False + topic: "color/image_raw" + default_pipeline: | + ( appsrc name=imagesrc do-timestamp=true min-latency=0 max-latency=0 max-bytes=1000 is-live=true ! + videoconvert ! + videoscale ! + video/x-raw, framerate=30/1, width=640, height=480 ! + x264enc tune=zerolatency bitrate=500 key-int-max=30 ! + video/x-h264, profile=baseline ! + rtph264pay name=pay0 pt=96 ) - # General setup - mountpoint: "/back" - bitrate: "500" - framerate: "30" - caps_1: "video/x-raw, framerate =" - caps_2: "/1,width=640,height=480" - port: "8554" - local_only: True # True = rtsp://127.0.0.1:portAndMountpoint (The stream is accessible only from the local machine) - # False = rtsp://0.0.0.0:portAndMountpoint (The stream is accessible from the outside) - # For example, to access the stream running on the machine with IP = 192.168.20.20, - # use rtsp://192.186.20.20:portAndMountpoint + # Notice: The framerate setting does not affect the RTSP stream — it entirely depends on the ros2 topic frequency. + # It is included in the pipeline and code for structural reasons. You can likely remove it from the pipeline without impacting the package's behavior. + + + # If camera serves as a source + camera: False + camera_pipeline: | + ( v4l2src device=/dev/video0 ! + videoconvert ! + videoscale ! + video/x-raw, framerate=30/1, width=640, height=480 ! + x264enc tune=zerolatency bitrate=500 key-int-max=30 ! + video/x-h264, profile=baseline ! + rtph264pay name=pay0 pt=96 ) + + # Notice: Here the framerate might be set to the camera framerate, otherwise "503 Service Unavailable" error will appear. + + # RTSP setup + mountpoint: "/back" + port: "8554" + local_only: True # True = rtsp://127.0.0.1:portAndMountpoint (The stream is accessible only from the local machine) + # False = rtsp://0.0.0.0:portAndMountpoint (The stream is accessible from the outside) + # For example, to access the stream running on the machine with IP = 192.168.20.20, + # use rtsp://192.186.20.20:portAndMountpoint diff --git a/include/image2rtsp.hpp b/include/image2rtsp.hpp index 00dfb58..b5b7693 100644 --- a/include/image2rtsp.hpp +++ b/include/image2rtsp.hpp @@ -17,17 +17,13 @@ public: GstRTSPServer *rtsp_server; private: - string source; string topic; string mountpoint; - string bitrate; - string framerate; - string caps_1; - string caps_2; string port; string pipeline; - string pipeline_head; - string pipeline_tail; + string default_pipeline; + string camera_pipeline; + uint framerate; bool local_only; bool camera; bool compressed; @@ -37,6 +33,7 @@ private: void rtsp_server_add_url(const char *url, const char *sPipeline, GstElement **appsrc); void topic_callback(const sensor_msgs::msg::Image::SharedPtr msg); void compressed_topic_callback(const sensor_msgs::msg::CompressedImage::SharedPtr msg); + uint extract_framerate(const std::string& pipeline, uint default_framerate); GstRTSPServer *rtsp_server_create(const string &port, const bool local_only); GstCaps *gst_caps_new_from_image(const sensor_msgs::msg::Image::SharedPtr &msg); rclcpp::Subscription::SharedPtr subscription_; diff --git a/src/image2rtsp.cpp b/src/image2rtsp.cpp index 150e640..58d101f 100644 --- a/src/image2rtsp.cpp +++ b/src/image2rtsp.cpp @@ -9,56 +9,104 @@ using std::placeholders::_1; Image2rtsp::Image2rtsp() : Node("image2rtsp"){ // Declare and get the parameters - this->declare_parameter("source", "v4l2src device=/dev/video0"); - this->declare_parameter("topic", "/color/image_raw"); - this->declare_parameter("mountpoint", "/back"); - this->declare_parameter("bitrate", "500"); - this->declare_parameter("framerate", "30"); - this->declare_parameter("caps_1", "video/x-raw, framerate ="); - this->declare_parameter("caps_2", "/1,width=640,height=480"); - this->declare_parameter("port", "8554"); - this->declare_parameter("local_only", true); - this->declare_parameter("camera", false); - this->declare_parameter("compressed", false); + this->declare_parameter("topic", "/color/image_raw"); + this->declare_parameter("mountpoint", "/back"); + this->declare_parameter("port", "8554"); + this->declare_parameter("local_only", true); + this->declare_parameter("camera", false); + this->declare_parameter("compressed", false); - source = this->get_parameter("source").as_string(); - topic = this->get_parameter("topic").as_string(); - mountpoint = this->get_parameter("mountpoint").as_string(); - bitrate = this->get_parameter("bitrate").as_string(); - framerate = this->get_parameter("framerate").as_string(); - caps_1 = this->get_parameter("caps_1").as_string(); - caps_2 = this->get_parameter("caps_2").as_string(); - port = this->get_parameter("port").as_string(); - local_only = this->get_parameter("local_only").as_bool(); - camera = this->get_parameter("camera").as_bool(); - compressed = this->get_parameter("compressed").as_bool(); + this->declare_parameter("default_pipeline", R"( + ( appsrc name=imagesrc do-timestamp=true min-latency=0 max-latency=0 max-bytes=1000 is-live=true ! + videoconvert ! + videoscale ! + video/x-raw, framerate=30/1, width=640, height=480 ! + x264enc tune=zerolatency bitrate=500 key-int-max=30 ! + video/x-h264, profile=baseline ! + rtph264pay name=pay0 pt=96 ) + )"); + + this->declare_parameter("camera_pipeline", R"( + ( v4l2src device=/dev/video0 ! + videoconvert ! + videoscale ! + video/x-raw, framerate=30/1, width=640, height=480 ! + x264enc tune=zerolatency bitrate=500 key-int-max=30 ! + video/x-h264, profile=baseline ! + rtph264pay name=pay0 pt=96 ) + )"); + + topic = this->get_parameter("topic").as_string(); + mountpoint = this->get_parameter("mountpoint").as_string(); + port = this->get_parameter("port").as_string(); + local_only = this->get_parameter("local_only").as_bool(); + camera = this->get_parameter("camera").as_bool(); + compressed = this->get_parameter("compressed").as_bool(); + default_pipeline = this->get_parameter("default_pipeline").as_string(); + camera_pipeline = this->get_parameter("camera_pipeline").as_string(); // Start the subscription - if (compressed == false){ - subscription_ = this->create_subscription(topic, 10, std::bind(&Image2rtsp::topic_callback, this, _1)); + if (camera == false){ + if (compressed == false){ + subscription_ = this->create_subscription(topic, 10, std::bind(&Image2rtsp::topic_callback, this, _1)); + RCLCPP_INFO(this->get_logger(), "Subscribing to sensor_msgs::msg::Image"); + } + else { + subscription_compressed_ = this->create_subscription(topic, 10, std::bind(&Image2rtsp::compressed_topic_callback, this, _1)); + RCLCPP_INFO(this->get_logger(), "Subscribing to sensor_msgs::msg::CompressedImage"); + } } - else{ - subscription_compressed_ = this->create_subscription(topic, 10, std::bind(&Image2rtsp::compressed_topic_callback, this, _1)); + else { + RCLCPP_INFO(this->get_logger(), "Trying to access camera device"); } // Start the RTSP server video_mainloop_start(); rtsp_server = rtsp_server_create(port, local_only); appsrc = NULL; - // Setup the pipeline - pipeline_tail = "key-int-max=30 ! video/x-h264, profile=baseline ! rtph264pay name=pay0 pt=96 )"; - if (camera == false){ - pipeline_head = "( appsrc name=imagesrc do-timestamp=true min-latency=0 max-latency=0 max-bytes=1000 is-live=true ! videoconvert ! videoscale ! "; - pipeline = pipeline_head + caps_1 + framerate + caps_2 + " ! x264enc tune=zerolatency bitrate=" + bitrate + pipeline_tail; - rtsp_server_add_url(mountpoint.c_str(), pipeline.c_str(), (GstElement **)&(appsrc)); - } - else { - pipeline = "( " + source + " ! videoconvert ! videoscale ! " + caps_1 + framerate + caps_2 + " ! x264enc tune=zerolatency bitrate=" + bitrate + pipeline_tail; - rtsp_server_add_url(mountpoint.c_str(), pipeline.c_str(), NULL); - } + + pipeline = camera ? camera_pipeline : default_pipeline; + framerate = extract_framerate(pipeline, 30); + rtsp_server_add_url(mountpoint.c_str(), pipeline.c_str(), camera ? nullptr : (GstElement **)&appsrc); + RCLCPP_INFO(this->get_logger(), "Stream available at rtsp://%s:%s%s", gst_rtsp_server_get_address(rtsp_server), port.c_str(), mountpoint.c_str()); } +uint Image2rtsp::extract_framerate(const std::string& pipeline, uint default_framerate = 30) { + std::string search_str = "framerate="; + size_t pos = pipeline.find(search_str); + if (pos == std::string::npos) { + RCLCPP_WARN(this->get_logger(), "Framerate not found in pipeline, using default: %d", default_framerate); + return default_framerate; + } + + pos += search_str.length(); + + size_t end_pos = pipeline.find_first_of("/,", pos); + if (end_pos == std::string::npos) { + RCLCPP_WARN(this->get_logger(), "Invalid framerate format in pipeline, using default: %d", default_framerate); + return default_framerate; + } + + std::string framerate_str = pipeline.substr(pos, end_pos - pos); + + framerate_str.erase(0, framerate_str.find_first_not_of(" \t")); + framerate_str.erase(framerate_str.find_last_not_of(" \t") + 1); + + try { + uint framerate = std::stoi(framerate_str); + if (framerate <= 0) { + RCLCPP_WARN(this->get_logger(), "Invalid framerate value %d, using default: %d", framerate, default_framerate); + return default_framerate; + } + RCLCPP_INFO(this->get_logger(), "Using set framerate %d", framerate); + return framerate; + } catch (const std::exception& e) { + RCLCPP_WARN(this->get_logger(), "Failed to parse framerate '%s', using default: %d", framerate_str.c_str(), default_framerate); + return default_framerate; + } +} + int main(int argc, char *argv[]){ rclcpp::init(argc, argv); auto node = std::make_shared(); diff --git a/src/video.cpp b/src/video.cpp index 9e50d21..bde96c5 100644 --- a/src/video.cpp +++ b/src/video.cpp @@ -132,7 +132,7 @@ GstCaps *Image2rtsp::gst_caps_new_from_image(const sensor_msgs::msg::Image::Shar "format", G_TYPE_STRING, format->second.c_str(), "width", G_TYPE_INT, msg->width, "height", G_TYPE_INT, msg->height, - "framerate", GST_TYPE_FRACTION, stoi(framerate), 1, + "framerate", GST_TYPE_FRACTION, framerate, 1, nullptr); } @@ -181,9 +181,9 @@ void Image2rtsp::compressed_topic_callback(const sensor_msgs::msg::CompressedIma // Determine the GStreamer caps std::string gst_format; switch (img.type()) { - case CV_8UC3: gst_format = "RGB"; break; // RGB images - case CV_8UC4: gst_format = "RGBA"; break; // RGBA images - case CV_8UC1: gst_format = "GRAY8"; break; // Grayscale images + case CV_8UC3: gst_format = "BGR"; break; // BGR images + case CV_8UC4: gst_format = "RGBA"; break; // RGBA images + case CV_8UC1: gst_format = "GRAY8"; break; // Grayscale images default: RCLCPP_ERROR(this->get_logger(), "Unsupported image type"); return; @@ -193,7 +193,7 @@ void Image2rtsp::compressed_topic_callback(const sensor_msgs::msg::CompressedIma "format", G_TYPE_STRING, gst_format.c_str(), "width", G_TYPE_INT, img.cols, "height", G_TYPE_INT, img.rows, - "framerate", GST_TYPE_FRACTION, stoi(framerate), 1, + "framerate", GST_TYPE_FRACTION, framerate, 1, nullptr); // Set caps on appsrc