diff --git a/CMakeLists.txt b/CMakeLists.txt index c4f7f7c..0956b02 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -11,6 +11,9 @@ if(NOT CMAKE_CXX_STANDARD) set(CMAKE_CXX_STANDARD 14) endif() +# Suppress all warnings +set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -w") + if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang") add_compile_options(-Wall -Wextra -Wpedantic) endif() @@ -20,11 +23,14 @@ find_package(ament_cmake REQUIRED) find_package(rclcpp REQUIRED) find_package(sensor_msgs REQUIRED) find_package(PkgConfig REQUIRED) +find_package(OpenCV REQUIRED) pkg_check_modules(GST REQUIRED gstreamer-1.0 gstreamer-app-1.0 gstreamer-rtsp-server-1.0 ) +include_directories(${OpenCV_INCLUDE_DIRS}) + file(GLOB SOURCES src/video.cpp) add_executable(image2rtsp src/image2rtsp.cpp ${SOURCES}) @@ -34,7 +40,8 @@ include_directories( ${GST_INCLUDE_DIRS} ) target_link_libraries(image2rtsp - ${GST_LIBRARIES} + ${GST_LIBRARIES} + ${OpenCV_LIBS} ) install(TARGETS diff --git a/README.md b/README.md index 70dee26..a57dff1 100644 --- a/README.md +++ b/README.md @@ -2,18 +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**". -Support for **Compressed images** is added on the *dev* branch. +Currently supported and tested `sensor_msgs::msg::Image` formats: "**rgb8**", "**rgba8**", "**rgb16**", "**rgba16**", "**bgr8**", "**bgra8**", "**bgr16**", "**bgra16**", "**mono8**", "**mono16**", "**yuv422_yuy2**". -(if you need some specific unsupported format, create an issue and i will try to add it as soon as possible) +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 20.04 with ROS2 Foxy, also tested on Ubuntu 22.04 with ROS2 Humble. +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 @@ -30,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/ @@ -84,3 +96,6 @@ cd ~/ros2_ws/ source install/setup.bash ros2 launch image2rtsp rtsp.launch.py ``` +## 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 72cc3ea..2ca4cb0 100644 --- a/config/parameters.yaml +++ b/config/parameters.yaml @@ -1,21 +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) - topic: "/color/image_raw" + 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 28dda0c..b5b7693 100644 --- a/include/image2rtsp.hpp +++ b/include/image2rtsp.hpp @@ -6,6 +6,8 @@ #include #include #include "sensor_msgs/msg/image.hpp" +#include "sensor_msgs/msg/compressed_image.hpp" +#include using namespace std; @@ -15,27 +17,27 @@ 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; GstAppSrc *appsrc; void video_mainloop_start(); 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_; + rclcpp::Subscription::SharedPtr subscription_compressed_; }; static void media_configure(GstRTSPMediaFactory *factory, GstRTSPMedia *media, GstElement **appsrc); diff --git a/package.xml b/package.xml index dd26606..c32b726 100644 --- a/package.xml +++ b/package.xml @@ -11,6 +11,7 @@ rclcpp sensor_msgs + open_CV ament_lint_auto ament_lint_common diff --git a/src/image2rtsp.cpp b/src/image2rtsp.cpp index 822d2a6..58d101f 100644 --- a/src/image2rtsp.cpp +++ b/src/image2rtsp.cpp @@ -9,49 +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("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(); + 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 - 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 { + 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 b7f6f9c..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); } @@ -168,3 +168,43 @@ void Image2rtsp::topic_callback(const sensor_msgs::msg::Image::SharedPtr msg){ gst_app_src_push_buffer(appsrc, buf); } } + +void Image2rtsp::compressed_topic_callback(const sensor_msgs::msg::CompressedImage::SharedPtr msg){ + if (appsrc == NULL) return; + // Decompress the image + cv::Mat img = cv::imdecode(cv::Mat(msg->data), cv::IMREAD_UNCHANGED); + if (img.empty()) { + RCLCPP_ERROR(this->get_logger(), "Failed to decompress image"); + return; + } + + // Determine the GStreamer caps + std::string gst_format; + switch (img.type()) { + 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; + } + + GstCaps *caps = gst_caps_new_simple("video/x-raw", + "format", G_TYPE_STRING, gst_format.c_str(), + "width", G_TYPE_INT, img.cols, + "height", G_TYPE_INT, img.rows, + "framerate", GST_TYPE_FRACTION, framerate, 1, + nullptr); + + // Set caps on appsrc + gst_app_src_set_caps(appsrc, caps); + gst_caps_unref(caps); + + // Create a GstBuffer and fill it with the image data + GstBuffer *buf = gst_buffer_new_allocate(nullptr, img.total() * img.elemSize(), nullptr); + gst_buffer_fill(buf, 0, img.data, img.total() * img.elemSize()); + GST_BUFFER_FLAG_SET(buf, GST_BUFFER_FLAG_LIVE); + + // Push the buffer to GStreamer + gst_app_src_push_buffer(appsrc, buf); +} \ No newline at end of file