Skip to content

Commit 3ee87d1

Browse files
authored
Merge pull request #133 from open-dynamic-robot-initiative/fkloss/sensor_info
Add optional sensor_info to sensor interface
2 parents ae3b768 + 631530a commit 3ee87d1

11 files changed

+166
-46
lines changed

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,10 @@
99
classes).
1010
- Method `RobotDriver::get_idle_action()` that is expected to return an action
1111
that is safe to apply while the robot is idle.
12+
- Option to provide static information about a sensor (e.g. camera calibration
13+
coefficients, frame rate, etc.). For this, a method `get_sensor_info()` is added to
14+
`SensorFrontend`. It defaults to an empty struct for backward compatibility. To use
15+
it, implement the method with the same name in `SensorDriver`.
1216

1317
### Changed
1418
- The return type of `RobotDriver::get_error()` is changed to

doc/custom_sensor.rst

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
****************
2+
Sensor Interface
3+
****************
4+
5+
``robot_interfaces`` also provides base classes for a pure sensor interface. In
6+
contrast to a robot, a sensor only provides observations but does not take actions.
7+
Apart from that, the overall structure is mostly the same as for a robot. That is,
8+
there are :cpp:class:`~robot_interfaces::SensorBackend` and
9+
:cpp:class:`~robot_interfaces::SensorFrontend` which communicate via
10+
:cpp:class:`~robot_interfaces::SensorData` (either single- or multi-process).
11+
12+
For implementing an interface for your sensor, you only need to implement an observation
13+
type and a driver class based on :cpp:class:`~robot_interfaces::SensorDriver`. Then
14+
simply create data/backend/frontend using those custom types as template arguments.
15+
16+
Optionally, your driver class can also provide a "sensor info" object by implementing
17+
:cpp:func:`~robot_interfaces::SensorDriver::get_sensor_info`. This object is then
18+
accessible by the user via
19+
:cpp:func:`~robot_interfaces::SensorFrontend::get_sensor_info`. You may use this, to
20+
provide static information about the sensor, that does not change over time (e.g. frame
21+
rate of a camera).
22+
If you don't implement the corresponding method in the driver, the front end will return
23+
an empty struct as placeholder.

doc_mainpage.rst

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ paper_ on the open-source version of the TriFinger robot.
1515
doc/real_time.rst
1616
doc/quick_start_example.rst
1717
doc/custom_driver.rst
18+
doc/custom_sensor.rst
1819

1920
.. toctree::
2021
:caption: Topic Guides

include/robot_interfaces/sensors/pybind_sensors.hpp

+23-16
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
#include <robot_interfaces/sensors/sensor_frontend.hpp>
1717
#include <robot_interfaces/sensors/sensor_log_reader.hpp>
1818
#include <robot_interfaces/sensors/sensor_logger.hpp>
19+
#include <robot_interfaces/utils.hpp>
1920

2021
namespace robot_interfaces
2122
{
@@ -24,7 +25,7 @@ namespace robot_interfaces
2425
*
2526
* @tparam The ObservationType
2627
*/
27-
template <typename ObservationType>
28+
template <typename ObservationType, typename InfoType = None>
2829
void create_sensor_bindings(pybind11::module& m)
2930
{
3031
pybind11::options options;
@@ -33,12 +34,15 @@ void create_sensor_bindings(pybind11::module& m)
3334
options.disable_function_signatures();
3435

3536
// some typedefs to keep code below shorter
36-
typedef SensorData<ObservationType> BaseData;
37-
typedef SingleProcessSensorData<ObservationType> SingleProcData;
38-
typedef MultiProcessSensorData<ObservationType> MultiProcData;
39-
typedef SensorLogger<ObservationType> Logger;
37+
typedef SensorData<ObservationType, InfoType> BaseData;
38+
typedef SingleProcessSensorData<ObservationType, InfoType> SingleProcData;
39+
typedef MultiProcessSensorData<ObservationType, InfoType> MultiProcData;
40+
typedef SensorLogger<ObservationType, InfoType> Logger;
4041
typedef SensorLogReader<ObservationType> LogReader;
4142

43+
pybind11::class_<None, std::shared_ptr<None>>(
44+
m, "None", pybind11::module_local());
45+
4246
pybind11::class_<BaseData, std::shared_ptr<BaseData>>(m, "BaseData");
4347

4448
pybind11::class_<SingleProcData, std::shared_ptr<SingleProcData>, BaseData>(
@@ -52,31 +56,34 @@ void create_sensor_bindings(pybind11::module& m)
5256
pybind11::arg("is_master"),
5357
pybind11::arg("history_size") = 1000);
5458

55-
pybind11::class_<SensorDriver<ObservationType>,
56-
std::shared_ptr<SensorDriver<ObservationType>>>(m,
57-
"Driver");
59+
pybind11::class_<SensorDriver<ObservationType, InfoType>,
60+
std::shared_ptr<SensorDriver<ObservationType, InfoType>>>(
61+
m, "Driver");
5862

59-
pybind11::class_<SensorBackend<ObservationType>>(m, "Backend")
63+
pybind11::class_<SensorBackend<ObservationType, InfoType>>(m, "Backend")
6064
.def(pybind11::init<
61-
typename std::shared_ptr<SensorDriver<ObservationType>>,
65+
typename std::shared_ptr<SensorDriver<ObservationType, InfoType>>,
6266
typename std::shared_ptr<BaseData>>())
6367
.def("shutdown",
64-
&SensorBackend<ObservationType>::shutdown,
68+
&SensorBackend<ObservationType, InfoType>::shutdown,
6569
pybind11::call_guard<pybind11::gil_scoped_release>());
6670

67-
pybind11::class_<SensorFrontend<ObservationType>>(m, "Frontend")
71+
pybind11::class_<SensorFrontend<ObservationType, InfoType>>(m, "Frontend")
6872
.def(pybind11::init<typename std::shared_ptr<BaseData>>())
73+
.def("get_sensor_info",
74+
&SensorFrontend<ObservationType, InfoType>::get_sensor_info,
75+
pybind11::call_guard<pybind11::gil_scoped_release>())
6976
.def("get_latest_observation",
70-
&SensorFrontend<ObservationType>::get_latest_observation,
77+
&SensorFrontend<ObservationType, InfoType>::get_latest_observation,
7178
pybind11::call_guard<pybind11::gil_scoped_release>())
7279
.def("get_observation",
73-
&SensorFrontend<ObservationType>::get_observation,
80+
&SensorFrontend<ObservationType, InfoType>::get_observation,
7481
pybind11::call_guard<pybind11::gil_scoped_release>())
7582
.def("get_timestamp_ms",
76-
&SensorFrontend<ObservationType>::get_timestamp_ms,
83+
&SensorFrontend<ObservationType, InfoType>::get_timestamp_ms,
7784
pybind11::call_guard<pybind11::gil_scoped_release>())
7885
.def("get_current_timeindex",
79-
&SensorFrontend<ObservationType>::get_current_timeindex,
86+
&SensorFrontend<ObservationType, InfoType>::get_current_timeindex,
8087
pybind11::call_guard<pybind11::gil_scoped_release>());
8188

8289
pybind11::class_<Logger, std::shared_ptr<Logger>>(m, "Logger")

include/robot_interfaces/sensors/sensor_backend.hpp

+16-8
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515

1616
#include <robot_interfaces/sensors/sensor_data.hpp>
1717
#include <robot_interfaces/sensors/sensor_driver.hpp>
18+
#include <robot_interfaces/utils.hpp>
1819

1920
namespace robot_interfaces
2021
{
@@ -27,24 +28,31 @@ namespace robot_interfaces
2728
*
2829
* @tparam ObservationType
2930
*/
30-
template <typename ObservationType>
31+
template <typename ObservationType, typename InfoType = None>
3132
class SensorBackend
3233
{
3334
public:
34-
typedef std::shared_ptr<SensorBackend<ObservationType>> Ptr;
35-
typedef std::shared_ptr<const SensorBackend<ObservationType>> ConstPtr;
35+
typedef std::shared_ptr<SensorBackend<ObservationType, InfoType>> Ptr;
36+
typedef std::shared_ptr<const SensorBackend<ObservationType, InfoType>>
37+
ConstPtr;
3638

3739
/**
3840
* @param sensor_driver Driver instance for the sensor.
3941
* @param sensor_data Data is sent to/retrieved from here.
4042
*/
41-
SensorBackend(std::shared_ptr<SensorDriver<ObservationType>> sensor_driver,
42-
std::shared_ptr<SensorData<ObservationType>> sensor_data)
43+
SensorBackend(
44+
std::shared_ptr<SensorDriver<ObservationType, InfoType>> sensor_driver,
45+
std::shared_ptr<SensorData<ObservationType, InfoType>> sensor_data)
4346
: sensor_driver_(sensor_driver),
4447
sensor_data_(sensor_data),
4548
shutdown_requested_(false)
4649
{
47-
thread_ = std::thread(&SensorBackend<ObservationType>::loop, this);
50+
// populate the sensor information field
51+
InfoType info = sensor_driver_->get_sensor_info();
52+
sensor_data_->sensor_info->append(info);
53+
54+
thread_ =
55+
std::thread(&SensorBackend<ObservationType, InfoType>::loop, this);
4856
}
4957

5058
// reinstate the implicit move constructor
@@ -67,8 +75,8 @@ class SensorBackend
6775
}
6876

6977
private:
70-
std::shared_ptr<SensorDriver<ObservationType>> sensor_driver_;
71-
std::shared_ptr<SensorData<ObservationType>> sensor_data_;
78+
std::shared_ptr<SensorDriver<ObservationType, InfoType>> sensor_driver_;
79+
std::shared_ptr<SensorData<ObservationType, InfoType>> sensor_data_;
7280

7381
bool shutdown_requested_;
7482

include/robot_interfaces/sensors/sensor_data.hpp

+41-11
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
21
/**
32
* @file
43
* @brief To store all the data from all the sensors in use
@@ -16,19 +15,30 @@
1615
#include <time_series/multiprocess_time_series.hpp>
1716
#include <time_series/time_series.hpp>
1817

18+
#include <robot_interfaces/utils.hpp>
19+
1920
namespace robot_interfaces
2021
{
2122
/**
2223
* @brief Contains the data coming from the sensors.
2324
*
2425
* @tparam Observation Type of the sensor observation.
2526
*/
26-
template <typename Observation>
27+
template <typename Observation, typename Info = None>
2728
class SensorData
2829
{
2930
public:
30-
typedef std::shared_ptr<SensorData<Observation>> Ptr;
31-
typedef std::shared_ptr<const SensorData<Observation>> ConstPtr;
31+
typedef std::shared_ptr<SensorData<Observation, Info>> Ptr;
32+
typedef std::shared_ptr<const SensorData<Observation, Info>> ConstPtr;
33+
34+
/**
35+
* @brief Static information about the sensor
36+
*
37+
* Note: A time series is used here for convenience to handle the shared
38+
* memory aspect. However, this is intended to only hold one element that
39+
* doesn't change over time.
40+
*/
41+
std::shared_ptr<time_series::TimeSeriesInterface<Info>> sensor_info;
3242

3343
//! @brief Time series of the sensor observations.
3444
std::shared_ptr<time_series::TimeSeriesInterface<Observation>> observation;
@@ -48,12 +58,15 @@ class SensorData
4858
* @copydoc SensorData
4959
* @see MultiProcessSensorData
5060
*/
51-
template <typename Observation>
52-
class SingleProcessSensorData : public SensorData<Observation>
61+
template <typename Observation, typename Info = None>
62+
class SingleProcessSensorData : public SensorData<Observation, Info>
5363
{
5464
public:
5565
SingleProcessSensorData(size_t history_length = 1000)
5666
{
67+
// sensor_info only contains a single static element, so length is set
68+
// to 1
69+
this->sensor_info = std::make_shared<time_series::TimeSeries<Info>>(1);
5770
this->observation =
5871
std::make_shared<time_series::TimeSeries<Observation>>(
5972
history_length);
@@ -71,27 +84,44 @@ class SingleProcessSensorData : public SensorData<Observation>
7184
* @copydoc SensorData
7285
* @see SingleProcessSensorData
7386
*/
74-
template <typename Observation>
75-
class MultiProcessSensorData : public SensorData<Observation>
87+
template <typename Observation, typename Info = None>
88+
class MultiProcessSensorData : public SensorData<Observation, Info>
7689
{
7790
public:
7891
MultiProcessSensorData(const std::string &shared_memory_id,
7992
bool is_master,
8093
size_t history_length = 1000)
8194
{
95+
// each time series needs its own shared memory ID, so add unique
96+
// suffixes to the given ID.
97+
const std::string shm_id_info = shared_memory_id + "_info";
98+
const std::string shm_id_observation =
99+
shared_memory_id + "_observation";
100+
82101
if (is_master)
83102
{
84103
// the master instance is in charge of cleaning the memory
85-
time_series::clear_memory(shared_memory_id);
104+
time_series::clear_memory(shm_id_info);
105+
time_series::clear_memory(shm_id_observation);
106+
107+
// sensor_info only contains a single static element, so length is
108+
// set to 1
109+
this->sensor_info =
110+
time_series::MultiprocessTimeSeries<Info>::create_leader_ptr(
111+
shm_id_info, 1);
86112

87113
this->observation = time_series::MultiprocessTimeSeries<
88-
Observation>::create_leader_ptr(shared_memory_id,
114+
Observation>::create_leader_ptr(shm_id_observation,
89115
history_length);
90116
}
91117
else
92118
{
119+
this->sensor_info =
120+
time_series::MultiprocessTimeSeries<Info>::create_follower_ptr(
121+
shm_id_info);
122+
93123
this->observation = time_series::MultiprocessTimeSeries<
94-
Observation>::create_follower_ptr(shared_memory_id);
124+
Observation>::create_follower_ptr(shm_id_observation);
95125
}
96126
}
97127
};

include/robot_interfaces/sensors/sensor_driver.hpp

+14-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@
1010

1111
#include <iostream>
1212

13+
#include <robot_interfaces/utils.hpp>
14+
1315
namespace robot_interfaces
1416
{
1517
/**
@@ -18,7 +20,7 @@ namespace robot_interfaces
1820
*
1921
* @tparam ObservationType
2022
*/
21-
template <typename ObservationType>
23+
template <typename ObservationType, typename InfoType = None>
2224
class SensorDriver
2325
{
2426
public:
@@ -27,6 +29,17 @@ class SensorDriver
2729
{
2830
}
2931

32+
/**
33+
* @brief Return static information about the sensor.
34+
*
35+
* This information is expected to be constructed during initialization and
36+
* to not change later on.
37+
*/
38+
virtual InfoType get_sensor_info()
39+
{
40+
return InfoType();
41+
}
42+
3043
/**
3144
* @brief return the observation
3245
* @return depends on the observation structure

include/robot_interfaces/sensors/sensor_frontend.hpp

+13-5
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
#include <time_series/time_series.hpp>
1616

1717
#include <robot_interfaces/sensors/sensor_data.hpp>
18+
#include <robot_interfaces/utils.hpp>
1819

1920
namespace robot_interfaces
2021
{
@@ -26,23 +27,30 @@ namespace robot_interfaces
2627
*
2728
* @tparam ObservationType
2829
*/
29-
template <typename ObservationType>
30+
template <typename ObservationType, typename InfoType = None>
3031
class SensorFrontend
3132
{
3233
public:
3334
template <typename Type>
3435
using Timeseries = time_series::TimeSeries<Type>;
3536

36-
typedef std::shared_ptr<SensorFrontend<ObservationType>> Ptr;
37-
typedef std::shared_ptr<const SensorFrontend<ObservationType>> ConstPtr;
37+
typedef std::shared_ptr<SensorFrontend<ObservationType, InfoType>> Ptr;
38+
typedef std::shared_ptr<const SensorFrontend<ObservationType, InfoType>>
39+
ConstPtr;
3840
typedef time_series::Timestamp TimeStamp;
3941
typedef time_series::Index TimeIndex;
4042

41-
SensorFrontend(std::shared_ptr<SensorData<ObservationType>> sensor_data)
43+
SensorFrontend(
44+
std::shared_ptr<SensorData<ObservationType, InfoType>> sensor_data)
4245
: sensor_data_(sensor_data)
4346
{
4447
}
4548

49+
InfoType get_sensor_info() const
50+
{
51+
return sensor_data_->sensor_info->newest_element();
52+
}
53+
4654
ObservationType get_observation(const TimeIndex t) const
4755
{
4856
return (*sensor_data_->observation)[t];
@@ -63,7 +71,7 @@ class SensorFrontend
6371
}
6472

6573
private:
66-
std::shared_ptr<SensorData<ObservationType>> sensor_data_;
74+
std::shared_ptr<SensorData<ObservationType, InfoType>> sensor_data_;
6775
};
6876

6977
} // namespace robot_interfaces

0 commit comments

Comments
 (0)