Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

sensor_msgs/Range with ros_gz #586

Open
EGAlberts opened this issue Aug 1, 2024 · 5 comments
Open

sensor_msgs/Range with ros_gz #586

EGAlberts opened this issue Aug 1, 2024 · 5 comments
Labels
help wanted Extra attention is needed

Comments

@EGAlberts
Copy link

Is there support for the sensor_msgs/Range type as there appears to be in Gazebo classic?

I am trying to migrate the following sensor from classic:

 <plugin name="${name}_gazebo_ros_range" filename="libgazebo_ros_ray_sensor.so">
           #<ros>
            <namespace></namespace>
            <remapping>~/out:=${ros_topic}</remapping>
          </ros>
          <output_type>sensor_msgs/Range</output_type>
          <radiation_type>${radiation}</radiation_type>
          <frame_name>${name}_link</frame_name>
</plugin>

As you can see it outputs to sensor_msgs/Range rather than LaserScan, while I believe the use of gpu_lidar as a sensor type as is the new convention creates a corresponding LaserScan gz msg which cannot be converted to Range.

If there is no established way to do so, I can resubmit this issue as a feature request, or attempt to add this myself as a custom sensor.

@azeey
Copy link
Contributor

azeey commented Aug 6, 2024

I don't know of an established way to do this. If it's just a matter of converting LaserScan messages to Range, how about creating a ros_gz bridge for gz.msgs.LaserScan<->sensor_msgs/Range?

@azeey azeey added the help wanted Extra attention is needed label Aug 6, 2024
@azeey azeey removed this from Core development Aug 6, 2024
@EGAlberts
Copy link
Author

Thanks for the reply, it seems the conversion was done here: https://github.com/ros-simulation/gazebo_ros_pkgs/blob/a1954425c3e0045e8591b6516f31edd77c96e7c8/gazebo_ros/include/gazebo_ros/conversions/sensor_msgs.hpp#L291 before, which I suppose can be integrated into the bridge. My only doubt is how to allow the xml element which prior were unique to this usage of the plugin (Like radiation type). Can it somehow be added as a child element of a gpu_lidar sensor without raising errors and then read in? Or should this be specified as an entry in the bridge? I am only asking if there is some established convention, in the mean time I will explore the implications of potential solutions.

@EGAlberts
Copy link
Author

To answer my own question. The only way to include the radiation_type would see the creation of a new gazebo message corresponding to Range which is then converted to Range. This is because the bridge only receives a copy of the gazebo msg which typed LaserScan which does not have a radiation type. I think the more realistic solution is to convert from LaserScan and give the radiation type field a default dummy value, which users should then be wary of. I will create a pull request for my suggested solution of converting LaserScans to Range messages based on the snippet linked in my previous reply.

@azeey
Copy link
Contributor

azeey commented Aug 7, 2024

This is really more of a workaround until we get a Sonar sensor (gazebosim/gz-sensors#19), so I think it would be okay to use a dummy value for radiation type. I only ask that we document this in the README.

@slgrobotics
Copy link

slgrobotics commented Jan 23, 2025

@EGAlberts @azeey - the following code can be used to convert "sensor_msgs/msg/LaserScan" to "sensor_msgs/msg/Range" on the ROS2 side (tested on ROS Jazzy):

import os

from ament_index_python.packages import get_package_share_directory

from launch import LaunchDescription
from launch_ros.actions import Node

def generate_launch_description():

    # Note: radiation_type ULTRASOUND=0 INFRARED=1
    # See https://github.com/ros-tooling/topic_tools/blob/jazzy/topic_tools/topic_tools/relay_field.py

    sonar_F_L_node = Node(
                package='topic_tools',
                executable='relay_field',
                name='sonar_F_L_relay',
                output='screen',
                respawn=True,
                respawn_delay=2.0,
                arguments=['/sonar_F_L_sim', '/sonar_F_L', 'sensor_msgs/msg/Range', \
                            '{ \
                                header: { \
                                        stamp: {sec: m.header.stamp.sec, nanosec: m.header.stamp.nanosec}, \
                                        frame_id: m.header.frame_id \
                                      }, \
                                field_of_view: 0.1, \
                                max_range: m.range_max, \
                                min_range: m.range_min, \
                                radiation_type: 0, \
                                range: "m.ranges[3]", \
                                variance: 0.01 \
                              } \
                         ']
            )

     # Create the launch description and populate
    ld = LaunchDescription()

    # Add the actions to launch all of the navigation nodes
    ld.add_action(sonar_F_L_node)

    return ld

Here are GZ bridge settings:

- ros_topic_name: "sonar_F_L_sim"
  gz_topic_name: "gz_sonar_F_L"
  ros_type_name: "sensor_msgs/msg/LaserScan"
  #ros_type_name: "sensor_msgs/msg/Range"    - this doesn't work. Use topic_tools relay_field, see sonars_sim.launch.py
  gz_type_name: "gz.msgs.LaserScan"
  direction: GZ_TO_ROS

Corresponding xacro code:

    <xacro:property name="sonars_angle_offset" value="${pi/20}"/>

    <joint name="sonar_F_L_joint" type="fixed">
        <parent link="chassis_link"/>
        <child link="sonar_F_L_frame"/>
        <origin xyz="${sonars_offset_x} ${sonars_offset_y} ${sonars_offset_z}" rpy="0 0 ${sonars_angle_offset}"/>
    </joint>

    <!-- the name "sonar_F_L_frame" should correspond to /scan topic header->frame_id as defined in *.launch.py -->
    <link name="sonar_F_L_frame">
        <visual>
            <origin xyz="0 0 0" rpy="0 ${pi/2} 0"/>
            <geometry>
                <cylinder radius="0.01" length="0.01"/>
            </geometry>
            <material name="sensor_blue"/>
        </visual>
    </link>

    <!-- gz model -m dragger -s sonar_F_L_frame -->
    <gazebo reference="sonar_F_L_frame">
        <!-- name must be "sonar_F_L_frame", specifically for sonars. Must reference an actual link in .xacro -->
        <sensor name='sonar_F_L_frame' type='gpu_lidar'>
          <pose>0 0 0 0 0 0</pose>
          <always_on>1</always_on>
          <!-- 6 Hz for LD14 - see "frequency" in dual_ekf_navsat_parameters.yaml -->
          <update_rate>6</update_rate>
          <topic>gz_sonar_F_L</topic>
          <!-- future Gazebo versions might require "<frame_id>" here. Must reference an actual link in .xacro -->
          <gz_frame_id>sonar_F_L_frame</gz_frame_id>
          <frame_id>sonar_F_L_frame</frame_id>
          <ray>
            <scan>
              <horizontal>
                <samples>7</samples>
                <resolution>1</resolution>
                <min_angle>-${pi/40}</min_angle>
                <max_angle>${pi/40}</max_angle>
              </horizontal>
              <vertical>
                <samples>1</samples>
                <resolution>1</resolution>
                <min_angle>0.0</min_angle>
                <max_angle>0.0</max_angle>
              </vertical>
            </scan>
            <range>
              <min>0.3</min>
              <max>3.0</max>
              <resolution>0.01</resolution>
            </range>
            <noise>
              <type>gaussian</type>
              <mean>0</mean>
              <stddev>0.01</stddev>
            </noise>
          </ray>
          <visualize>true</visualize>
        </sensor>
    </gazebo>

I hope this helps with gazebosim/gz-sensors#19 - at least temporarily.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
help wanted Extra attention is needed
Projects
None yet
Development

No branches or pull requests

3 participants