Skip to content

ROS2 node for access to Basler camera via Pylon CBaslerUniversalInstantCamera API. Supports high-performance, arbitrary framerate, free-running mode.

License

Notifications You must be signed in to change notification settings

fhwedel-hoe/pylon_instant_camera

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

38 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Experimental ROS2 node for access to Basler camera via pylon CBaslerUniversalInstantCamera API.
Supports USB and GigE cameras. Tested with a Basler daA1280-54uc camera.
Known to work on Ubuntu 22.04 "Jammy" with ROS2 "humble".

Supports low-latency, high-speed, arbitrary framerate, free-running mode.

At time of writing, the official driver does not support "acquisition overlap".
Usage of the software trigger is hard-coded, effectively halving the maximum frame-rate.

Some background information and discussion:

Usage

Topic

If the camera provides pixels in bayer pattern (raw), the default topic name is image_raw.
If the camera provides pixels in RGB (color) or MONO (grayscale), the default topic name is image.

As a Node (default)

Starts a node which publishes image data on topic /image.
The camera will be opened using its default parameters:

ros2 run pylon_instant_camera node

As a Node (advanced)

Starts a node which publishes image data on topic /pylon_camera/image_raw.
The camera will be opened and configured with the feature-set stored in settings.pfs.
Camera calibration data will be loaded from camera_calibration.yaml and published at /pylon_camera/camera_info for image rectification.

ros2 run pylon_instant_camera node --ros-args \
-r __ns:=/pylon_camera \
--param camera_settings_pfs:=settings_bayer.pfs \
--param camera_info_yaml:=camera_calibration.yaml

Within a Composite Node

Launch file based on this example, with software debayer and camera image rectification:

import launch
from launch_ros.actions import ComposableNodeContainer
from launch_ros.descriptions import ComposableNode
def generate_launch_description():
    """Use composition for all image-processing nodes.
    Keeps overhead low since image data can – theoretically – reside in shared memory."""
    image_processing = ComposableNodeContainer(
            name = "container",
            namespace = "pylon_camera_node",
            package = "rclcpp_components",
            executable = "component_container",
            composable_node_descriptions = [
                ComposableNode(
                    name = "pylon_camera",
                    namespace = "pylon_camera_node",
                    package = "pylon_instant_camera",
                    plugin = "pylon_instant_camera::PylonCameraNode",
                    parameters = [{
                        "camera_settings_pfs": "settings.pfs",
                        "camera_info_yaml": "camera_calibration.yaml"
                        }]
                ),
                ComposableNode(
                    package="image_proc",
                    plugin="image_proc::DebayerNode",
                    name="debayer_node",
                    namespace="pylon_camera_node"
                ),
                ComposableNode(
                    package="image_proc",
                    plugin="image_proc::RectifyNode",
                    name="pylon_camera_rectify_color",
                    namespace="pylon_camera_node"
                ),
                ComposableNode(
                    package="image_proc",
                    plugin="image_proc::RectifyNode",
                    name="pylon_camera_rectify_mono",
                    namespace="pylon_camera_node",
                    remappings=[
                        ("image", "image_mono")
                    ],
                )
            ]
    )
    return launch.LaunchDescription([image_processing])

Viewing the unprocessed camera image is possible by manually loading the ShowImage tool like this:

ros2 component load /pylon_camera_node/container image_tools image_tools::ShowImage --node-namespace /pylon_camera_node

Pylon Feature Stream (pfs)

The pylon Viewer included in the pylon SDK stores all the settings in one file of tab-separated key-value-pairs. Set the pixel type like this:

# {05D8C294-F295-4dfb-9D01-096BD04049F4}
# GenApi persistence file (version 3.1.0)
PixelFormat	RGB8

With the pixel format RGB8, you do not need the DebayerNode and remappings mentioned in the example above.

Multiple Cameras

In order to select one of multiple cameras, you may supply one or more parameter to identify the particular camera:

  • serial_number – Can be viewed in the pylon Viewer.
  • user_defined_name – Can be set via the pylon Viewer.
  • ip_address – For ethernet cameras.
  • full_name – For USB cameras. This value may change upon device re-connect. Use with caution.

These values are also displayed in the log.

You may also want to remap the namespace so the output topic will not clash:

ros2 run pylon_instant_camera node --ros-args --param serial_number:="123456789" -r __ns:=/front_camera

Known Issues

Payload Discarded

This warning can be emitted:

Payload data has been discarded. Payload data can be discarded by the camera device if the available bandwidth is insufficient.

This is normal for USB connections and should only happen once after camera start.

Performance

ROS 2 does not offer a true zero-copy communication mechanism – not even via shared memory – see discussion here.

About

ROS2 node for access to Basler camera via Pylon CBaslerUniversalInstantCamera API. Supports high-performance, arbitrary framerate, free-running mode.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published