20. Sensors
20.1. Lidar Sensor
AGX Dynamics for Unreal offers the ability to simulate many types of real world Lidar sensors, as well as custom Lidar sensors. Hardware accelerated ray tracing is used to enable good performance with high output rates. Output data such as Position, Intensity, Normal vector and many others are available and can be configured by the user. Custom ray patterns are also supported.
The three main parts that needs to be set up to simulate Lidar are: Sensor Environment, Lidar Sensor Component and Lidar Output, each described in more details in subsections below.
The main steps needed to simulate Lidar are:
Add a Sensor Environment Actor to the Level.
Add a Lidar Sensor Component to a Blueprint and configure it in the Details Panel.
Add one or more Lidar Outputs to the Lidar Sensor Component in the Event Graph of the Blueprint.
Instantiate the Blueprint in the Level and register the Lidar Sensor Component in the Sensor Environment Actors Details Panel.
These steps are explained in more detail in the subsections below.
Try out the Lidar
level, included in the Tutorial Hallways Demo Project
20.1.1. Sensor Environment
The Sensor Environment Actor is responsible for keeping track of objects that should be detectable by a lidar, as well as any active Lidar Sensor Component in the Level.
That means that everything that should be detectable by a Lidar Sensor Component needs to be added to the Sensor Environment.
This can be done automatically by enabling the Auto Add Objects
property in the Details Panel:
For more details about the Auto Add Objects
property, see Auto Add Objects.
Objects can only be added to a Sensor Environment during Play.
If more granular control over which objects in the Level should be added to the Sensor Environment, the Auto Add Objects
property can be left unchecked, and objects can be added manually by calling any of the Add
functions exposed by the Sensor Environment Actor.
To access these Add
functions from a Blueprint, a Blueprint with parent class Sensor Environment Actor can be used:
When a Static Mesh or Instanced Static Mesh is added to the Sensor Environment, the triangle data of that Mesh is read and given to the underlying AGX Dynamics agxSensor::Environment during Play.
All active Lidar Sensor Component in the Level has to be registered with the Sensor Environment. This is done by adding a reference to it from the Sensor Environment Actor Details Panel.
20.1.1.1. Auto Add Objects
When the Auto Add Objects
property of a Sensor Environment is enabled, the Sensor Environment will automatically add objects in the Level to itself so that they can be detected by Lidars in the same Level.
To do this in a memory efficient manner, not all objects in the Level are added at once, but instead objects are dynamically added and removed depending if they are within the range of any Lidars in the Level.
More concretely, the Sensor Environment does this by automatically creating a “collision” sphere around each Lidar with radius equal to the maximum range of the Lidar.
These collision spheres follow their respective Lidar as they move about in the Level.
Each time an object overlap with a collision sphere is detected, the object is added to the Sensor Environment. Similarly, each time an objects stops being overlapped by a collision sphere, the objects is removed. To support multiple Lidars in the same Level, reference counting is used internally.
The detection of object overlaps with a collision sphere relies upon Unreal Engine’s UPrimitiveComponent overlap events. For this to trigger for a Static Mesh, two criteria must be met:
The Static Mesh Component has the
Generate Overlap Events
property enabled:
There is Simple collision data available for the Mesh asset:
The Component is Visible (relevant if using the
IgnoreInvisibleObjects
property of the Sensor Environment).
Note
It is NOT recommended to use the Use Complex Collision as Simple
option neither for Static Mesh Assets nor in the Unreal Project settings.
This is because of an Unreal issue where end/begin overlap events are triggered each tick during overlap when using this option which may cause performance issues.
Also note that Static Meshes imported from an AGX Dynamics archive will automatically get Simple Collision from the version that Lidar was released in AGX Dynamics for Unreal. Older imported assets may need to be synchronized, see Synchronize Model.
AGX primitive Shape Components will generate overlap events by default, but can be disabled by setting the Additional Unreal Collision
to No Collision
in the AGX Shape category in the Details Panel.
If a Mesh asset lacks Simple collision data, and is thus not added to the Sensor Environment when using the Auto Add feature, it can be added by selecting the Collision
menu in the Mesh asset editor:
20.1.1.2. Ambient Material
It is possible to assign an Ambient Material to the Sensor Environment. This material is used to simulate atmospheric effects on the Lidar laser rays such as rain, fog, snowfall etc. Assigning an Ambient Material to the Sensor Environment is done from it’s Details Panel.
A number of pre-defined Ambient Materials come with AGX Dynamics for Unreal, see Material Library for details.
20.1.2. Lidar Sensor Component
The Lidar Sensor Component is a central part of setting up a Lidar Simulation.
It can be created from a number of pre-defined models in its Details Panel, by selecting a Model
.
Using a custom ray pattern is also supported, see Custom Ray Pattern for details.
A Lidar Sensor must be added to a Sensor Environment in order to be active, see Sensor Environment.
For a Lidar to produce some output data, a Lidar Output must be added to it, see Lidar Output for details.
To attach a Lidar to any other Component moving in the Level, simply make it a child to that Component.
20.1.2.1. Custom Ray Pattern
It is possible to define and use a custom ray pattern with the Lidar Sensor Component. The responsibility of anyone defining a custom ray pattern is to set the rays as an array of transforms once. Then, each time the Simulation steps, an interval within the set rays array must be returned so that AGX knows which rays to use for that step.
To use a custom ray pattern, select the CustomRayPattern Model
in the Lidar’s Details Panel.
Assign a Model Parameters
Asset of the appropriate type.
Then, assign a function to the On Fetch Ray Transforms
delegate.
Do the same for the On Fetch Next Pattern Interval
delegate.
In the function assigned to the On Fetch Ray Transforms
delegate, the rays, represented as an array of transforms, must be set.
This function is called once at the start of the Simulation.
Each transform in the array represents a single ray, and will shoot out in the z-direction.
The transforms are expressed in local coordinates to the Lidar frame.
Each time the Simulation ticks, the function assigned to the On Fetch Next Pattern Interval
delegate will be called.
Here, an interval within the rays transforms array must be returned.
20.1.3. Lidar Surface Material
A Lidar Surface Material determines how Lidar laser rays interact with surfaces that they hit. This is important for example when the intensity of the returning Lidar laser beam is calculated.
The Lidar Surface Material can be assigned to any Component or Actor supported by the Sensor Environment. These include AGX primitive Shapes, Static Mesh Components (including Instanced and Hierarchical), Terrains etc.
To assign a Lidar Surface Material to a Component, the Lidar Surface Material Component can be used.
By setting the Selection
property, the Lidar Surface Material Component can assign a Lidar Surface material to either its parent, siblings or children in the Component attachment hierarchy.
For example, by making a Lidar Surface Material Component the parent of several Static Mesh Components, and then setting the Selection
property to Children
, all Static Mesh Components attached to the Lidar Surface Material Component will be assigned the selected Lidar Surface Material.
The assignment is done in Begin Play, and works by it adding a UAGX_SurfaceMaterialAssetUserData instance containing a reference to the selected Lidar Surface Material to the relevant Component(s) Asset User Data. Then, when a Component is added to the Sensor Environment, the Asset User Data is queried to find the Lidar Surface Material, and the information is handed to AGX Dynamics. As an advanced usage, it is therefore possible to manually add UAGX_SurfaceMaterialAssetUserData to any Component supported by the Sensor Environment instead of using the Lidar Surface Material Component.
If no Lidar Surface Material has been set (or nullptr has been assigned) on a Component that is added to the Sensor Environment, the default Lidar Surface Material, defined in AGX Dynamics, is assigned.
The Terrain is an Actor and therefore the Asset User Data approach cannot be used for it.
Instead, a Lidar Surface Material can be assigned directly to the Terrain Actor from the Details Panel under the AGX Lidar
category.
While live updating the Lidar Surface Material properties during Play is supported, it is currently not supported to assign new Lidar Surface Materials to objects in the Level during play. See Limitations.
20.1.4. Lidar Output
A Lidar Output determines what data is produced by the Lidar Sensor Component.
To add a Lidar Output to an instance of a Lidar, the AddTo
function is called on it, which can be done either from Blueprint or C++.
The built in Lidar Output types are:
AGX Lidar Output Position - produces position (x, y, z) data for each point.
AGX Lidar Output Position Intensity - produces position (x, y, z) and intensity data for each point.
If working with Blueprint, typically a Lidar Output is created as a variable (see figure below), and then the AddTo
function is called to add it to a Lidar.
A Lidar Output should only ever be added to a single instance of a Lidar.
20.1.4.1. Getting Output Data
To access the Lidar Output data, the GetData
function should be called.
This should be done during Simulation Post Step Forward (see Events and Delegates for details):
The Lidar Output data can be used directly or converted to a ROS2 message sensor_msgs::PointCloud2 which can then be sent via ROS2:
For more details on ROS2 publishers and subscribers in AGX Dynamics for Unreal see ROS2.
20.1.4.2. Rendering of Output Data
To render the Lidar Output data, the Render
function may be called.
This rendering works by the Lidar Output writing data to a Niagara System owned by the Lidar Sensor Component.
The Enable Rendering
property must also be enabled on the Lidar Sensor Component for the rendering to be available.
Data for each point is written to the Niagara particle system in buffers using the Niagara Data Interface from C++. The buffers written are:
Positions - array containing x, y and z position of each point in global coordinates.
Colors - array containing Linear Color for each point, can be used to visualize intensity for example.
User.NumPoints - integer value of how many points to spawn.
User.Lifetime - floating point value of the lifetime of the rendered points.
User.ZeroDistanceSize - floating point value of the smallest size of a rendered point.
Note that rendering of the Lidar Output data comes with a performance cost. Also note that rendering of multiple Lidar Outputs added to the same Lidar Sensor Component is not supported.
20.1.5. GPU Selection for Multi-GPU Systems
If the computer used has multiple RTX enabled GPU devices available, it is possible to select which of them the Lidar Raytrcing should be executed on.
This is done from Edit > Project Settings... > Plugins > AGX Dynamics
.
20.1.6. Troubleshooting
20.2. No output data is created
Make sure the Lidar Output has been properly set up, see Lidar Output.
Check the Output Log for any Warnings or Errors.
Make sure your license has the
AGX-Sensor
module. Modules are listed in theAGX > License > Activate service license...
window.Check if the same issue also occurs in one of the Demo Projects that uses Lidar. For example the
Lidar
Level of theTutorialHallways
demo.
20.3. Framerate Drops Significantly
From within the Unreal Editor, run the command stat unit
to get information about GPU and CPU usage.
If the GPU usage too high when using Lidar in the Level, you may try the following:
Improve the overall performance of running the Level.
Use a more powerful Graphical Processing Unit (GPU).
Run the Lidar Raytracing on a different GPU, see GPU Selection for Multi-GPU Systems.
20.4. Meshes not added to Sensor Environment using Auto Add
Go through the requirements listed in Auto Add Objects. If Simple collision data is lacking from the Mesh asset, it can be created manually in the Mesh asset editor, see Auto Add Objects for details.
20.4.1. Limitations
Lidar Raytracing is only supported on computers with at least one RTX enabled Graphical Processing Unit (GPU). Assigning Lidar Surface Materials to objects during runtime is currently not supported. Import of Lidar or Sensor Environment from an AGX Dynamics archive is currently not supported. Rendering of multiple Lidar Outputs added to the same Lidar Sensor Component is not supported.
20.5. Camera Sensor
AGX Dynamics for Unreal provides two Camera Sensor Components, an 8-bit and a 16-bit variant, that can be used to capture the scene from that camera’s point of view. It also provides convenience functions for getting the image data either as an array of pixels (RGB) or as a ROS2 sensor_msgs::Image message. See Getting Image Data for more details. The image data is encoded in linear color space.
A field of view (FOV) and Resolution can be specified from the Details Panel:
The Camera Sensor Component renders the image information to a Render Target which must be assigned to the Component.
To make this simple, a button labeled Generate Runtime Assets
is available in the Details Panel as can be seen in the image above.
It will generate a Render Target with the appropriate settings according to the specified Resolution.
This also means that if this parameter is changed, the Generate Runtime Assets
procedure must be done again.
If an old Render Target is currently assigned when the Generate Runtime Assets
procedure is performed, it will be updated and no new Render Target will be created.
Changing the FOV can be done during Play either from the Details Panel or by calling the function SetFOV.
To attach the Camera Sensor to a simulated machine, make it a child to the Scene Component it should follow.
20.5.1. Getting Image Data
Note
Getting the image as an array of pixels is only available in C++ and not available in Blueprint when using the CameraSensor16BitComponent due to the underlying datatype not beeing supported in Blueprint. Only getting the image as a ROS2 message is supported in Blueprint when using the CameraSensor16BitComponent.
Getting image data from the Camera Sensor Component can be done in two main ways; synchronously or asynchronously. The synchronous functions are GetImagePixels and GetImageROS2 which will return the captured image either as an array of pixels (RGB) or as a ROS2 sensor_msgs::Image message. These functions does some synchronization between the game thread and the render thread in order to return the image to the caller and can therefore be very slow.
The recommended approach is using the asynchronous versions of these, namely GetImagePixelsAsync and GetImageROS2Async. These will return immediately and whenever the image is ready an Event, or delegate, will be triggered and the image is passed to anyone listening to the Event. The workflow then is to bind to the Event, for example in BeginPlay, and then during runtime call the async function whenever a new image should be captured.
In the examples below, both getting the image as pixels as well as a sensor_msgs::Image ROS2 message are shown. In the latter example, the image is also sent via ROS2 using a ROS2 Publisher. See ROS2 for more details.
When using the 16-bit variant of the Camera Sensor, i.e. the CameraSensor16BitComponent, each pixel channel (RGB) is encoded as two consecutive 8-bit values, i.e. 48 bits per pixel in little endian format.
To do the same thing in C++, simply bind do the delegate:
CameraSensor->NewImagePixels.AddDynamic(this, &UMyClass::MyFunction);
Here, MyFunction must be a UFUNCTION taking a single const TArray<FColor>&
parameter.
20.5.2. Limitations
Getting the image as an array of pixels is only available in C++ and not available in Blueprint when using the CameraSensor16BitComponent due to the underlying datatype not beeing supported in Blueprint. Only getting the image as a ROS2 message is supported in Blueprint when using the CameraSensor16BitComponent.
20.6. Lidar Sensor Line Trace
Note
This is an experimental feature meaning it may be changed in the future with no guaranteed backwards compatibility. It is recommended to use the Lidar Sensor whenever possible since that is the prioritized and main way of simulating Lidar sensors in AGX Dynamics for Unreal.
The Lidar Sensor Line Trace Component makes it possible to generate point cloud data in real-time.
The Lidar Sensor can be run in one of two Execution Modes: Auto
or Manual
.
In mode Auto
, the Lidar Sensor will perform partial scans in each Simulation tick in a pace determined by the selected Scan Frequency
.
The Scan Frequency
determines how often a complete scan cycle is scanned, i.e. how often the entire Scan Pattern is covered.
In mode Manual
, the Lidar Sensor does nothing unless the RequestManualScan
function is called, see image below.
Here, Fraction Start
and Fraction End
determine at what fraction of the total scan pattern the scan should start and end respectively.
FOVWindow Horizontal
and FOVWindow Vertical
can be used to mask a part of the scan pattern so that any point outside this Field of View (FOV) Window is disregarded.
The Output Frequency
determines how often the Lidar Sensor outputs point cloud data, see Getting Point Cloud Data for more details.
This is only used with Execution Mode Auto
.
The Scan Pattern determines what points are scanned within the selected Field of View (FOV) and in what order.
For example, with the Horizontal Sweep
Scan Pattern, all points along the first vertical line are scanned in succession before moving to the next vertical line.
FOV determines the Field of View of the Lidar Sensor, in degrees, along the vertical and horizontal directions respectively. The Resolution determines the minimum angular distance between two neighbouring laser rays in degrees.
The Range determines the maximum detection distance for the Lidar Sensor in centimeters. Any object further away than this distance will not be detected by the Lidar Sensor.
The Calculate Intensity option determines whether an intensity value for each Laser ray should be approximated or not. If not, the corresponding intensity value for all rays will be set to zero. To calculate the intensity, the angle of incident, material roughness parameter of the object being hit, as well as the distance is taken into account. The intensity dropoff over distance is a function of the Beam Exit Diameter and Beam Divergence that can be set for each Lidar Sensor instance.
To make a primitive AGX Shape not detectable by the Lidar Sensor, select an Additional Unreal Collision
setting for that Shape that does not include query.
See Additional Unreal Collisions for more details.
To attach the Lidar Sensor to a simulated machine, make it a child to the Scene Component it should follow.
20.6.1. Getting Point Cloud Data
The way the Lidar Sensor outputs its point cloud data is via the delegate Point Cloud Data Output
.
This is used in both Execution Mode Auto
and Manual
.
To access the point cloud data output, the user must bind to this delegate.
An example of this in Blueprint is shown below:
To do the same thing in C++, simply bind do the delegate:
LidarSensor->PointCloudDataOutput.AddDynamic(this, &UMyClass::MyFunction);
Here, MyFunction must be a UFUNCTION taking a single const TArray<FAGX_LidarScanPoint>&
parameter.
Note that in Execution Mode Auto
, it is the Output Frequency property of the Lidar Sensor that determines how often this delegate is executed.
In Execution Mode Manual
, this delegate is always called once upon calling the RequestManualScan
function, as long as valid input parameters are given.
The point cloud data returned by the Point Cloud Data Output
delegate is such that any laser ray miss is represented by a FAGX_LidarScanPoint with the IsValid
property set to false.
20.6.1.1. ROS2 sensor_msgs::PointCloud2
A convenience function Convert XYZ is available that takes as input an array of FAGX_LidarScanPoint and converts it to a ROS2 message sensor_msgs::PointCloud2. This can then be send via ROS2 using a ROS2 publisher, see ROS2 for more details about ROS2 in AGX Dynamics for Unreal.
An alternative version to the Convert XYZ is the convenience function Convert Angles TOF, available from Blueprint and C++, which represents each point as two angles and time-of-flight.
20.6.2. Limitations
With the current implementation, Terrain height changes as well as Terrain particles are not detected by the Lidar Sensor Component. Providing this functionality is currently in the process of being developed.