31#ifndef DW_FRAMEWORK_MANAGEDPORT_HPP_
32#define DW_FRAMEWORK_MANAGEDPORT_HPP_
39#include <dwshared/dwfoundation/dw/core/language/Optional.hpp>
40#include <dwshared/dwfoundation/dw/core/container/RingBuffer.hpp>
41#include <dwshared/dwfoundation/dw/core/container/StringView.hpp>
42#include <dwshared/dwfoundation/dw/core/language/Function.hpp>
48#include <fiu/FaultRegistry.hpp>
90 virtual bool isBound() const noexcept = 0;
99 virtual
void setNodeName(const
dw::core::StringView& nodeName) noexcept;
143 static constexpr char LOG_TAG[]{
"ManagedPortOutputBase"};
155 bool syncEnabled =
false;
190 void send(const dwTime_t* publishTimestamp =
nullptr);
197 void setCallbackBeforeSend(
dw::core::Function<dwStatus()> callback);
218 void sendAdvTimestamp();
224 return m_channelProducer;
233 void preSend(
ChannelMetadata& header,
const dwTime_t* publishTimestamp);
234 void checkFiInstances(
bool& sendMessage);
235 void checkFiInstanceInvalidateMessageHeaders();
236 void checkFiInstanceZeroSequenceNumber();
237 void checkFiInstanceDropMessages(
bool& sendMessage);
238 dwStatus maybeSendMessage(
bool sendMessage);
242 dw::core::Optional<GenericData> m_buffer;
243 dw::core::Function<dwStatus()> m_callbackBeforeSend;
244 uint32_t m_sendSeqNum;
245 std::type_index m_typeIndex;
246 bool m_isNvsci{
false};
262 uint32_t maxBuffers = 1U;
268 dwTime_t waitTime = 0;
282 bool enableReuse =
false;
287 bool syncEnabled =
false;
292 uint32_t dataOffset = 0U;
339 bool isBufferAvailable() const noexcept;
354 void setCallbackAfterRecv(
dw::core::Function<dwStatus()> callback);
365 void sendAdvTimestamp();
371 void releaseToChannel(
void* data);
375 bool recvSingle(dwTime_t waitTime);
376 bool stashConsumed();
378 void handleReuseDrop();
380 void releaseSingle();
381 bool postProcessLockstepReplayData(
GenericData packet);
382 dwTime_t getWaitTime();
385 void handleWaitFailure(dwStatus status);
386 void handleCallbackAfterRecv();
391 bool m_shouldDropFirstBuffer;
392 dw::core::Function<dwStatus()> m_callbackAfterRecv;
406 static_assert(std::is_constructible<T>::value,
"T must be constructible");
408 "Channel packet type not declared. Ensure channel packet type "
409 "handling is declared with DWFRAMEWORK_DECLARE_PACKET_TYPE_POD "
410 "or DWFRAMEWORK_DECLARE_PACKET_TYPE_RELATION");
462 return detail::vectorIterable<T>(getAllBuffers());
472 return detail::getBufferTyped<T>(getBufferGeneric());
482 static_assert(std::is_constructible<T>::value,
"T must be constructible");
484 "Channel packet type not declared. Ensure channel packet type "
485 "handling is declared with DWFRAMEWORK_DECLARE_PACKET_TYPE_POD "
486 "or DWFRAMEWORK_DECLARE_PACKET_TYPE_RELATION");
490 struct PacketDeleter;
524 return detail::vectorIterable<T>(getAllBuffers());
544 return detail::getBufferTyped<T>(getBufferGeneric());
554 if (isBufferAvailable())
556 const dw::core::Optional<dwValidityStatus> valid{
getValidityStatus(getMetadata())};
559 const bool isValid{!valid.has_value() || (DW_VALIDITY_VALID == valid->validity)};
576 return isBufferAvailable() ? getBuffer() :
nullptr;
584 T* ptr{detail::getBufferTyped<T>(packet)};
598 class iterator :
public ManagedPortInputBase::RingBuffer::iterator
600 static_assert(std::is_constructible<TT>::value,
"TT must be constructible");
602 using Base = ManagedPortInputBase::RingBuffer::iterator;
607 : Base(std::move(base))
620 return detail::getBufferTyped<TT>(buffer);
642 void operator()(T* p)
644 static_cast<void>(p);
647 ManagedPortInput* port;
661 uint64_t DescriptorIndex>
662struct IsOutputNonPOD : std::integral_constant<bool, Direction == PortDirection::OUTPUT && parameter_traits<decltype(portDescriptorType<NodeT, Direction, DescriptorIndex>())>::PacketTID != DWFRAMEWORK_PACKET_ID_DEFAULT>
665 static_assert(DescriptorIndex < portDescriptorSize<NodeT, Direction>(),
"Invalid PortIndex.");
671 uint64_t DescriptorIndex>
672typename std::enable_if<
673 !IsOutputNonPOD<NodeT, Direction, DescriptorIndex>::value,
676 createPortSpecimenByStaticIndex()
678 GenericDataReference ref{make_specimen<decltype(portDescriptorType<NodeT, Direction, DescriptorIndex>())>(
nullptr)};
686 uint64_t DescriptorIndex>
687typename std::enable_if<
688 IsOutputNonPOD<NodeT, Direction, DescriptorIndex>::value,
691 createPortSpecimenByStaticIndex()
693 throw ExceptionWithStatus(DW_NOT_SUPPORTED,
"createPortSpecimenByStaticIndex: Non POD output port is not supported");
702 createPortSpecimenImpl(
size_t descriptorIndex, std::index_sequence<Idx...>)
704 constexpr size_t ArraySize{
sizeof...(Idx)};
708 std::array<GenericDataReference, ArraySize> specimens{
712 throw ExceptionWithStatus(DW_OUT_OF_BOUNDS,
"createPortSpecimenImpl: index out of bound.");
724 createPortSpecimenImpl(
size_t descriptorIndex, std::index_sequence<Idx...>)
726 constexpr size_t ArraySize{
sizeof...(Idx)};
729 return createPortSpecimenByStaticIndex<NodeT, Direction, 0>();
731 throw ExceptionWithStatus(DW_OUT_OF_BOUNDS,
"createPortSpecimenImpl: index out of bound.");
740 createPortSpecimenImpl(
size_t, std::index_sequence<Idx...>)
742 throw ExceptionWithStatus(DW_OUT_OF_BOUNDS,
"createPortSpecimenImpl: index out of bound.");
751 return detail::createPortSpecimenImpl<NodeT, Direction>(
753 std::make_index_sequence<portDescriptorSize<NodeT, Direction>()>{});
void * getPointer() const
dw::core::StringView m_name
The unique name within set of ports with the same direction.
ManagedPortBase & operator=(ManagedPortBase &&other)=delete
virtual bool isBound() const noexcept=0
FI_DECLARE_INSTANCE_SET_HANDLE(m_fiHandle)
virtual ~ManagedPortBase()=default
const dw::core::StringView & getNodeName() const
Get the name of the node this port belongs to.
virtual void setNodeName(const dw::core::StringView &nodeName) noexcept
Set the name of the node this port belongs to.
ManagedPortBase(const ManagedPortBase &other)=delete
virtual void bindChannel(ChannelObject *channel)=0
ChannelObject * getChannel()
ChannelObject * m_channel
ManagedPortBase(ManagedPortBase &&other)=delete
ManagedPortBase & operator=(const ManagedPortBase &other)=delete
const dw::core::StringView & getName() const
Get the name of the port.
void bindLockstepSyncClient(dw::framework::lockstep::ILockstepSyncClient *syncClient)
dw::framework::lockstep::ILockstepSyncClient * m_lockstepSyncClient
ManagedPortBase(const dw::core::StringView &name)
dw::core::StringView m_nodeName
The name of the node this port belongs to.
void setCycleCount(uint32_t cycleCount)
void setPeriod(uint32_t period)
GenericData getBufferGeneric()
ManagedPortOutputBase(const dw::core::StringView &name, ConstructProperties props, GenericDataReference &&ref, std::type_index typeIndex)
ConstructProperties constructProperties
BoundProperties boundProperties
bool isBufferAvailable() const noexcept
ManagedPortOutput(const dw::core::StringView &name, SpecimenT &ref)
ManagedPortOutput(const dw::core::StringView &name, ConstructProperties props)
ManagedPortOutput(const dw::core::StringView &name)
ManagedPortOutput(const dw::core::StringView &name, ConstructProperties props, SpecimenT &ref)
detail::vectorIterable< T > getAllBufferIter()
static constexpr const uint32_t DWFRAMEWORK_METADATA_PACKET_TYPE_ID_OFFSET
static GenericDataReference make_specimen(typename parameter_traits< T >::SpecimenT *specimen)
ChannelPacketTypeID packetTypeID
The ID of the type of the endpoint.
dw::core::Optional< dwValidityStatus > getValidityStatus(ChannelMetadata const &header)
uint32_t ChannelPacketTypeID
constexpr ChannelPacketTypeID DWFRAMEWORK_PACKET_ID_DEFAULT
typename ManagedPortInput< T >::UniquePacketPtr UniquePacketPtr
constexpr size_t descriptorIndex()