Go2Py/deploy/dds_bridge/thirdparty/include/ddscxx/dds/sub/TDataReader.hpp

1690 lines
78 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#ifndef OMG_DDS_SUB_TDATA_READER_HPP_
#define OMG_DDS_SUB_TDATA_READER_HPP_
/* Copyright 2010, Object Management Group, Inc.
* Copyright 2010, PrismTech, Corp.
* Copyright 2010, Real-Time Innovations, Inc.
* All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <dds/core/detail/conformance.hpp>
#include <dds/sub/AnyDataReader.hpp>
#include <dds/topic/ContentFilteredTopic.hpp>
#include <dds/topic/TopicInstance.hpp>
#include <dds/sub/LoanedSamples.hpp>
#include <dds/sub/Subscriber.hpp>
namespace dds
{
namespace sub
{
template <typename T, template <typename Q> class DELEGATE>
class DataReader;
template <typename T>
class DataReaderListener;
}
}
/**
* @brief
* DataReader allows the applicatin to access published sample data.
*
* A DataReader allows the application:
* - to declare the data it wishes to receive (i.e., make a subscription)
* - to access the data received by the attached Subscriber
*
* A DataReader refers to exactly one TopicDescription (either a Topic, a
* ContentFilteredTopic or a MultiTopic) that identifies the samples to be
* read. The Topic must exist prior to the DataReader creation.
*
* A DataReader is attached to exactly one Subscriber which acts as a factory
* for it.
*
* The DataReader may give access to several instances of the data type, which
* are distinguished from each other by their key.
*
* The pre-processor generates from IDL type descriptions the application
* DataReader<type> classes. For each application data type that is used as Topic
* data type, a typed class DataReader<type> is derived from the AnyDataReader
* class.
*
* For instance, for an application, the definitions are located in the Foo.idl file.
* The pre-processor will generate a ccpp_Foo.h include file.
*
* <b>General note:</b> The name ccpp_Foo.h is derived from the IDL file Foo.idl,
* that defines Foo::Bar, for all relevant DataReader<Foo::Bar> operations.
*
* @note Apart from idl files, Google protocol buffers are also supported. For the
* API itself, it doesn't matter if the type header files were generated from
* idl or protocol buffers. The resulting API usage and includes remain the same.
*
* @anchor anchor_dds_sub_datareader_example
* <b>Example</b>
* @code{.cpp}
* // Default creation of a DataReader
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* {
* // Default read of a sample on the DataReader
* dds::sub::LoanedSamples<Foo::Bar> samples;
* samples = reader.read();
*
* // Default way of accessing the loaned samples
* dds::sub::LoanedSamples<Foo::Bar>::const_iterator it;
* for (it = samples.begin(); it != samples.end(); ++it) {
* const dds::sub::Sample<Foo::Bar>& sample = *it;
* const Foo::Bar& data = sample.data();
* const dds::sub::SampleInfo& info = sample.info();
* // Use sample data and meta information.
* }
* }
* // Like the name says, the read samples are loans from the DataReader. The loan
* // was automatically returned when the LoanedSamples went out of scope.
* @endcode
*
* @see for more information: @ref DCPS_Modules_Subscription "Subscription concept"
* @see for more information: @ref DCPS_Modules_Subscription_DataReader "DataReader concept"
*/
template <typename T, template <typename Q> class DELEGATE>
class dds::sub::DataReader : public dds::sub::TAnyDataReader< DELEGATE<T> >
{
public:
/**
* Local convenience typedef for dds::sub::DataReaderListener.
*/
typedef ::dds::sub::DataReaderListener<T> Listener;
public:
/**
* The Selector class is used by the DataReader to compose read operations.
*
* A Selector can perform complex data selections, such as per-instance selection,
* content and status filtering, etc, when reading or taking samples. These settings
* on a Selector can be concatenated.
*
* The DataReader has the select() operation, which can be used to aqcuire the Selector
* functionality on the reader implicitly.
* @code{.cpp}
* // Take a maximum of 3 new samples of a certain instance.
* samples = reader.select()
* .max_samples(3)
* .state(dds::sub::status::DataState::new_data())
* .instance(someValidInstanceHandle)
* .take();
* @endcode
* However, this will create and destroy a Selector for every read, which is not
* very performance friendly.
*
* The performance can be increase by creating a Selector up front and doing the
* reading on that Selector directly and re-using it.
* @code{.cpp}
* // Create a Selector as selective reader up front.
* dds::sub::DataReader<Foo::Bar>::Selector selectiveReader(reader);
* // Configure it to take a maximum of 3 new samples of a certain instance
* selectiveReader.max_samples(3);
* selectiveReader.state(dds::sub::status::DataState::new_data());
* selectiveReader.instance(someValidInstanceHandle);
*
* // Use the configured Selector to -take- a maximum of 3 new samples of a
* // certain instance (which it was configured to do).
* // This can be used in loops for example, reducing the need for creating
* // implicit Selectors for every take.
* samples = selectiveReader.take();
* @endcode
*
* <i>Defaults</i>
* Element | Default Value
* --------------- | --------------------
* state | dds::sub::status::DataState::any
* content | Empty dds::sub::Query
* max_samples | dds::core::LENGTH_UNLIMITED
* instance | dds::core::InstanceHandle nil
*
* @see for more information: @link dds::sub::DataReader::select() DataReader select() @endlink
*/
class Selector
{
public:
/**
* Construct a Selector for a DataReader.
*
* @param dr DataReader
*/
Selector(DataReader& dr);
/**
* Set InstanceHandle to filter with during the read or take.
*
* <i>Example</i><br>
* Read only samples of the given instance.
* @code{.cpp}
* dds::core::InstanceHandle hdl = someValidInstanceHandle;
*
* // Implicit use of Selector
* samples = reader.select().instance(hdl).read();
*
* // Explicit use of Selector
* dds::sub::DataReader<Foo::Bar>::Selector selectiveReader(reader);
* selectiveReader.instance(hdl);
* samples = selectiveReader.read();
* @endcode
* See also @link dds::sub::DataReader::select() DataReader select() @endlink operation.
*
* @param handle the InstanceHandle to read/take for
* @return a Selector to be able to concatenate Selector settings.
*/
Selector& instance(const dds::core::InstanceHandle& handle);
/**
* Set next InstanceHandle to filter with during the read or take.
*
* <i>Example</i><br>
* Read all samples, instance by instance.
* @code{.cpp}
* // Implicit use of Selector
* {
* // Get sample(s) of first instance
* dds::core::InstanceHandle hdl; //nil
* samples = reader.select().next_instance(hdl).read();
* while (samples.length() > 0) {
* // Handle the sample(s) of this instance (just the first one in this case)
* const dds::sub::Sample<Foo::Bar>& sample = *(samples.begin());
* // Get sample(s) of the next instance
* hdl = sample.info().instance_handle();
* samples = reader.select().next_instance(hdl).read();
* }
* }
*
* // Explicit use of Selector
* {
* // Get sample(s) of first instance
* dds::sub::DataReader<Foo::Bar>::Selector selectiveReader(reader);
* dds::core::InstanceHandle hdl; //nil
* selectiveReader.next_instance(hdl);
* samples = selectiveReader.read();
* while (samples.length() > 0) {
* // Handle the sample(s) of this instance (just the first one in this case)
* const dds::sub::Sample<Foo::Bar>& sample = *(samples.begin());
* // Get sample(s) of the next instance
* hdl = sample.info().instance_handle();
* selectiveReader.next_instance(hdl);
* samples = selectiveReader.read();
* }
* }
* @endcode
* See also @link dds::sub::DataReader::select() DataReader select() @endlink operation.
*
* @param handle the 'previous' InstanceHandle associated with new the read/take
* @return a Selector to be able to concatenate Selector settings.
*/
Selector& next_instance(const dds::core::InstanceHandle& handle);
/**
* Set DataState to filter with during the read or take.
*
* <i>Example</i><br>
* Read only new data.
* @code{.cpp}
* // DataState to filter only new data
* dds::sub::status::DataState newData = dds::sub::status::DataState::new_data();
*
* // Implicit use of Selector
* samples = reader.select().state(newData).read();
*
* // Explicit use of Selector
* dds::sub::DataReader<Foo::Bar>::Selector selectiveReader(reader);
* selectiveReader.state(newData);
* samples = selectiveReader.read();
* @endcode
* See also @link dds::sub::DataReader::select() DataReader select() @endlink operation.
*
* @param state the requested DataState of the samples
* @return a Selector to be able to concatenate Selector settings.
*/
Selector& state(const dds::sub::status::DataState& state);
/**
* Set the Query to filter with during the read or take.
*
* <i>Example</i><br>
* Read only samples that will be filtered according to the given dds::sub::Query.
* @code{.cpp}
* // Assume data type has an element called long_1
* dds::sub::Query query(reader, "long_1 > 1 and long_1 < 7");
*
* // Implicit use of Selector
* samples = reader.select().content(query).read();
*
* // Explicit use of Selector
* dds::sub::DataReader<Foo::Bar>::Selector selectiveReader(reader);
* selectiveReader.content(query);
* samples = selectiveReader.read();
* @endcode
* See also @link dds::sub::DataReader::select() DataReader select() @endlink operation.
*
* @param query the Query to apply to the selector
* @return a Selector to be able to concatenate Selector settings.
*/
Selector& content(const dds::sub::Query& query);
/**
* Set max_samples to limit the number of sample to get during the read or take.
*
* <i>Example</i><br>
* Read a maximum of three samples.
* @code{.cpp}
* // Implicit use of Selector
* samples = reader.select().max_samples(3).read();
*
* // Explicit use of Selector
* dds::sub::DataReader<Foo::Bar>::Selector selectiveReader(reader);
* selectiveReader.max_samples(3);
* samples = selectiveReader.read();
* @endcode
* See also @link dds::sub::DataReader::select() DataReader select() @endlink operation.
*
* @param maxsamples maximum number of samples to read/take
* @return a Selector to be able to concatenate Selector settings.
*/
Selector& max_samples(uint32_t maxsamples);
/**
* This operation works the same as the @link DataReader::read() default
* DataReader read() @endlink, except that it is performed on this Selector
* with possible filters set.
*
* @return The samples in the LoanedSamples container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
dds::sub::LoanedSamples<T> read();
/**
* This operation works the same as the @link DataReader::take() default
* DataReader take() @endlink, except that it is performed on this Selector
* with possible filters set.
*
* @return The samples in the LoanedSamples container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
dds::sub::LoanedSamples<T> take();
// --- Forward Iterators: --- //
/**
* This operation works the same as the @link DataReader::read(SamplesFWIterator sfit, uint32_t max_samples)
* forward iterator DataReader read() @endlink, except that it is performed on this
* Selector with possible filters set.
*
* @param sfit Forward-inserting container iterator
* @param max_samples Maximum samples to read and copy into the given container
* @return The number of samples in the LoanedSamples container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename SamplesFWIterator>
uint32_t
read(SamplesFWIterator sfit, uint32_t max_samples);
/**
* This operation works the same as the @link DataReader::take(SamplesFWIterator sfit, uint32_t max_samples)
* forward iterator DataReader take() @endlink, except that it is performed on this
* Selector with possible filters set.
*
* @param sfit Forward-inserting container iterator
* @param max_samples Maximum samples to read and copy into the given container
* @return The number of samples in the given container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename SamplesFWIterator>
uint32_t
take(SamplesFWIterator sfit, uint32_t max_samples);
// --- Back-Inserting Iterators: --- //
/**
* This operation works the same as the @link DataReader::read(SamplesBIIterator sbit)
* backward iterator DataReader read() @endlink, except that it is performed on this
* Selector with possible filters set.
*
* @param sbit Back-inserting container iterator
* @return The number of samples in the given container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename SamplesBIIterator>
uint32_t
read(SamplesBIIterator sbit);
/**
* This operation works the same as the @link DataReader::take(SamplesBIIterator sbit)
* backward iterator DataReader take() @endlink, except that it is performed on this
* Selector with possible filters set.
*
* @param sbit Back-inserting container iterator
* @return The number of samples in the given container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename SamplesBIIterator>
uint32_t
take(SamplesBIIterator sbit);
private:
typename DELEGATE<T>::Selector impl_;
};
/**
* The ManipulatorSelector class is used by the DataReader to compose streaming
* read operations.
*
* A ManipulatorSelector can perform complex data selections, such as per-instance
* selection, content and status filtering, etc, when reading or taking samples
* through the streaming operator.
*
* <i>Convenience functors</i><br>
* The following convenience functors use a ManipulatorSelector implicitly and can be
* used in the streaming operator:
* - dds::sub::read
* - dds::sub::take
* - dds::sub::max_samples
* - dds::sub::content
* - dds::sub::state
* - dds::sub::instance
* - dds::sub::next_instance
*
* @code{.cpp}
* // Take a maximum of 3 new samples of a certain instance.
* reader >> dds::sub::take
* >> dds::sub::max_samples(3)
* >> dds::sub::state(dds::sub::status::DataState::new_data())
* >> dds::sub::instance(someValidInstanceHandle)
* >> samples;
* @endcode
* However, this will create and destroy ManipulatorSelectors and Functors for
* every read, which is not very performance friendly.
*
* The performance can be increase by creating a ManipulatorSelector up front and
* doing the reading on that ManipulatorSelector directly and re-using it.
* @code{.cpp}
* // Create a ManipulatorSelector as selective reader up front.
* dds::sub::DataReader<Foo::Bar>::ManipulatorSelector selectiveReader(reader);
* // Configure it to take a maximum of 3 new samples of a certain instance
* selectiveReader.max_samples(3);
* selectiveReader.state(dds::sub::status::DataState::new_data());
* selectiveReader.instance(someValidInstanceHandle);
* selectiveReader.read_mode(false); // take
*
* // Use the configured ManipulatorSelector to -take- a maximum of 3 samples of a
* // certain instance (which it was configured to do).
* // This can be used in loops for example, reducing the need for creating
* // implicit ManipulatorSelectors for every take.
* selectiveReader >> samples;
* @endcode
*
* <i>Defaults</i>
* Element | Default Value
* --------------- | --------------------
* read_mode | true (read)
* state | dds::sub::status::DataState::any
* content | Empty dds::sub::Query
* max_samples | dds::core::LENGTH_UNLIMITED
* instance | dds::core::InstanceHandle nil
*
* @see for more information: @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls) DataReader stream operator>> @endlink
*/
class ManipulatorSelector
{
public:
/**
* Construct a ManipulatorSelector for a DataReader.
*
* See also @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink
*
* @param DataReader
*/
ManipulatorSelector(DataReader& dr);
/**
* Get the read_mode.
*
* The read_mode specifies if a sample should be read or taken:
* - true = read (default)
* - false = take
*
* @return true if read_mode is set to read
*/
bool read_mode();
/**
* Set the read_mode.
*
* The read_mode specifies if a sample should be read or taken:
* - true = read (default)
* - false = take
*
* <i>Convenience Functor:</i> dds::sub::read<br>
* <i>Convenience Functor:</i> dds::sub::take
*
* <i>Example</i><br>
* Determine to read or take samples.
* @code{.cpp}
* // No usage of ManipulatorSelector, means a read iso take as default.
* reader >> samples;
*
* // Implicit use of ManipulatorSelector
* reader >> dds::sub::read >> samples;
* reader >> dds::sub::take >> samples;
*
* // Explicit use of ManipulatorSelector
* dds::sub::DataReader<Foo::Bar>::ManipulatorSelector readingReader(reader);
* readingReader.read_mode(true); // Read, which is already the default.
* readingReader >> samples;
*
* dds::sub::DataReader<Foo::Bar>::ManipulatorSelector takingReader(reader);
* takingReader.read_mode(false); // Take.
* takingReader >> samples;
* @endcode
* See also @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink
*
* @param readmode the read mode of the DataReader
*/
void read_mode(bool readmode);
/**
* Set max_samples to limit the number of sample to get during the read or take.
*
* <i>Convenience Functor:</i> dds::sub::max_samples
*
* <i>Example</i><br>
* Read a maximum of three samples.
* @code{.cpp}
* // Implicit use of ManipulatorSelector
* reader >> dds::sub::max_samples(3) >> samples;
*
* // Explicit use of ManipulatorSelector
* dds::sub::DataReader<Foo::Bar>::ManipulatorSelector selectiveReader(reader);
* selectiveReader.max_samples(3);
* selectiveReader >> samples;
* @endcode
* See also @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink
*
* @param n maximum number of samples
*/
ManipulatorSelector& max_samples(uint32_t n);
/**
* Set InstanceHandle to filter with during the read or take.
*
* <i>Convenience Functor:</i> dds::sub::instance
*
* <i>Example</i><br>
* Read only samples of the given instance.
* @code{.cpp}
* dds::core::InstanceHandle hdl = someValidInstanceHandle;
*
* // Implicit use of ManipulatorSelector
* reader >> dds::sub::instance(hdl) >> samples;
*
* // Explicit use of ManipulatorSelector
* dds::sub::DataReader<Foo::Bar>::ManipulatorSelector selectiveReader(reader);
* selectiveReader.instance(hdl);
* selectiveReader >> samples;
* @endcode
* See also @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink
*
* @param handle the InstanceHandle for the read/take
*/
ManipulatorSelector& instance(const dds::core::InstanceHandle& handle);
/**
* Set next InstanceHandle to filter with during the read or take.
*
* <i>Convenience Functor:</i> dds::sub::next_instance
*
* <i>Example</i><br>
* Read all samples, instance by instance.
* @code{.cpp}
* // Implicit use of ManipulatorSelector
* {
* // Get sample(s) of first instance
* dds::core::InstanceHandle hdl; //nil
* reader >> dds::sub::next_instance(hdl) >> samples;
* while (samples.length() > 0) {
* // Handle the sample(s) of this instance (just the first one in this case)
* const dds::sub::Sample<Foo::Bar>& sample = *(samples.begin());
* // Get sample(s) of the next instance
* hdl = sample.info().instance_handle();
* reader >> dds::sub::next_instance(hdl) >> samples;
* }
* }
*
* // Explicit use of ManipulatorSelector
* {
* // Get sample(s) of first instance
* dds::sub::DataReader<Foo::Bar>::ManipulatorSelector selectiveReader(reader);
* dds::core::InstanceHandle hdl; //nil
* selectiveReader.next_instance(hdl);
* selectiveReader >> samples;
* while (samples.length() > 0) {
* // Handle the sample(s) of this instance (just the first one in this case)
* const dds::sub::Sample<Foo::Bar>& sample = *(samples.begin());
* // Get sample(s) of the next instance
* hdl = sample.info().instance_handle();
* selectiveReader.next_instance(hdl);
* selectiveReader >> samples;
* }
* }
* @endcode
* See also @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink
*
* @param handle the 'previous' InstanceHandle associated with new the read/take
*/
ManipulatorSelector& next_instance(const dds::core::InstanceHandle& handle);
/**
* Set DataState to filter with during the read or take.
*
* <i>Convenience Functor:</i> dds::sub::state
*
* <i>Example</i><br>
* Read only new data.
* @code{.cpp}
* // DataState to filter only new data
* dds::sub::status::DataState newData = dds::sub::status::DataState::new_data();
*
* // Implicit use of ManipulatorSelector
* reader >> dds::sub::state(newData) >> samples;
*
* // Explicit use of ManipulatorSelector
* dds::sub::DataReader<Foo::Bar>::ManipulatorSelector selectiveReader(reader);
* selectiveReader.state(newData);
* selectiveReader.read() >> samples;
* @endcode
* See also @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink
*
* @param state the required DataState of the samples
*/
ManipulatorSelector& state(const dds::sub::status::DataState& state);
/**
* Set Query to filter with during the read or take.
*
* <i>Convenience Functor:</i> dds::sub::content
*
* <i>Example</i><br>
* Read only samples that will be filtered according to the given dds::sub::Query.
* @code{.cpp}
* // Assume data type has an element called long_1
* dds::sub::Query query(reader, "long_1 > 1 and long_1 < 7");
*
* // Implicit use of ManipulatorSelector
* reader >> dds::sub::content(query) >> samples;
*
* // Explicit use of ManipulatorSelector
* dds::sub::DataReader<Foo::Bar>::ManipulatorSelector selectiveReader(reader);
* selectiveReader.content(query);
* selectiveReader >> read;
* @endcode
* See also @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink
*
* @param query The Query to apply to a read/take
*/
ManipulatorSelector& content(const dds::sub::Query& query);
/**
* This operation works the same as the @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink, except that it is performed on this ManipulatorSelector
* with possible filters set.
*
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
ManipulatorSelector&
operator >>(dds::sub::LoanedSamples<T>& samples);
/**
* This operation works the same as the @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink, except that it is performed on this ManipulatorSelector
* with possible filters set.
*
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
ManipulatorSelector&
operator >> (ManipulatorSelector & (manipulator)(ManipulatorSelector&));
/**
* This operation works the same as the @link dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls)
* DataReader stream operator>> @endlink, except that it is performed on this ManipulatorSelector
* with possible filters set.
*
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename Functor>
ManipulatorSelector
operator >> (Functor f);
private:
typename DELEGATE<T>::ManipulatorSelector impl_;
};
public:
OMG_DDS_REF_TYPE_PROTECTED_DC_T(DataReader, dds::sub::TAnyDataReader, T, DELEGATE)
OMG_DDS_IMPLICIT_REF_BASE(DataReader)
OMG_DDS_COMPLETE_RULE_OF_FIVE_VIRTUAL_DEFAULT(DataReader)
public:
/**
* Create a new DataReader for the desired Topic, ContentFilteredTopic or MultiTopic,
* using the given Subscriber.
*
* <i>QoS</i><br>
* The DataReader will be created with the QoS values specified on the last
* successful call to @link dds::sub::Subscriber::default_datareader_qos(const dds::sub::qos::DataReaderQos& qos)
* sub.default_datareader_qos(qos) @endlink or, if the call was never made,
* the @ref anchor_dds_sub_datareader_qos_defaults "default" values.
*
* @param sub the Subscriber that will contain this DataReader
* @param topic the Topic associated with this DataReader
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
*/
DataReader(const dds::sub::Subscriber& sub,
const ::dds::topic::Topic<T>& topic);
/**
* Create a new DataReader for the desired Topic, ContentFilteredTopic or MultiTopic,
* using the given Subscriber and DataReaderQos and attaches the optionally specified
* DataReaderListener to it.
*
* <i>QoS</i><br>
* A possible application pattern to construct the DataReaderQos for the
* DataReader is to:
* @code{.cpp}
* // 1) Retrieve the QosPolicy settings on the associated Topic
* dds::topic::qos::TopicQos topicQos = topic.qos();
* // 2) Retrieve the default DataReaderQos from the related Subscriber
* dds::sub::qos::DataReaderQos readerQos = subscriber.default_datareader_qos();
* // 3) Combine those two lists of QosPolicy settings by overwriting DataReaderQos
* // policies that are also present TopicQos
* readerQos = topicQos;
* // 4) Selectively modify QosPolicy settings as desired.
* readerQos << dds::core::policy::Durability::Transient();
* // 5) Use the resulting QoS to construct the DataReader.
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic, readerQos);
* @endcode
*
* <i>Listener</i><br>
* The following statuses are applicable to the DataReaderListener:
* - dds::core::status::StatusMask::requested_deadline_missed()
* - dds::core::status::StatusMask::requested_incompatible_qos()
* - dds::core::status::StatusMask::sample_lost()
* - dds::core::status::StatusMask::sample_rejected()
* - dds::core::status::StatusMask::data_available()
* - dds::core::status::StatusMask::liveliness_changed()
* - dds::core::status::StatusMask::subscription_matched()
*
* See @ref DCPS_Modules_Infrastructure_Listener "listener concept",
* @ref anchor_dds_sub_datareader_commstatus "communication status" and
* @ref anchor_dds_sub_datareader_commpropagation "communication propagation"
* for more information.
*
* @param sub the Subscriber that will contain this DataReader
* @param topic the Topic, ContentFilteredTopic or MultiTopic associated with this DataReader
* @param qos the DataReader qos.
* @param listener the DataReader listener.
* @param mask the listener event mask.
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::InconsistentPolicyError
* The parameter qos contains conflicting QosPolicy settings.
*/
DataReader(const dds::sub::Subscriber& sub,
const ::dds::topic::Topic<T>& topic,
const dds::sub::qos::DataReaderQos& qos,
dds::sub::DataReaderListener<T>* listener = NULL,
const dds::core::status::StatusMask& mask = ::dds::core::status::StatusMask::none());
#ifdef OMG_DDS_CONTENT_SUBSCRIPTION_SUPPORT
/** @copydoc dds::sub::DataReader::DataReader(const dds::sub::Subscriber& sub, const ::dds::topic::Topic<T>& topic) */
DataReader(const dds::sub::Subscriber& sub,
const ::dds::topic::ContentFilteredTopic<T>& topic);
/** @copydoc dds::sub::DataReader::DataReader(const dds::sub::Subscriber& sub, const ::dds::topic::Topic<T>& topic, const dds::sub::qos::DataReaderQos& qos, dds::sub::DataReaderListener<T>* listener, const dds::core::status::StatusMask& mask) */
DataReader(const dds::sub::Subscriber& sub,
const ::dds::topic::ContentFilteredTopic<T>& topic,
const dds::sub::qos::DataReaderQos& qos,
dds::sub::DataReaderListener<T>* listener = NULL,
const dds::core::status::StatusMask& mask = ::dds::core::status::StatusMask::none());
#endif /* OMG_DDS_CONTENT_SUBSCRIPTION_SUPPORT */
#ifdef OMG_DDS_MULTI_TOPIC_SUPPORT
/** @copydoc dds::sub::DataReader::DataReader(const dds::sub::Subscriber& sub, const ::dds::topic::Topic<T>& topic) */
DataReader(const dds::sub::Subscriber& sub,
const ::dds::topic::MultiTopic<T>& topic);
/** @copydoc dds::sub::DataReader::DataReader(const dds::sub::Subscriber& sub, const ::dds::topic::Topic<T>& topic, const dds::sub::qos::DataReaderQos& qos, dds::sub::DataReaderListener<T>* listener, const dds::core::status::StatusMask& mask) */
DataReader(const dds::sub::Subscriber& sub,
const ::dds::topic::MultiTopic<T>& topic,
const dds::sub::qos::DataReaderQos& qos,
dds::sub::DataReaderListener<T>* listener = NULL,
const dds::core::status::StatusMask& mask = ::dds::core::status::StatusMask::none());
#endif /* OMG_DDS_MULTI_TOPIC_SUPPORT */
public:
// == ReadState Management
/**
* Returns the @ref anchor_dds_sub_datareader_defaultstatefilter "default state filter"
* for read/take operations.
*
* The default value of default_filter_state is dds::sub::status::DataState::any().
*
* @return the default state to filter for
*/
dds::sub::status::DataState default_filter_state();
/**
* Set the default state filter for read/take operations.
*
* @anchor anchor_dds_sub_datareader_defaultstatefilter
* <i>Default State Filter</i><br>
* The default_filter_state indicates what the dds::sub::status::DataState of samples
* should be for the read to filter them out of the total data samples pool.
*
* This filter can be overruled by using dds::sub::DataReader::Selector::state or
* dds::sub::DataReader::ManipulatorSelector::state during the actual read action.
*
* @param state the state mask that will be used to read/take samples
*/
DataReader& default_filter_state(const dds::sub::status::DataState& state);
//== Streaming read/take
/**
* This operation reads a sequence of typed samples from the DataReader by means
* of the shift operator.
*
* What samples are read depends on what @link dds::sub::DataReader::default_filter_state(const dds::sub::status::DataState& state)
* default state filter @endlink has been set (default any).
*
* This operation reads a sequence of typed samples from the DataReader<type>. The
* data is put into a dds::sub::LoanedSamples, which is basically a sequence of samples,
* which in turn contains the actual data and meta information.
*
* The memory used for storing the sample may be loaned by the middleware thus allowing zero
* copy operations.
*
* If the DataReader has no samples that meet the constraints, the resulting
* dds::sub::LoanedSamples will be empty.
* @code{.cpp}
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* {
* dds::sub::LoanedSamples<Foo::Bar> samples;
* reader >> samples;
* }
* // The LoanedSamples went out of scope, meaning the loan was returned.
* @endcode
*
* <b><i>Take</i></b><br>
* The default behaviour of the DataReader stream operator>> is to read samples. It can be
* explicitly stated if the operator should do a read or take.
* @code{.cpp}
* reader >> dds::sub::read >> samples;
* reader >> dds::sub::take >> samples;
* @endcode
* These are two of the available convenience manipulators (see next paragraph).
*
* <b><i>Manipulators</i></b><br>
* Manipulators are defined externally to make it possible to control which data is read
* and whether the streaming operators reads or takes.
*
* Available convenience stream manipulators:
* - dds::sub::read
* - dds::sub::take
* - dds::sub::max_samples
* - dds::sub::content
* - dds::sub::state
* - dds::sub::instance
* - dds::sub::next_instance
*
* The manipulators can be concatenated:
* @code{.cpp}
* // Take (iso read) a maximum of 3 new samples of a certain instance.
* reader >> dds::sub::take
* >> dds::sub::max_samples(3)
* >> dds::sub::state(dds::sub::status::DataState::new_data())
* >> dds::sub::instance(someValidInstanceHandle)
* >> samples;
* @endcode
*
* <i>Please be aware that using pre-set filters on the DataReader and then
* call read() or take() on that reader explicitly will perform better than
* having to create Manipulators for every read (which is what essentially
* happens in the code example above). Performance can be increase by creating
* a manipulator up front and using that multiple times
* (see dds::sub::DataReader::ManipulatorSelector).</i>
*
* <b><i>Invalid Data</i></b><br>
* Some elements in the returned sequence may not have valid data: the valid_data
* field in the SampleInfo indicates whether the corresponding data value contains
* any meaningful data. If not, the data value is just a dummy sample for which only
* the keyfields have been assigned. It is used to accompany the SampleInfo that
* communicates a change in the instance_state of an instance for which there is
* no real sample available.
*
* For example, when an application always takes all available samples of a
* particular instance, there is no sample available to report the disposal of that
* instance. In such a case the DataReader will insert a dummy sample into the
* data_values sequence to accompany the SampleInfo element in the info_seq
* sequence that communicates the disposal of the instance.
*
* The act of reading a sample sets its sample_state to READ_SAMPLE_STATE. If
* the sample belongs to the most recent generation of the instance, it also sets the
* view_state of the instance to NOT_NEW_VIEW_STATE. It does not affect the
* instance_state of the instance.
*
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
DataReader& operator>>(dds::sub::LoanedSamples<T>& ls);
/** @copydoc dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls) */
ManipulatorSelector
operator>> (ManipulatorSelector & (manipulator)(ManipulatorSelector&));
/** @copydoc dds::sub::DataReader::operator>>(dds::sub::LoanedSamples<T>& ls) */
template <typename Functor>
ManipulatorSelector
operator>> (Functor f);
///////////////////////////////////////////////////////////////////////
public:
//== Loan Read/Take API ==================================================
/**
* This operation reads a sequence of typed samples from the DataReader.
*
* What samples are read depends on what @link dds::sub::DataReader::default_filter_state(const dds::sub::status::DataState& state)
* default state filter @endlink has been set (default any).
*
* This operation reads a sequence of typed samples from the DataReader<type>. The
* data is put into a dds::sub::LoanedSamples, which is basically a sequence of samples,
* which in turn contains the actual data and meta information.
*
* The memory used for storing the sample may be loaned by the middleware thus allowing zero
* copy operations.
*
* If the DataReader has no samples that meet the constraints, the resulting
* dds::sub::LoanedSamples will be empty.
* @code{.cpp}
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* {
* // Read the available samples.
* dds::sub::LoanedSamples<Foo::Bar> samples;
* samples = reader.read();
*
* // Access the information.
* dds::sub::LoanedSamples<Foo::Bar>::const_iterator it;
* for (it = samples.begin(); it != samples.end(); ++it) {
* const dds::sub::Sample<Foo::Bar>& sample = *it;
* }
* }
* // The LoanedSamples went out of scope, meaning the loan resources were taken care of.
* @endcode
*
* <b><i>Selectors</i></b><br>
* What data is read already depends on the
* @ref anchor_dds_sub_datareader_defaultstatefilter "default state filter".
* But it can be manipulated even more when using dds::sub::DataReader::select()
* operation.
*
* @anchor anchor_dds_sub_datareader_invalidsamples
* <b><i>Invalid Data</i></b><br>
* Some elements in the returned sequence may not have valid data: the valid_data
* field in the SampleInfo indicates whether the corresponding data value contains
* any meaningful data. If not, the data value is just a dummy sample for which only
* the keyfields have been assigned. It is used to accompany the SampleInfo that
* communicates a change in the instance_state of an instance for which there is
* no real sample available.
*
* For example, when an application always takes all available samples of a
* particular instance, there is no sample available to report the disposal of that
* instance. In such a case the DataReader will insert a dummy sample into the
* data_values sequence to accompany the SampleInfo element in the info_seq
* sequence that communicates the disposal of the instance.
*
* The act of reading a sample sets its sample_state to READ_SAMPLE_STATE. If
* the sample belongs to the most recent generation of the instance, it also sets the
* view_state of the instance to NOT_NEW_VIEW_STATE. It does not affect the
* instance_state of the instance.
*
* @return The samples in the LoanedSamples container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
LoanedSamples<T> read();
/**
* This operation takes a sequence of typed samples from the DataReader.
*
* The behaviour is identical to read except for that the samples are removed
* from the DataReader.
*
* What samples are taken depends on what @link dds::sub::DataReader::default_filter_state(const dds::sub::status::DataState& state)
* default state filter @endlink has been set (default any).
*
* This operation takes a sequence of typed samples from the DataReader<type>. The
* data is put into a dds::sub::LoanedSamples, which is basically a sequence of samples,
* which in turn contains the actual data and meta information.
*
* The memory used for storing the sample may be loaned by the middleware thus allowing zero
* copy operations.
*
* If the DataReader has no samples that meet the constraints, the resulting
* dds::sub::LoanedSamples will be empty.
* @code{.cpp}
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* {
* // Take the available samples.
* dds::sub::LoanedSamples<Foo::Bar> samples;
* samples = reader.take();
*
* // Access the information.
* dds::sub::LoanedSamples<Foo::Bar>::const_iterator it;
* for (it = samples.begin(); it != samples.end(); ++it) {
* const dds::sub::Sample<Foo::Bar>& sample = *it;
* }
* }
* // The LoanedSamples went out of scope, meaning the loan resources were taken care of.
* @endcode
*
* <b><i>Selectors</i></b><br>
* What data is taken, already depends on the
* @ref anchor_dds_sub_datareader_defaultstatefilter "default state filter".
* But it can be manipulated even more when using dds::sub::DataReader::select()
* operation.
*
* <b><i>Invalid Data</i></b><br>
* Some elements in the returned sequence may not have valid data: the valid_data
* field in the SampleInfo indicates whether the corresponding data value contains
* any meaningful data.<br>
* Look @ref anchor_dds_sub_datareader_invalidsamples "here" for more information.
*
* @return The samples in the LoanedSamples container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
LoanedSamples<T> take();
//== Copy Read/Take API ==================================================
// --- Forward Iterators: --- //
/**
* This operation reads a sequence of typed samples from the DataReader.
*
* What samples are read depends on what @link dds::sub::DataReader::default_filter_state(const dds::sub::status::DataState& state)
* default state filter @endlink has been set (default any).
*
* The samples are copied into the application provided container using the
* forward iterator parameter.
*
* If the DataReader has no samples that meet the constraints, the resulting
* dds::sub::LoanedSamples will be empty.
* @code{.cpp}
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* // Prepare container to store samples in
* const uint32_t MAX_SAMPLES (3)
* std::vector<dds::sub::Sample<Foo::Bar> > samples(MAX_SAMPLES);
* std::vector<dds::sub::Sample<Foo::Bar> >::iterator iter = samples.begin();
*
* // Read and copy the available samples into the vector by means of the iterator
* uint32_t len = reader.read(iter, MAX_SAMPLES);
*
* // Access the information.
* for (iter = samples.begin(); iter != samples.end(); ++iter) {
* const dds::sub::Sample<Foo::Bar>& sample = *iter;
* }
* @endcode
*
* <b><i>Selectors</i></b><br>
* What data is read already depends on the
* @ref anchor_dds_sub_datareader_defaultstatefilter "default state filter".
* But it can be manipulated even more when using dds::sub::DataReader::select()
* operation.
*
* <b><i>Invalid Data</i></b><br>
* Some elements in the returned sequence may not have valid data: the valid_data
* field in the SampleInfo indicates whether the corresponding data value contains
* any meaningful data.<br>
* Look @ref anchor_dds_sub_datareader_invalidsamples "here" for more information.
*
* @param sfit Forward-inserting container iterator
* @param max_samples Maximum samples to read and copy into the given container
* @return The number of samples in the LoanedSamples container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename SamplesFWIterator>
uint32_t
read(SamplesFWIterator sfit,
uint32_t max_samples);
/**
* This operation takes a sequence of typed samples from the DataReader.
*
* What samples are take depends on what @link dds::sub::DataReader::default_filter_state(const dds::sub::status::DataState& state)
* default state filter @endlink has been set (default any).
*
* The samples are copied into the application provided container using the
* forward iterator parameter.
*
* If the DataReader has no samples that meet the constraints, the resulting
* dds::sub::LoanedSamples will be empty.
* @code{.cpp}
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* // Prepare container to store samples in
* const uint32_t MAX_SAMPLES (3)
* std::vector<dds::sub::Sample<Foo::Bar> > samples(MAX_SAMPLES);
* std::vector<dds::sub::Sample<Foo::Bar> >::iterator iter = samples.begin();
*
* // Take and copy the available samples into the vector by means of the iterator
* uint32_t len = reader.take(iter, MAX_SAMPLES);
*
* // Access the information.
* for (iter = samples.begin(); iter != samples.end(); ++iter) {
* const dds::sub::Sample<Foo::Bar>& sample = *iter;
* }
* @endcode
*
* <b><i>Selectors</i></b><br>
* What data is taken, already depends on the
* @ref anchor_dds_sub_datareader_defaultstatefilter "default state filter".
* But it can be manipulated even more when using dds::sub::DataReader::select()
* operation.
*
* <b><i>Invalid Data</i></b><br>
* Some elements in the returned sequence may not have valid data: the valid_data
* field in the SampleInfo indicates whether the corresponding data value contains
* any meaningful data.<br>
* Look @ref anchor_dds_sub_datareader_invalidsamples "here" for more information.
*
* @param sfit Forward-inserting container iterator
* @param max_samples Maximum samples to take and copy into the given container
* @return The number of samples in the given container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename SamplesFWIterator>
uint32_t
take(SamplesFWIterator sfit,
uint32_t max_samples);
// --- Back-Inserting Iterators: --- //
/**
* This operation reads a sequence of typed samples from the DataReader.
*
* What samples are read depends on what @link dds::sub::DataReader::default_filter_state(const dds::sub::status::DataState& state)
* default state filter @endlink has been set (default any).
*
* The samples are copied into the application provided container using a
* back-inserting iterator. Notice that as a consequence of using a back-inserting
* iterator, this operation may allocate memory to resize the underlying container.
*
* If the DataReader has no samples that meet the constraints, the resulting
* dds::sub::LoanedSamples will be empty.
* @code{.cpp}
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* // Prepare container to store samples in
* std::vector<dds::sub::Sample<Foo::Bar> > samples;
* std::back_insert_iterator< std::vector<dds::sub::Sample<Foo::Bar> > > bi(samples);
*
* // Read and copy the available samples into the vector by means of the iterator
* uint32_t len = reader.read(bi);
*
* // Access the information.
* std::vector<dds::sub::Sample<Space::Type1> >::iterator iter = samples.begin();
* for (iter = samples.begin(); iter != samples.end(); ++iter) {
* const dds::sub::Sample<Foo::Bar>& sample = *iter;
* }
* @endcode
*
* <b><i>Selectors</i></b><br>
* What data is read already depends on the
* @ref anchor_dds_sub_datareader_defaultstatefilter "default state filter".
* But it can be manipulated even more when using dds::sub::DataReader::select()
* operation.
*
* <b><i>Invalid Data</i></b><br>
* Some elements in the returned sequence may not have valid data: the valid_data
* field in the SampleInfo indicates whether the corresponding data value contains
* any meaningful data.<br>
* Look @ref anchor_dds_sub_datareader_invalidsamples "here" for more information.
*
* @param sbit Back-inserting container iterator
* @return The number of samples in the given container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename SamplesBIIterator>
uint32_t
read(SamplesBIIterator sbit);
/**
* This operation takes a sequence of typed samples from the DataReader.
*
* What samples are take depends on what @link dds::sub::DataReader::default_filter_state(const dds::sub::status::DataState& state)
* default state filter @endlink has been set (default any).
*
* The samples are copied into the application provided container using a
* back-inserting iterator. Notice that as a consequence of using a back-inserting
* iterator, this operation may allocate memory to resize the underlying container.
*
* If the DataReader has no samples that meet the constraints, the resulting
* dds::sub::LoanedSamples will be empty.
* @code{.cpp}
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* // Prepare container to store samples in
* std::vector<dds::sub::Sample<Foo::Bar> > samples;
* std::back_insert_iterator< std::vector<dds::sub::Sample<Foo::Bar> > > bi(samples);
*
* // Take and copy the available samples into the vector by means of the iterator
* uint32_t len = reader.take(bi);
*
* // Access the information.
* std::vector<dds::sub::Sample<Space::Type1> >::iterator iter = samples.begin();
* for (iter = samples.begin(); iter != samples.end(); ++iter) {
* const dds::sub::Sample<Foo::Bar>& sample = *iter;
* }
* @endcode
*
* <b><i>Selectors</i></b><br>
* What data is taken, already depends on the
* @ref anchor_dds_sub_datareader_defaultstatefilter "default state filter".
* But it can be manipulated even more when using dds::sub::DataReader::select()
* operation.
*
* <b><i>Invalid Data</i></b><br>
* Some elements in the returned sequence may not have valid data: the valid_data
* field in the SampleInfo indicates whether the corresponding data value contains
* any meaningful data.<br>
* Look @ref anchor_dds_sub_datareader_invalidsamples "here" for more information.
*
* @param sbit Back-inserting container iterator
* @return The number of samples in the given container
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
template <typename SamplesBIIterator>
uint32_t
take(SamplesBIIterator sbit);
public:
//========================================================================
//== DSL Method for dealing with instances, content and status filters.
/**
* Get a dds::sub::DataReader::Selector instance that acts on this DataReader.
*
* The DataReader::read and DataReader::take read all samples that are
* available within the DataReader (provided that the
* @ref anchor_dds_sub_datareader_defaultstatefilter "default state filter"
* hasn't been changed).
*
* <b><i>Selectors</i></b><br>
* A Selector can perform complex data selections, such as per-instance selection,
* content and status filtering, etc when reading or taking. Such a selector is
* returned by this operation. Setting filters on the Selector can be concatenated,
* resulting in the following example code.
* @code{.cpp}
* dds::domain::DomainParticipant participant(org::eclipse::cyclonedds::domain::default_id());
* dds::topic::Topic<Foo::Bar> topic(participant, "TopicName");
* dds::sub::Subscriber subscriber(participant);
* dds::sub::DataReader<Foo::Bar> reader(subscriber, topic);
*
* // Take a maximum of 3 new samples of a certain instance.
* {
* dds::sub::LoanedSamples<Foo::Bar> samples;
* samples = reader.select()
* .max_samples(3)
* .state(dds::sub::status::DataState::new_data())
* .instance(someValidInstanceHandle)
* .take();
* }
* @endcode
*
* <i>Please be aware that using pre-set filters on the DataReader and then
* call read() or take() on that reader explicitly will perform better than
* having to create Selectors for every read (which is what essentially
* happens in the code example above). Performance can be increase by creating
* a selector up front and using that multiple times
* (see dds::sub::DataReader::Selector).</i>
*
* @return A Selector that acts on the DataReader
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
*/
Selector select();
//========================================================================
//== Instance Management
public:
/**
* This operation retrieves the key value of a specific instance.
*
* This operation can be used to retrieve the instance key that corresponds
* to an instance_handle. The operation will only fill the fields that form
* the key inside the sample instance.
*
* This operation may raise a InvalidArgumentError exception if the InstanceHandle
* does not correspond to an existing data-object known to the DataReader.
* If the implementation is not able to check invalid handles, then the
* result in this situation is unspecified.
*
* @param h The instance handle
* @return A topic instance with the handle and key fields set
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::InvalidArgumentError
* The InstanceHandle is not a valid handle.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
* @throws dds::core::PreconditionNotMetError
* The handle has not been registered with this DataReader.
*/
dds::topic::TopicInstance<T> key_value(const dds::core::InstanceHandle& h);
/**
* This operation retrieves the key value of a specific instance.
*
* This operation can be used to retrieve the instance key that corresponds
* to an instance_handle. The operation will only fill the fields that form
* the key inside the sample instance.
*
* This operation may raise a InvalidArgumentError exception if the InstanceHandle
* does not correspond to an existing data-object known to the DataReader.
* If the implementation is not able to check invalid handles, then the
* result in this situation is unspecified.
*
* The Sample is added as parameter to be able to overload this operation.
*
* @param[out] sample A sample to set the key fields of
* @param[in] h The instance handle
* @return The given sample with the key fields set
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::InvalidArgumentError
* The InstanceHandle is not a valid handle.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
* @throws dds::core::NotEnabledError
* The DataReader has not yet been enabled.
* @throws dds::core::PreconditionNotMetError
* The handle has not been registered with this DataReader.
*/
T& key_value(T& sample, const dds::core::InstanceHandle& h);
/**
* This operation returns the value of the instance handle which corresponds
* to the instance_data.
*
* The instance handle can be used in read operations that operate
* on a specific instance. Note that DataReader instance handles are local, and are
* not interchangeable with DataWriter instance handles nor with instance handles
* of an other DataReader.
*
* This operation does not register the instance in question. If the instance has not been
* previously registered or if for any other
* reason the Service is unable to provide an instance handle, the Service will return
* the default nil handle (InstanceHandle.is_nil() == true).
*
* @param key the sample
* @return the instance handle
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
*/
const dds::core::InstanceHandle
lookup_instance(const T& key) const;
public:
/**
* Register a listener with the DataReader.
*
* This operation attaches a DataReaderListener to the DataReader. Only one
* DataReaderListener can be attached to each DataReader. If a
* DataReaderListener was already attached, the operation will replace it with the
* new one. When the listener is the NULL pointer, it represents a listener that is
* treated as a NOOP for all statuses activated in the bit mask.
*
* Listener un-registration is performed by setting the listener to NULL and mask none().
*
* @anchor anchor_dds_sub_datareader_commstatus
* <i>Communication Status</i><br>
* For each communication status, the StatusChangedFlag flag is initially set to
* FALSE. It becomes TRUE whenever that communication status changes. For each
* communication status activated in the mask, the associated DataReaderListener
* operation is invoked and the communication status is reset to FALSE, as the listener
* implicitly accesses the status which is passed as a parameter to that operation. The
* status is reset prior to calling the listener, so if the application calls the
* get_<status_name>_status from inside the listener it will see the status
* already reset. An exception to this rule is the NULL listener, which does not reset the
* communication statuses for which it is invoked.
*
* The following statuses are applicable to the DataReaderListener:
* - dds::core::status::StatusMask::requested_deadline_missed()
* - dds::core::status::StatusMask::requested_incompatible_qos()
* - dds::core::status::StatusMask::sample_lost()
* - dds::core::status::StatusMask::sample_rejected()
* - dds::core::status::StatusMask::data_available()
* - dds::core::status::StatusMask::liveliness_changed()
* - dds::core::status::StatusMask::subscription_matched()
*
* Be aware that the SUBSCRIPTION_MATCHED_STATUS is not applicable when the
* infrastructure does not have the information available to determine connectivity.
* This is the case when OpenSplice is configured not to maintain discovery
* information in the Networking Service. (See the description for the
* NetworkingService/Discovery/enabled property in the Deployment
* Manual for more information about this subject.) In this case the operation will
* throw UnsupportedError.
*
* Status bits are declared as a constant and can be used by the application in an OR
* operation to create a tailored mask. The special constant dds::core::status::StatusMask::none()
* can be used to indicate that the created entity should not respond to any of its available
* statuses. The DDS will therefore attempt to propagate these statuses to its factory.
* The special constant dds::core::status::StatusMask::all() can be used to select all applicable
* statuses specified in the "Data Distribution Service for Real-time Systems Version
* 1.2" specification which are applicable to the PublisherListener.
*
* @anchor anchor_dds_sub_datareader_commpropagation
* <i>Status Propagation</i><br>
* In case a communication status is not activated in the mask of the
* DataReaderListener, the SubscriberListener of the containing Subscriber
* is invoked (if attached and activated for the status that occurred). This allows the
* application to set a default behaviour in the SubscriberListener of the containing
* Subscriber and a DataReader specific behaviour when needed. In case the
* communication status is not activated in the mask of the SubscriberListener as
* well, the communication status will be propagated to the
* DomainParticipantListener of the containing DomainParticipant. In case
* the DomainParticipantListener is also not attached or the communication
* status is not activated in its mask, the application is not notified of the change.
*
* The statuses DATA_ON_READERS_STATUS and DATA_AVAILABLE_STATUS are
* "Read Communication Statuses" and are an exception to all other plain
* communication statuses: they have no corresponding status structure that can be
* obtained with a get_<status_name>_status operation and they are mutually
* exclusive. When new information becomes available to a DataReader, the Data
* Distribution Service will first look in an attached and activated
* SubscriberListener or DomainParticipantListener (in that order) for the
* DATA_ON_READERS_STATUS. In case the DATA_ON_READERS_STATUS can not be
* handled, the Data Distribution Service will look in an attached and activated
* DataReaderListener, SubscriberListener or DomainParticipant
* Listener for the DATA_AVAILABLE_STATUS (in that order).
*
* See also @ref DCPS_Modules_Infrastructure_Listener "listener information".
*
* @param listener the listener
* @param event_mask the mask defining the events for which the listener
* will be notified.
* @throws dds::core::Error
* An internal error has occurred.
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
* @throws dds::core::AlreadyClosedError
* The entity has already been closed.
* @throws dds::core::UnsupportedError
* A status was selected that cannot be supported because
* the infrastructure does not maintain the required connectivity information.
* @throws dds::core::OutOfResourcesError
* The Data Distribution Service ran out of resources to
* complete this operation.
*/
void listener(Listener* listener,
const dds::core::status::StatusMask& event_mask);
/**
* Get the listener of this DataReader.
*
* See also @ref DCPS_Modules_Infrastructure_Listener "listener information".
*
* @return the listener
* @throws dds::core::NullReferenceError
* The entity was not properly created and references to dds::core::null.
*/
Listener* listener() const;
};
#endif /* OMG_DDS_SUB_TDATA_READER_HPP_ */