icehello🔗
Introduction🔗
This example demonstrates a basic data transmission with zero-copy inter-process communication (IPC). It provides a publisher and a subscriber application.
Expected Output🔗
Code walkthrough🔗
Publisher🔗
At first, we need to define what kind of data type the publisher and subscriber application will exchange:
struct RadarObject
{
double x = 0.0;
double y = 0.0;
double z = 0.0;
};
It is included via:
#include "topic_data.hpp"
Next, we include the publisher and the runtime:
#include "iceoryx_posh/popo/publisher.hpp"
#include "iceoryx_posh/runtime/posh_runtime.hpp"
We create a runtime object to communicate with the RouDi daemon. We use a unique string for identifying our application:
constexpr char APP_NAME[] = "iox-cpp-publisher-helloworld";
iox::runtime::PoshRuntime::initRuntime(APP_NAME);
Now we create a publisher instance for our charming struct. Notice that the topic type is passed as a template parameter:
iox::popo::Publisher<RadarObject> publisher({"Radar", "FrontLeft", "Object"});
The three strings which are passed as parameter to the constructor of iox::popo::Publisher
define our
capro::ServiceDescription
. capro
stands for canionical protocol and is used to abstract different
SoA protocols. Radar
is the service name, FrontLeft
an instance of the service Radar
and the third string the specific event Object
of the instance.
In iceoryx, a publisher and a subscriber are connected only if all three IDs match.
For exiting on Ctrl+C, we use the SignalWatcher
#include "iceoryx_hoofs/posix_wrapper/signal_watcher.hpp"
and loop in our while
loop until it states that SIGINT
or SIGTERM
was sent via
the function hasTerminationRequested
.
while (!iox::posix::hasTerminationRequested())
In order to send our sample, we loan some shared memory inside the while
loop:
auto loanResult = publisher.loan();
If loaning was successful, we assign the incremented counter to all three values in RadarObject
and publish()
to the subscriber application:
if (!loanResult.has_error())
{
auto& sample = loanResult.value();
// Sample can be held until ready to publish
sample->x = ct;
sample->y = ct;
sample->z = ct;
sample.publish();
}
In case an error occurred during loaning, we need to handle it:
else
{
auto error = loanResult.get_error();
// Do something with error
std::cerr << "Unable to loan sample, error code: " << error << std::endl;
}
Topics are printed and published every second:
std::cout << APP_NAME << " sent value: " << ct << std::endl;
std::this_thread::sleep_for(std::chrono::seconds(1));
Subscriber🔗
The subscriber needs to have similar includes, but unlike the publisher subscriber.hpp
is included:
#include "topic_data.hpp"
#include "iceoryx_hoofs/posix_wrapper/signal_watcher.hpp"
#include "iceoryx_posh/popo/subscriber.hpp"
#include "iceoryx_posh/runtime/posh_runtime.hpp"
As well as the publisher, the subscriber needs to register with the daemon RouDi:
constexpr char APP_NAME[] = "iox-cpp-subscriber-helloworld";
iox::runtime::PoshRuntime::initRuntime(APP_NAME);
Next, the subscriber object is created, again passing the topic type RadarObject
as template parameter:
iox::popo::Subscriber<RadarObject> subscriber({"Radar", "FrontLeft", "Object"});
Publisher and subscriber will only be connected if they both use exactly these same three strings, our capro::ServiceDescription
.
Inside the while
loop, we take the sample from shared memory and print it if we acquired it successfully.
auto takeResult = subscriber.take();
if (!takeResult.has_error())
{
std::cout << APP_NAME << " got value: " << takeResult.value()->x << std::endl;
}
In case an error occurred during taking, we need to handle it:
if (takeResult.get_error() == iox::popo::ChunkReceiveResult::NO_CHUNK_AVAILABLE)
{
std::cout << "No chunk available." << std::endl;
}
else
{
std::cout << "Error receiving chunk." << std::endl;
}
The subscriber application polls for the sample ten times faster than the publisher is sending it. Therefore no samples should be missed, but not every time the subscriber tries to take a sample, it will get some. In this case, we print "No chunk available.".
std::this_thread::sleep_for(std::chrono::milliseconds(100));
Increasing the polling rate is just one approach for reliable communication. iceoptions explains how to configure the history size of a subscriber. In the WaitSet example you learn how to avoid polling altogether.