Skip to content

Latest commit

 

History

History
313 lines (227 loc) · 14.5 KB

overview.md

File metadata and controls

313 lines (227 loc) · 14.5 KB

Overview

This document covers the core functionality of Eclipse iceoryx and is intended to quickly get started to set up iceoryx applications.

General

To set up a collection of applications using iceoryx (an iceoryx system), the applications need to initialize a runtime and create publishers and subscribers. The publishers send data of a specific topic which can be received by subscribers of the same topic. To enable publishers to offer their topic and subscribers to subscribe to these offered topics, the middleware daemon, called RouDi, must be running.

But before we get into more details, let's start with a simple example.

A first example

We need to create a runtime with a unique name along all applications for each application to enable communication with RouDi.

iox::runtime::PoshRuntime::initRuntime("some_unique_name");

Now this application is ready to communicate with RouDi and we can define the data type we want to send.

struct CounterTopic
{
    uint32_t counter;
};

Then we create a publisher that offers our CounterTopic.

iox::popo::Publisher<CounterTopic> publisher({"Group", "Instance", "CounterTopic"});

Now we can use the publisher to send the data.

auto result = publisher.loan();
if(!result.has_error())
{
    auto& sample = result.value();
    sample->counter = 30;
    sample.publish();
}
else
{
    // handle the error
}

Here result is an expected and hence we may get an error. This can happen if we try to loan too many samples and exhaust memory. We have to handle this potential error since the expected class has the nodiscard keyword attached. This means we get a warning (or an error when build in strict mode) when we don't handle it. We could also explicitly discard it with IOX_DISCARD_RESULT which is discouraged. If you want to know more about expected, take a look at How optional and error values are returned in iceoryx.

Let's create a corresponding subscriber.

iox::popo::Subscriber<CounterTopic> subscriber({"Group", "Instance", "CounterTopic"});

Now we can use the subscriber to receive data. For simplicity, we assume that we periodically check for new data. It is also possible to explicitly wait for data using the WaitSet or the Listener. The code to receive the data is the same, the only difference is the way we wake up before checking for data.

while (keepRunning)
{
    // wait for new data (either sleep and wake up periodically or by notification from the waitset)

    auto result = subscriber.take();

    if(!result.has_error())
    {
        auto& sample = result.value();
        uint32_t counter = sample->counter;
        //process the data
    }
    else
    {
        //handle the error
    }

    std::this_thread::sleep_for(std::chrono::milliseconds(100));
}

By calling take we get an expected and hence we have to handle the potential error.

And that's it. We have created our first simple iceoryx example. Here you can find further examples which demonstrate how iceoryx can be used and describe our API in more detail.

Now that we have applications capable of sending and receiving data, we can run the complete iceoryx system.

!!! note RouDi and all Apps have to be built with the same compiler and the same compiler flags.

First we need to start RouDi.

# If installed and available in PATH environment variable
iox-roudi
# If build from scratch with script in tools
$ICEORYX_ROOT/build/iox-roudi

Afterwards, we can start the applications which immediately connect to the RouDi via their runtime.

When the application terminates, the runtime cleans up all resources needed for communication with RouDi. This includes all memory chunks used for the data transmission which may still be held by the application.

We now briefly define the main entities of an iceoryx system which were partially already used in the example above.

RouDi

RouDi is an abbreviation for Routing and Discovery. RouDi takes care of the communication setup but does not actually participate in the communication between the publisher and the subscriber. RouDi can be thought of as the switchboard operator of iceoryx. One of its other major tasks is the setup of the shared memory, which the applications use for exchanging payload data. Sometimes referred to as daemon, RouDi manages the shared memory and is responsible for the service discovery, i.e. enabling subscribers to find topics offered by publishers. It also keeps track of all applications which have initialized a runtime and are hence able to use publishers or subscribers.

When an application crashes, RouDi cleans up all resources. Due to our mostly lock-free inter-process mechanisms (only one last lock; we are working to remove it), iceoryx-based communication is much more reliable compared to traditional mechanism using locking.

To view the available command line options for RouDi call $ICEORYX_ROOT/build/iox-roudi --help.

Shared memory

To enable zero-copy inter-process communication, iceoryx uses the shared memory approach, i.e. publishers and subscribers can communicate via shared memory resulting in zero-copy communication.

Shared memory is physical memory that is made accessible to multiple processes via a mapping to a memory area in their virtual address spaces.

For further information have a look at our conceptual guide.

Runtime

Each application that wants to use iceoryx has to instantiate its runtime, which essentially enables communication with RouDi.

To do so, the following lines of code are required

iox::runtime::PoshRuntime::initRuntime("some_unique_application_name");

The runtime is an object inside the user application that maps the shared memory into the user application's address space.

!!! note Only one runtime object per user application is allowed.

Creating service descriptions for topics

A ServiceDescription in iceoryx represents a topic under which publisher and subscribers can exchange data and is uniquely identified by three string identifiers.

  1. Group name
  2. Instance name
  3. Topic name

A triple consisting of such strings is called a ServiceDescription. Two ServiceDescriptions are considered matching if all these three strings are element-wise equal, i.e. group, instance and topic names are the same for both of them. This means the group and instance identifier can be ignored to create different ServiceDescriptions. They will be needed for advanced filtering functionality in the future.

The service model of iceoryx is derived from AUTOSAR and is still used in the API with these names (Service, Instance, Event). The so called canonical protocol is implemented in the namespace capro.

The following table gives an overview of the different terminologies and the current mapping:

Group Instance Topic
rmw_iceoryx Type Namespace/Topic -
AUTOSAR Service Instance Event
DDS Gateway - - /Group/Instance/Topic
Cyclone DDS - Type Name Topic Name

Service is related to instance like classes are related to objects in C++. Service describes an abstract topic and an instance is one instantiation of that abstraction. Like an object is an instantiated class. Events are in this context like members of a class.

Example:

class MyRadarService {
   public:
      bool hasObstacleDetected;
      float distanceToObstacle;
};

MyRadarService frontLeftRadarInstance;
std::cout << frontLeftRadarInstance.hasObstacleDetected << std::endl;

In the iceoryx world we would subscribe to the service ("MyRadarService", "frontLeftRadarInstance", "hasObstacleDetected") and would receive a sample whenever an obstacle was detected. Or we would subscribe to distanceToObstacle and would receive a constant stream of data which presents the distance to the obstacle.

Restrictions

The data type of the transmitted data can be any C++ class, struct or plain old data type as long as it satisfies the following conditions:

  • no heap is used
  • the data structure is entirely contained in the shared memory - no pointers to process local memory, no references to process local constructs, no dynamic allocators
  • the data structure has to be relocatable and therefore must not internally use pointers/references
  • no virtual member functions

!!! note Most of the STL types cannot be used, but we reimplemented some of them so that they meet the conditions above. You can find an overview here.

Publisher

A publisher is tied to a topic and needs a service description to be constructed. If it is typed, one needs to additionally specify the data type as a template parameter. Otherwise, the publisher is only aware of raw memory and the user has to ensure that it is interpreted correctly.

Once it has offered its topic, it is able to publish (send) data of the specific type. Note that the default is to have multiple publishers for the same topic (n:m communication). A compile-time option to restrict iceoryx to 1:n communication is available. Should 1:n communication be used RouDi checks for multiple publishers on the same topics and raises an error if there is more than one publisher for a topic.

Subscriber

Symmetrically a subscriber also corresponds to a topic and thus needs a service description to be constructed. As for publishers we distinguish between typed and untyped subscribers.

Once a subscriber is subscribed to some topic, it is able to receive data of the type tied to this topic. In the untyped case this is raw memory and the user must take care that it is interpreted in a way that is compatible to the data that was actually sent.

When multiple publishers have offered the same topic the subscriber will receive the data of all of them (but in indeterminate order between different publishers).

Avoid polling

The easiest way to receive data is to periodically poll whether data is available. This is sufficient for simple use cases but inefficient in general, as it often leads to unnecessary latency and wake-ups without data. An alternative approach to receive data is to wait for user-defined events to occur. This is provided by our WaitSet and Listener which are introduced in the following sections.

WaitSet

The WaitSet can be used to relinquish control (putting the thread to sleep) and wait for user-defined events to occur. Usually, these events correspond to the availability of data at specific subscribers. This way we can immediately wake up when data is available and avoid unnecessary wake-ups if no data is available.

One typical use case is to create a WaitSet, attach multiple subscribers and user triggers and then wait until one or many of the attached objects signal an event. If this happens one receives a list of all occured events. This makes it possible to collect data directly from the subscriber when it signals the WaitSet that new data is available.

The WaitSet uses the reactor pattern and is informed with a push strategy that one of the attached events occured at which it informs the user.

For more information on how to use the WaitSet see our WaitSet examples.

Listener

The Listener can be used to connect custom callbacks to user-defined events. Unlike the WaitSet, it reacts to those events by executing the connected custom callbacks in a background thread.

!!! note The Listener is completely thread-safe but please be aware that most of the objects which can be attached to the Listener are not thread-safe! This means either the object is handled solely by the Listener, which should be the most common use case, or the user has to ensure the thread safety with other means like encapsulating the object in a thread-safe class.

One use case could be that one creates a Listener and attaches multiple subscribers. Every time new data is available, the corresponding connected callback will be executed, e.g. print something to the console or calculate something.

Like the WaitSet, the Listener uses the reactor pattern.

For more information about the Listener see our callbacks example.

API

The API is offered in two languages, C++ and C. Detailed information can be found in the C++ example and C example.

Many parts of the C++ API follow a functional programming approach which is less error-prone. This requires using the monadic types cxx::expected and cxx::optional which are introduced here.

With the C++ API, we distinguish between the typed API and the untyped API. In the typed API, the underlying data type is made apparent by typed pointers or references to some data type T (often a template parameter). This allows working with the data in a C++ idiomatic and type-safe way and should be preferred whenever possible. The typed API is mainly used when iceoryx is used stand-alone, i.e. not integrated into a third party framework.

The untyped API provides opaque (i.e. void) pointers to data, which is flexible and efficient but also requires that the user takes care to interpret received data correctly, i.e. as a type compatible to what was actually sent. This is required for interaction with other lower level APIs and integration into third party frameworks such as ROS. For further information see the respective header files.