Skip to content

iceoryx_posh/popo/listener.hpp🔗

Namespaces🔗

Name
iox
iox::popo
iox::popo::internal

Classes🔗

Name
class iox::popo::ListenerImpl
The Listener is a class which reacts to registered events by executing a corresponding callback concurrently. This is achieved via an encapsulated thread inside this class.
class iox::popo::Listener

Source code🔗

// Copyright (c) 2021 by Apex.AI 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.
//
// SPDX-License-Identifier: Apache-2.0

#ifndef IOX_POSH_POPO_LISTENER_HPP
#define IOX_POSH_POPO_LISTENER_HPP

#include "iceoryx_hoofs/cxx/expected.hpp"
#include "iceoryx_hoofs/cxx/method_callback.hpp"
#include "iceoryx_hoofs/cxx/type_traits.hpp"
#include "iceoryx_hoofs/internal/concurrent/loffli.hpp"
#include "iceoryx_hoofs/internal/concurrent/smart_lock.hpp"
#include "iceoryx_posh/internal/popo/building_blocks/condition_listener.hpp"
#include "iceoryx_posh/popo/enum_trigger_type.hpp"
#include "iceoryx_posh/popo/notification_attorney.hpp"
#include "iceoryx_posh/popo/notification_callback.hpp"
#include "iceoryx_posh/popo/trigger_handle.hpp"
#include "iceoryx_posh/runtime/posh_runtime.hpp"

#include <thread>

namespace iox
{
namespace popo
{
namespace internal
{
class Event_t
{
  public:
    ~Event_t() noexcept;

    bool isEqualTo(const void* const origin, const uint64_t eventType, const uint64_t eventTypeHash) const noexcept;
    bool reset() noexcept;
    bool init(const uint64_t eventId,
              void* const origin,
              void* const userType,
              const uint64_t eventType,
              const uint64_t eventTypeHash,
              internal::GenericCallbackRef_t callback,
              internal::TranslationCallbackRef_t translationCallback,
              const cxx::MethodCallback<void, uint64_t> invalidationCallback) noexcept;
    void executeCallback() noexcept;
    bool isInitialized() const noexcept;

  private:
    static constexpr uint64_t INVALID_ID = std::numeric_limits<uint64_t>::max();

    void* m_origin = nullptr;
    uint64_t m_eventType = INVALID_ID;
    uint64_t m_eventTypeHash = INVALID_ID;

    internal::GenericCallbackPtr_t m_callback = nullptr;
    internal::TranslationCallbackPtr_t m_translationCallback = nullptr;
    void* m_userType = nullptr;

    uint64_t m_eventId = INVALID_ID;
    cxx::MethodCallback<void, uint64_t> m_invalidationCallback;
};
} // namespace internal

enum class ListenerError
{
    LISTENER_FULL,
    EVENT_ALREADY_ATTACHED,
    EMPTY_EVENT_CALLBACK,
    EMPTY_INVALIDATION_CALLBACK
};

template <uint64_t Capacity>
class ListenerImpl
{
  public:
    ListenerImpl() noexcept;
    ListenerImpl(const ListenerImpl&) = delete;
    ListenerImpl(ListenerImpl&&) = delete;
    ~ListenerImpl() noexcept;

    ListenerImpl& operator=(const ListenerImpl&) = delete;
    ListenerImpl& operator=(ListenerImpl&&) = delete;

    template <typename T,
              typename EventType,
              typename ContextDataType,
              typename = std::enable_if_t<std::is_enum<EventType>::value>>
    cxx::expected<ListenerError> attachEvent(T& eventOrigin,
                                             const EventType eventType,
                                             const NotificationCallback<T, ContextDataType>& eventCallback) noexcept;

    template <typename T, typename ContextDataType>
    cxx::expected<ListenerError> attachEvent(T& eventOrigin,
                                             const NotificationCallback<T, ContextDataType>& eventCallback) noexcept;

    template <typename T, typename EventType, typename = std::enable_if_t<std::is_enum<EventType>::value>>
    void detachEvent(T& eventOrigin, const EventType eventType) noexcept;

    template <typename T>
    void detachEvent(T& eventOrigin) noexcept;

    static constexpr uint64_t capacity() noexcept;

    uint64_t size() const noexcept;

  protected:
    ListenerImpl(ConditionVariableData& conditionVariableData) noexcept;

  private:
    class Event_t;

    void threadLoop() noexcept;
    cxx::expected<uint32_t, ListenerError>
    addEvent(void* const origin,
             void* const userType,
             const uint64_t eventType,
             const uint64_t eventTypeHash,
             internal::GenericCallbackRef_t callback,
             internal::TranslationCallbackRef_t translationCallback,
             const cxx::MethodCallback<void, uint64_t> invalidationCallback) noexcept;

    void removeTrigger(const uint64_t index) noexcept;

  private:
    enum class NoEnumUsed : EventEnumIdentifier
    {
        PLACEHOLDER = 0
    };


    class IndexManager_t
    {
      public:
        IndexManager_t() noexcept;
        bool pop(uint32_t& index) noexcept;
        void push(const uint32_t index) noexcept;
        uint64_t indicesInUse() const noexcept;

        using LoFFLi = concurrent::LoFFLi;
        LoFFLi::Index_t m_loffliStorage[LoFFLi::requiredIndexMemorySize(Capacity) / sizeof(uint32_t)];
        LoFFLi m_loffli;
        std::atomic<uint64_t> m_indicesInUse{0U};
    } m_indexManager;


    std::thread m_thread;
    concurrent::smart_lock<internal::Event_t, std::recursive_mutex> m_events[Capacity];
    std::mutex m_addEventMutex;

    std::atomic_bool m_wasDtorCalled{false};
    ConditionVariableData* m_conditionVariableData = nullptr;
    ConditionListener m_conditionListener;
};

class Listener : public ListenerImpl<MAX_NUMBER_OF_EVENTS_PER_LISTENER>
{
  public:
    using Parent = ListenerImpl<MAX_NUMBER_OF_EVENTS_PER_LISTENER>;
    Listener() noexcept;

  protected:
    Listener(ConditionVariableData& conditionVariableData) noexcept;
};

} // namespace popo
} // namespace iox

#include "iceoryx_posh/internal/popo/listener.inl"

#endif

Updated on 2 April 2022 at 16:37:47 CEST