sick_safetyscanners_base
- Provides an Interface to read the sensor output of a SICK
- Safety Scanner 
README
Sick_Safetyscanners_Base CPP Driver
Table of contents
A CPP standalone Driver which reads the raw data from the SICK Safety Scanners and takes custom functions to publish the data.
Supported Hardware
Supported are all microScan3, nanoScan3 and outdoorScan3 variants with Ethernet connection.

Getting started
The driver will be released on this github repository, and can then be installed from source.
Prerequisites
- Linux 
- Correctly setup SICK Safety Scanner 
- Connected SICK Safety Scanner and a correctly setup ethernet network. Both the host and the sensor have to be in the same network. 
- Installed libboost 
Installation
For installation this github repository has to be cloned and afterwards installed. If a custom install directory is wanted use the -DCMAKE_INSTALL_PREFIX option to specify a path.
git clone https://github.com/SICKAG/sick_safetyscanners_base.git
cd sick_safetyscanners_base
mkdir build
cd build
cmake -DCMAKE_INSTALL_PREFIX=<path to install folder> ..
make -j8
make install
Usage
To use the library in a driver the path of the installation has to be added to the cmake prefix path of your application. You can achieve this by using, bevor invoking cmake on your application.
export CMAKE_PREFIX_PATH=<path to install folder>
Afterwards the driver and the settings for the driver can be included with:
#include <sick_safetyscanners_base/SickSafetyscanners.h>
#include <sick_safetyscanners_base/Exceptions.h>
#include <sick_safetyscanners_base/Types.h>
#include <sick_safetyscanners_base/datastructure/CommSettings.h>
To get the driver up and running you need first to choose between the synchronous and asynchronous APIs based on your needs.
In the latter case you can also pass an instance of boost::asio::io_service to the constructor of the AsyncSickSafetyScanner.
API
Synchronous Client
In cases where you do not want the driver to spawn internal child threads to asynchronously process incomming sensor data you can use the SyncSickSafetyScanner class.
| Function | Information | 
|---|---|
| SyncSickSafetyScanner | Constructor call. | 
| bool isDataAvailable(); | Non-blocking call that indicates wheether sensor data is available to fetch via the receive-function from the internal sensor data receiving buffer. | 
| const Data receive | Blocking call to receive one sensor data packet at a time. | 
Example
// Sensor IP and Port
std::string sensor_ip_str = "192.168.1.11";
sick::types::ip_address_t sensor_ip = boost::asio::ip::address_v4::from_string(sensor_ip_str);
sick::types::port_t tcp_port {2122};
// Prepare the CommSettings for Sensor streaming data
sick::datastructure::CommSettings comm_settings;
std::string host_ip_str = "192.168.1.9";
comm_settings.host_ip = boost::asio::ip::address_v4::from_string(host_ip_str);
comm_settings.host_udp_port = 0;
// Create a sensor instance
auto safety_scanner = std::make_unique<sick::SyncSickSafetyScanner>(sensor_ip, tcp_port, comm_settings);
// Receive one sensor data packet
auto timeout = boost::posix_time::seconds(5);
sick::datastructure::Data data = safety_scanner->receive(timeout);
// ...
Asynchronous Client
If you dont need restrict your client to work with single-threaded blocking receive calls in a strict sequential manner you can use the AsyncSickSafetyScanner class. The sensor data callback required by the constructor has the following signature
void callback(const sick::datastructure::Data& data);
| Function | Information | 
|---|---|
| AsyncSickSafetyScanner | Constructor call with custom data sensor callback. | 
| AsyncSickSafetyScanner | Constructor call with custom data sensor callback for multicast ip addresses. | 
| AsyncSickSafetyScanner | Constructor call. This variant allows the user to pass through an instance of boost::asio::io_service to get full control over the thread execution. In this case the driver is not spawning an internal child thread but relies on the user to perform regular io_service.run() operations and keep the service busy. | 
| void run(); | Starts to receive sensor data via UDP and passes the data to the callback as specified in the constructor. | 
| void stop(); | Stops all asynchronous receiving and processing operations. | 
Example
// Sensor IP and Port
std::string sensor_ip_str = 192.168.1.11
sick::types::ip_address_t sensor_ip = boost::asio::ip::address_v4::from_string(sensor_ip_str);
sick::types::port_t tcp_port {2122};
// Prepare the CommSettings for Sensor streaming data
sick::datastructure::CommSettings comm_settings;
std::string host_ip_str = "192.168.1.9"
comm_settings.host_ip = boost::asio::ip::address_v4::from_string(host_ip_str);
comm_settings.host_udp_port = 0;
// Define a sensor data callback
sick::types::ScanDataCb cb = [](const sick::datastructure::Data &data) {
    std::cout << "Number of beams: " << data.getMeasurementDataPtr()->getNumberOfBeams() << std::endl;
};
// Create a sensor instance
auto safety_scanner = std::make_unique<sick::AsyncSickSafetyScanner>(sensor_ip, tcp_port, comm_settings, cb);
// Special case if a multicast IP is seltected as host_ip
// std::string host_ip_str = "235.235.235.2"
// comm_settings.host_ip = boost::asio::ip::address_v4::from_string(host_ip_str);
// std::string interface_ip_str = "192.168.1.9"
// auto interface_ip = boost::asio::ip::address_v4::from_string(interface_ip_str);
// auto safety_scanner = std::make_unique<sick::AsyncSickSafetyScanner>(sensor_ip, tcp_port, comm_settings, interface_ip, cb);
// Start async receiving and processing of sensor data
safety_scanner->run();
// ... Do other stuff
// Stop async processing
safety_scanner->stop();
Parameters of Communication Settings
The parameters can be set using the setters of the CommSettings class. To set for example to host_ip the following function can be called.
sick::datastructure::CommSettings m_communication_settings;
m_communication_settings.setHostIp("192.168.1.100");
| Parameter Name | API | Default | Information | 
|---|---|---|---|
| sensor_ip | void setSensorIp(const std::string& sensor_ip); | 192.168.1.11 | Sensor IP address. Can be passed as an argument to the launch file. | 
| host_ip | void setHostIp(const std::string& host_ip); | 192.168.1.9 | Host IP address. Can be passed as an argument to the launch file. | 
| host_udp_port | void setHostUdpPort(const uint16_t& host_udp_port); | 0 | Host UDP Port. Can be passed as an argument to the launch file. Zero allows system chosen port. | 
| publishing_frequency | void setPublishingFrequency(const uint16_t& publishing_frequency) | 1 | Publish every n_th scan, where n is the publishing frequency. For a 25Hz laser, setting to 1 makes it publish at 25Hz, to 2 makes it publish at 12.5Hz. | 
| start_angle | void setStartAngle(const uint32_t& start_angle) | 0.0 | Start angle of scan in radians, if both start and end angle are equal, all angels are regarded. 0° is at the front of the scanner. | 
| end_angle | void setEndAngle(const uint32_t& start_angle) | 0.0 | End angle of scan in radians, if both start and end angle are equal, all angels are regarded. 0° is at the front of the scanner. | 
| channel_enabled | void setEnabled(bool enabled); | true | If the channel should be enabled | 
| e_interface_type | void setEInterfaceType(const uint8_t& e_interface_type) | 0 | Sets the interface type of the sensor  | 
| features | void CommSettings::setFeatures | all true | Enables the individual data outputs. | 
COLA2 Functions (available on both driver API variants)
The Library allows to access variables of the sensor and invoke methods to change settings using the COLA2 protocol. The following methods can be called:
| Function | Information | 
|---|---|
| void changeSensorSettings | Updates the sensor settings to the passed configuration. | 
| void findSensor | Lets the sensor flash the display for the specified time. | 
| void requestApplicationName | Returns the name of the current application. | 
| void requestConfigMetadata | Returns the metadata of the current configuration of the sensor. | 
| void requestDeviceName | Returns the device name. | 
| void requestDeviceStatus | Returns the device status. | 
| void requestFieldData | Returns the field data of the warning and safety fields. | 
| void requestFirmwareVersion | Returns the firmware version. | 
| void requestLatestTelegram | Returns the latest telegram for the channel index. Up to 4 Channels can be supported by the cola protocol, the actual number depends on the used scanner. Channel 0 is set as default value. | 
| void requestMonitoringCases | Returns the data of the Monitoring Cases. | 
| void requestOrderNumber | Returns the order Number. | 
| void requestPersistentConfig | Returns the Persistent configuration of the sensor, which was set in the Safety Designer. | 
| void requestProjectName | Returns the project name. | 
| void requestRequiredUserAction | Returns the required user actions as specified in the cola 2 manual. | 
| void requestSerialNumber | Returns the serial number of the sensor. | 
| void requestStatusOverview | Returns the status overview. | 
| void requestTypeCode | Returns the type code of the sensor. | 
| void requestUserName | Returns the user name. | 
Troubleshooting
- Check if the sensor has power and is connected to the host. 
- Check if both sensor and host are in the same subnet e.g. 192.168.1 
- Are the correct IPs configured for the application? 
- Is the correct Interface Type configured? 
Creators
Lennart Puck 
Martin Schulze
FZI Forschungszentrum Informatik
on behalf of SICK AG