/radiation_layer

Plugin layer for ROS costmap_2d to include ionising radiation as a hazard

Primary LanguageC++

Radiation Costmap Plugin

Overview

Ionising radiation can damage electronics and materials of robot platforms. To increase the lifetime of robotic systems deployed in environments with high levels of ionising radiation, minimisation of exposure is critical. Avoidance of radiation is also important to limit the amount of cross contamination which may occur from wheels, tracks or propeller wash picking up and transporting loose powdered radioactive material.

The existing ROS navigation stack allows for occupancy grid representations of hazards (such as obstacles) or objects of interest (such as people). Path planners can use this occupancy grid for optimal planning whilst avoiding obstacles, based on total cost.

This plugin allows for additional layers of the stock ROS costmap_2d implementation to be added based on data from radiation instrumentation. The cost associated with radiation levels can be scaled on an ad hoc basis.

Referencing

This work is described in the paper:

Andrew West, Thomas Wright, Ioannis Tsitsimpelis, Keir Groves, Malcolm J. Joyce, and Barry Lennox. 2022. “Real-Time Avoidance of Ionising Radiation Using Layered Costmaps for Mobile Robots.” Frontiers in Robotics and AI 9 (March). https://doi.org/10.3389/frobt.2022.862067.

Installation

This package has dependancy on the radiation_msgs package, and this must be installed first:

cd catkin_ws/src
git clone https://github.com/EEEManchester/radiation_msgs.git
cd ..
catkin_make

Once the radiation_msgs package is installed, to build from this repo, clone the latest version from this repository into your catkin workspace and compile using:

cd catkin_ws/src
git clone -b melodic https://github.com/EEEManchester/radiation_layer.git
cd ..
catkin_make

This package is intended to be used with the ROS costmap_2D package from the navigation stack, therefore, this package will also need to be installed. However, this is likely already installed on your system.

How it works

The plugin layer subscribes to a stamped topic representing ionising radiation, and parses the location (using TF) and intensity of radiation to a cost on the costmap. The radiation layer operates at the same resolution and size as the main costmap, or defined in a yaml file.

The average radiation value at each costmap cell is maintained, even when the costmap is resized. Radiation value can be any units or scale (e.g. counts per second, Sv/hr), as the actual cost is calculated based on upper and lower thresholds. Using thresholds allows for the robot to ignore areas of low radiation (e.g. background levels), and completely avoid areas with elevated radiation (e.g. high enough to immediately cause an electronic fault).

Cost in each cell ranges from 0 (free space - robot can travel with no penalty) to 252 (highest cost for a non-lethal obstacle), and therefore the average radiation value is scaled linearly between 0-252 based on:

cost = 252 * (radiation_value - lower_threshold) / (upper_threshold - lower_threshold)

The upper and lower thresholds are managed by the user, and can be changed at any time through use of dynamically reconfigurable parameters.

When the costmap is resized, all the averaged radiation values on a per grid cell basis are temporarily stored, then the costmap is repopulated once the costmap has been expanded. In the event that the threshold values are altered (and therefore the cost 0-252 has changed), the entire plugin layer cost is recalculated and sent out to the rest of the costmap.

Inflating Cost

Updating individual cells is useful, however, it is better for path planners to inflate the cost to a reasonable scale length of the robot. Users can define this scale length using radiation_radius or radiation_footprint options in the yaml file (which is covered later). These act identically to the usual robot footprint parameters, but can not be reconfigured once the costmap layer is running. The radiation_footprint must have three or more vertices. If no valid radius or footprint is provided, the layer will default to averaging individual based on observation location. The method of averaging is explained in the next section.

Inverse Distance Weighting

To gain the estimated average of a costmap cell based on surrounding observations, the principle of distance weighting is used, i.e. observations which are closer to the cell are more likely to be closer to the real value in the cell compared to those further away. Things that are close are more likely to be similar.

This is achieved using a Gaussian kernel, values close to the observation (distance = 0.0) have weightings close to 1.0, which tails off as a function of distance. The scale length of this Gaussian kernel is given by averaging_scale_length in the appropriate yaml file. If 0.0 or undefined, the averaging will happen with equal weighting (1.0) across the footprint regardless of distance from the observation. Equal weighting is generally not recommended.

The weight, , is given by: where is the euclidean distance between the observation and the cell centre, and is the averaging scale length. The larger the scale length factor, the greater the cells weighting as a function of distance. This means the weighting can be tuned to an appropriate length for the system.

The averaged value of a cell in the costmap, , given a new observation in time, is geometrically averaged with the existing total and weight and averaged value previous to the new observation:

This is strictly the average for raw observation values, i.e. they are stored as floating point values, not integer 0-255 values. The scaling factor mentioned previously converts the floating averages into integer cost values, allowing for cost to be scaled dynamically.

Running an example

Once the package is installed, the example launch file can be run using:

roslaunch radiation_layer costmap_demonstrator.launch

This will start RVIZ for visualisation, a node to publish radiation data (mimicking a robot driving around a radioactive environment) and a costmap_2d node with the radiation_layer plugin. The costmap_2d node also listens to a map produced by SLAM (provided by the map_server).

The launch file will also start the rqt_reconfigure GUI (accessible via):

rosrun rqt_reconfigure rqt_reconfigure

The dynamically reconfigurable options can be changed by the user in real-time. These parameters can be modified by other ROS nodes (allowing for autonomous re-weighting of cost). Check out how to setup a dynamic reconfigure client here.

In this example, the range of published radiation values are floating point values between 0-1000, with default lower threshold = 1, and upper_threshold = 200, i.e. values above 200 will be set at lethal obstacles. For the local and global costmaps, these have been set in the launch file or yaml file as examples of how to set this manually. Radiation information is conveyed using the radiation_msgs/DoseRate type.

Dynamic Reconfigure Parameters

The plugin can be enabled and disabled using the enable functionality, the plugin layer will continue to process radiation data and perform averaging but no additional cost will be added to the costmap. Behaviour of a robot with respect to radiation can be easily switched on and off using this enable functionality.

To accommodate the wide range of values and units of measuring radiation (for example 300 counts per second may be equivalent to 1 nSv/hr), the lower and upper thresholds consist of a double value (between -999 and 999) and a power multiplier. For example to set an upper threshold of 1000 units, the value would be set to 1 and the mulitplier to kilo (3), i.e. one with three zeros.

The power modifier for thresholds ranges from:

  • nano (E-9)
  • micro (E-6)
  • milli (E-3)
  • unit (E0)
  • kilo (E3)
  • mega (E6)
  • giga (E9)

Loading the plugin for costmap_2d or move_base

Using the plugin for both costmap_2d or directly for path planning into move_base is achieved through the use of .yaml files. An example yaml file can be found in the package launch/yaml folder.

Costmap layers are accessed as per the order specified in the yaml file, for example,

plugins: 
    - {name: static_map,       type: "costmap_2d::StaticLayer"}
    - {name: obstacles,        type: "costmap_2d::VoxelLayer"}
    - {name: inflation,        type: "costmap_2d::InflationLayer"}
    - {name: radiation,        type: "radiation_layer_namespace::RadLayer"}

results in the inflation plugin layer only operating on the levels above itself (i.e. static_map and obstacles), not the radiation layer. Order of plugins may affect the desired output and performance of the costmap. To specify the topic name of the radiation sensor, use the snippet

radiation:
    radiation_topic: radiationTopic

with radiation: being the same name as in the previous plugins lists. The default topic name is radiation_topic.

There are more options which can passed to the costmap layer including:

    radiation_footprint: # List of vertices in format [[x0, y0], [x1, y1], [x2, y2], ..., [xN, yN]]
    radiation_radius: # radius in metres e.g. 0.5
    averaging_scale_length:  # distance in metres e.g. 0.2
    combination_method: # How are costmap costs integrated with other layers above, using integer values e.g. 1
    minimum_weight: # Total weight of cell before added to costmap (used with IDW)

Footprint and radius perform identically to those used of the robot footprint, however, this is a footprint around the frame of the radiation sensor, not the robot. The footprint argument will supersede any radius arguments, ensure the footprint argument is missing or commented out if wishing to use radius value. The average scale length has been covered earlier, and combination_method allows the user to define how cost is incorporated in the layered costmap. This includes taking the maximum value of all layers (0), overwriting all other layers (1), addition with cost of other layers (2), and maximum value whilst preserving unknown space (3). Default is 0 (keep maximum value of all layers).

The option minimum_weight can be useful when the edges of an inflated space are greatly different from the local average, and is causing issues with path planning. By requiring a minimum total cell weight, only cells which have seen multiple observations, or very nearby observations are updated in the costmap. It is recommended this is kept to the default value of 0.0.

Settings for Different Radiation Types

Alpha and Beta

Alpha and Beta radiation have short path lengths, therefore are typically only observed in close proximity to a source. It can be assumed, most of the time, that sources of Alpha and Beta can be treated as highly localised.

Therefore, it is better to only use the footprint of the detector, e.g. a paddle sensor, for inflation/adding cost to the costmap. Furthermore, the user can set the averaging_scale_length = 0.0, and no distance weighting is applied. This may be a more reliable approach when twinned with avoidance strategies.

Gamma

As gamma radiation changes on larger length scales, interpolation can happen at sizes greater than the detector volume. To have a robot sucessfullly avoid radiation, this enlarged interpolation area should be larger than the robot. Tuning of this value is discussed in the journal article. Furthermore, distance weighted averaging is then also encouraged to help build more realistic maps.

Neutrons

Neutrons are likely to have scale lengths more akin to Gamma radiation transport, certainly further than Alpha or Beta radiation, therefore, it is best to treat them similarly to gamma radiation using distance weighted interpolation.

Bugs & Feature Requests

Please report bugs and request features using the Issue Tracker.