cpp-common/bt2: add `plugin-dev.hpp`
[babeltrace.git] / src / cpp-common / bt2 / plugin-dev.hpp
diff --git a/src/cpp-common/bt2/plugin-dev.hpp b/src/cpp-common/bt2/plugin-dev.hpp
new file mode 100644 (file)
index 0000000..220d106
--- /dev/null
@@ -0,0 +1,1157 @@
+/*
+ * Copyright (c) 2023 Simon Marchi <simon.marchi@efficios.com>
+ * Copyright (c) 2023 Philippe Proulx <pproulx@efficios.com>
+ *
+ * SPDX-License-Identifier: MIT
+ */
+
+#ifndef BABELTRACE_CPP_COMMON_BT2_PLUGIN_DEV_HPP
+#define BABELTRACE_CPP_COMMON_BT2_PLUGIN_DEV_HPP
+
+#include <cstdint>
+
+#include <glib.h>
+
+#include <babeltrace2/babeltrace.h>
+
+#include "cpp-common/bt2c/logging.hpp"
+#include "cpp-common/vendor/fmt/core.h"
+
+#include "exc.hpp"
+#include "wrap.hpp"
+
+namespace bt2 {
+namespace internal {
+
+constexpr const char *unhandledExcLogStr() noexcept
+{
+    return "Unhandled exception.";
+}
+
+constexpr const char *unhandledExcLogTag() noexcept
+{
+    return "PLUGIN-DEV-HPP";
+}
+
+/*
+ * Base class of any component class bridge.
+ *
+ * `UserCompClsT` is the actual C++ user component class and `LibTypesT`
+ * is a structure offering the following specific library types:
+ *
+ * `SelfCompCls`:
+ *     Self component class.
+ *
+ * `SelfComp`:
+ *     Self component.
+ *
+ * `SelfCompCfg`:
+ *     Self component configuration.
+ */
+template <typename UserCompClsT, typename LibTypesT>
+class CompClsBridge
+{
+private:
+    using _LibSelfCompPtr = typename LibTypesT::SelfComp *;
+
+public:
+    static UserCompClsT& userCompFromLibSelfCompPtr(const _LibSelfCompPtr libSelfCompPtr) noexcept
+    {
+        return wrap(libSelfCompPtr).template data<UserCompClsT>();
+    }
+
+    static bt_component_class_initialize_method_status init(const _LibSelfCompPtr libSelfCompPtr,
+                                                            typename LibTypesT::SelfCompCfg *,
+                                                            const bt_value * const libParamsPtr,
+                                                            void *) noexcept
+    {
+        const auto selfComp = wrap(libSelfCompPtr);
+
+        try {
+            const auto comp = new UserCompClsT {selfComp, wrap(libParamsPtr).asMap()};
+
+            selfComp.data(*comp);
+        } catch (const std::bad_alloc&) {
+            return BT_COMPONENT_CLASS_INITIALIZE_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const Error&) {
+            return BT_COMPONENT_CLASS_INITIALIZE_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(BT_LOG_WARNING, static_cast<int>(selfComp.loggingLevel()),
+                                 unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_COMPONENT_CLASS_INITIALIZE_METHOD_STATUS_ERROR;
+        }
+
+        return BT_COMPONENT_CLASS_INITIALIZE_METHOD_STATUS_OK;
+    }
+
+    static void finalize(const _LibSelfCompPtr libSelfCompPtr) noexcept
+    {
+        delete &userCompFromLibSelfCompPtr(libSelfCompPtr);
+    }
+
+    static bt_component_class_get_supported_mip_versions_method_status
+    getSupportedMipVersions(typename LibTypesT::SelfCompCls * const libSelfCompClsPtr,
+                            const bt_value * const libParamsPtr, void *,
+                            const bt_logging_level logLevel,
+                            bt_integer_range_set_unsigned * const libSupportedVersionsPtr) noexcept
+    {
+        try {
+            UserCompClsT::getSupportedMipVersions(wrap(libSelfCompClsPtr), wrap(libParamsPtr),
+                                                  static_cast<LoggingLevel>(logLevel),
+                                                  wrap(libSupportedVersionsPtr));
+            return BT_COMPONENT_CLASS_GET_SUPPORTED_MIP_VERSIONS_METHOD_STATUS_OK;
+        } catch (const std::bad_alloc&) {
+            return BT_COMPONENT_CLASS_GET_SUPPORTED_MIP_VERSIONS_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const Error&) {
+            return BT_COMPONENT_CLASS_GET_SUPPORTED_MIP_VERSIONS_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(BT_LOG_WARNING, static_cast<int>(logLevel), unhandledExcLogTag(),
+                                 unhandledExcLogStr());
+            return BT_COMPONENT_CLASS_GET_SUPPORTED_MIP_VERSIONS_METHOD_STATUS_ERROR;
+        }
+    }
+
+    static bt_component_class_query_method_status
+    query(typename LibTypesT::SelfCompCls * const libSelfCompClsPtr,
+          bt_private_query_executor * const libPrivQueryExecPtr, const char * const object,
+          const bt_value * const libParamsPtr, void *,
+          const bt_value ** const libResultPtr) noexcept
+    {
+        const auto privQueryExec = wrap(libPrivQueryExecPtr);
+
+        try {
+            auto result = UserCompClsT::query(wrap(libSelfCompClsPtr), privQueryExec, object,
+                                              wrap(libParamsPtr));
+
+            *libResultPtr = result.release().libObjPtr();
+            return BT_COMPONENT_CLASS_QUERY_METHOD_STATUS_OK;
+        } catch (const TryAgain&) {
+            return BT_COMPONENT_CLASS_QUERY_METHOD_STATUS_AGAIN;
+        } catch (const UnknownObject&) {
+            return BT_COMPONENT_CLASS_QUERY_METHOD_STATUS_UNKNOWN_OBJECT;
+        } catch (const std::bad_alloc&) {
+            return BT_COMPONENT_CLASS_QUERY_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const Error&) {
+            return BT_COMPONENT_CLASS_QUERY_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(BT_LOG_WARNING, static_cast<int>(privQueryExec.loggingLevel()),
+                                 unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_COMPONENT_CLASS_QUERY_METHOD_STATUS_ERROR;
+        }
+    }
+};
+
+template <typename SpecCompClsBridgeT, typename LibTypesT>
+struct CompClsBridgeWithInputPorts
+{
+    static bt_component_class_port_connected_method_status
+    inputPortConnected(typename LibTypesT::SelfComp * const libSelfCompPtr,
+                       bt_self_component_port_input * const libSelfCompPortPtr,
+                       const bt_port_output * const libOtherPortPtr) noexcept
+    {
+        try {
+            SpecCompClsBridgeT::userCompFromLibSelfCompPtr(libSelfCompPtr)
+                .inputPortConnected(wrap(libSelfCompPortPtr), wrap(libOtherPortPtr));
+        } catch (const std::bad_alloc&) {
+            return BT_COMPONENT_CLASS_PORT_CONNECTED_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const Error&) {
+            return BT_COMPONENT_CLASS_PORT_CONNECTED_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(BT_LOG_WARNING,
+                                 static_cast<int>(wrap(libSelfCompPtr).loggingLevel()),
+                                 unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_COMPONENT_CLASS_PORT_CONNECTED_METHOD_STATUS_ERROR;
+        }
+
+        return BT_COMPONENT_CLASS_PORT_CONNECTED_METHOD_STATUS_OK;
+    }
+};
+
+template <typename SpecCompClsBridgeT, typename LibTypesT>
+struct CompClsBridgeWithOutputPorts
+{
+    static bt_component_class_port_connected_method_status
+    outputPortConnected(typename LibTypesT::SelfComp * const libSelfCompPtr,
+                        bt_self_component_port_output * const libSelfCompPortPtr,
+                        const bt_port_input * const libOtherPortPtr) noexcept
+    {
+        try {
+            SpecCompClsBridgeT::userCompFromLibSelfCompPtr(libSelfCompPtr)
+                .outputPortConnected(wrap(libSelfCompPortPtr), wrap(libOtherPortPtr));
+        } catch (const std::bad_alloc&) {
+            return BT_COMPONENT_CLASS_PORT_CONNECTED_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const Error&) {
+            return BT_COMPONENT_CLASS_PORT_CONNECTED_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(BT_LOG_WARNING,
+                                 static_cast<int>(wrap(libSelfCompPtr).loggingLevel()),
+                                 unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_COMPONENT_CLASS_PORT_CONNECTED_METHOD_STATUS_ERROR;
+        }
+
+        return BT_COMPONENT_CLASS_PORT_CONNECTED_METHOD_STATUS_OK;
+    }
+};
+
+struct SrcCompClsLibTypes final
+{
+    using SelfCompCls = bt_self_component_class_source;
+    using SelfComp = bt_self_component_source;
+    using SelfCompCfg = bt_self_component_source_configuration;
+};
+
+template <typename UserCompClsT>
+class SrcCompClsBridge final :
+    public CompClsBridge<UserCompClsT, SrcCompClsLibTypes>,
+    public CompClsBridgeWithOutputPorts<SrcCompClsBridge<UserCompClsT>, SrcCompClsLibTypes>
+{
+};
+
+struct FltCompClsLibTypes final
+{
+    using SelfCompCls = bt_self_component_class_filter;
+    using SelfComp = bt_self_component_filter;
+    using SelfCompCfg = bt_self_component_filter_configuration;
+};
+
+template <typename UserCompClsT>
+class FltCompClsBridge final :
+    public CompClsBridge<UserCompClsT, FltCompClsLibTypes>,
+    public CompClsBridgeWithInputPorts<FltCompClsBridge<UserCompClsT>, FltCompClsLibTypes>,
+    public CompClsBridgeWithOutputPorts<FltCompClsBridge<UserCompClsT>, FltCompClsLibTypes>
+{
+};
+
+struct SinkCompClsLibTypes final
+{
+    using SelfCompCls = bt_self_component_class_sink;
+    using SelfComp = bt_self_component_sink;
+    using SelfCompCfg = bt_self_component_sink_configuration;
+};
+
+template <typename UserCompClsT>
+class SinkCompClsBridge final :
+    CompClsBridge<UserCompClsT, SinkCompClsLibTypes>,
+    CompClsBridgeWithInputPorts<SinkCompClsBridge<UserCompClsT>, SinkCompClsLibTypes>
+{
+private:
+    using CompClsBridge<UserCompClsT, SinkCompClsLibTypes>::userCompFromLibSelfCompPtr;
+
+public:
+    static bt_component_class_sink_consume_method_status
+    consume(bt_self_component_sink * const libSelfCompPtr) noexcept
+    {
+        try {
+            if (userCompFromLibSelfCompPtr(libSelfCompPtr).consume()) {
+                return BT_COMPONENT_CLASS_SINK_CONSUME_METHOD_STATUS_OK;
+            } else {
+                return BT_COMPONENT_CLASS_SINK_CONSUME_METHOD_STATUS_END;
+            }
+        } catch (const TryAgain&) {
+            return BT_COMPONENT_CLASS_SINK_CONSUME_METHOD_STATUS_AGAIN;
+        } catch (const std::bad_alloc&) {
+            return BT_COMPONENT_CLASS_SINK_CONSUME_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const Error&) {
+            return BT_COMPONENT_CLASS_SINK_CONSUME_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(BT_LOG_WARNING,
+                                 static_cast<int>(wrap(libSelfCompPtr).loggingLevel()),
+                                 unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_COMPONENT_CLASS_SINK_CONSUME_METHOD_STATUS_ERROR;
+        }
+    }
+
+    static bt_component_class_sink_graph_is_configured_method_status
+    graphIsConfigured(bt_self_component_sink * const libSelfCompPtr) noexcept
+    {
+        try {
+            userCompFromLibSelfCompPtr(libSelfCompPtr).graphIsConfigured();
+        } catch (const std::bad_alloc&) {
+            return BT_COMPONENT_CLASS_SINK_GRAPH_IS_CONFIGURED_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const Error&) {
+            return BT_COMPONENT_CLASS_SINK_GRAPH_IS_CONFIGURED_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(BT_LOG_WARNING,
+                                 static_cast<int>(wrap(libSelfCompPtr).loggingLevel()),
+                                 unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_COMPONENT_CLASS_SINK_GRAPH_IS_CONFIGURED_METHOD_STATUS_ERROR;
+        }
+    }
+};
+
+template <typename UserMsgIterT>
+class MsgIterClsBridge final
+{
+public:
+    static UserMsgIterT&
+    userMsgIterFromLibSelfMsgIterPtr(bt_self_message_iterator * const libSelfMsgIterPtr) noexcept
+    {
+        return bt2::wrap(libSelfMsgIterPtr).data<UserMsgIterT>();
+    }
+
+    static bt_message_iterator_class_initialize_method_status
+    init(bt_self_message_iterator * const libSelfMsgIterPtr,
+         bt_self_message_iterator_configuration * const libSelfMsgIterConfigPtr,
+         bt_self_component_port_output * const libSelfCompPortPtr) noexcept
+    {
+        const auto selfMsgIter = bt2::wrap(libSelfMsgIterPtr);
+
+        try {
+            const auto msgIter = new UserMsgIterT {selfMsgIter, bt2::wrap(libSelfMsgIterConfigPtr),
+                                                   bt2::wrap(libSelfCompPortPtr)};
+
+            selfMsgIter.data(*msgIter);
+        } catch (const std::bad_alloc&) {
+            return BT_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const bt2::Error&) {
+            return BT_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(
+                BT_LOG_WARNING,
+                static_cast<int>(wrap(libSelfMsgIterPtr).component().loggingLevel()),
+                unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD_STATUS_ERROR;
+        }
+
+        return BT_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD_STATUS_OK;
+    }
+
+    static void finalize(bt_self_message_iterator * const libSelfMsgIterPtr) noexcept
+    {
+        delete &userMsgIterFromLibSelfMsgIterPtr(libSelfMsgIterPtr);
+    }
+
+    static bt_message_iterator_class_next_method_status
+    next(bt_self_message_iterator * const libSelfMsgIterPtr, bt_message_array_const libMsgsPtr,
+         const uint64_t capacity, uint64_t * const count) noexcept
+    {
+        try {
+            auto msgArray = bt2::ConstMessageArray::wrapEmpty(libMsgsPtr, capacity);
+            auto& msgIter = userMsgIterFromLibSelfMsgIterPtr(libSelfMsgIterPtr);
+
+            msgIter.next(msgArray);
+            *count = msgArray.release();
+
+            if (G_LIKELY(*count > 0)) {
+                return BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_OK;
+            } else {
+                return BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_END;
+            }
+        } catch (const bt2::TryAgain&) {
+            return BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_AGAIN;
+        } catch (const std::bad_alloc&) {
+            return BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const bt2::Error&) {
+            return BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(
+                BT_LOG_WARNING,
+                static_cast<int>(wrap(libSelfMsgIterPtr).component().loggingLevel()),
+                unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_MESSAGE_ITERATOR_CLASS_NEXT_METHOD_STATUS_ERROR;
+        }
+    }
+
+    static bt_message_iterator_class_can_seek_beginning_method_status
+    canSeekBeginning(bt_self_message_iterator * const libSelfMsgIterPtr,
+                     bt_bool * const canSeek) noexcept
+    {
+        try {
+            *canSeek = static_cast<bt_bool>(
+                userMsgIterFromLibSelfMsgIterPtr(libSelfMsgIterPtr).canSeekBeginning());
+        } catch (const bt2::TryAgain&) {
+            return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_BEGINNING_METHOD_STATUS_AGAIN;
+        } catch (const std::bad_alloc&) {
+            return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_BEGINNING_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const bt2::Error&) {
+            return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_BEGINNING_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(
+                BT_LOG_WARNING,
+                static_cast<int>(wrap(libSelfMsgIterPtr).component().loggingLevel()),
+                unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_BEGINNING_METHOD_STATUS_ERROR;
+        }
+
+        return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_BEGINNING_METHOD_STATUS_OK;
+    }
+
+    static bt_message_iterator_class_seek_beginning_method_status
+    seekBeginning(bt_self_message_iterator * const libSelfMsgIterPtr) noexcept
+    {
+        try {
+            userMsgIterFromLibSelfMsgIterPtr(libSelfMsgIterPtr).seekBeginning();
+        } catch (const bt2::TryAgain&) {
+            return BT_MESSAGE_ITERATOR_CLASS_SEEK_BEGINNING_METHOD_STATUS_AGAIN;
+        } catch (const std::bad_alloc&) {
+            return BT_MESSAGE_ITERATOR_CLASS_SEEK_BEGINNING_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const bt2::Error&) {
+            return BT_MESSAGE_ITERATOR_CLASS_SEEK_BEGINNING_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(
+                BT_LOG_WARNING,
+                static_cast<int>(wrap(libSelfMsgIterPtr).component().loggingLevel()),
+                unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_MESSAGE_ITERATOR_CLASS_SEEK_BEGINNING_METHOD_STATUS_ERROR;
+        }
+
+        return BT_MESSAGE_ITERATOR_CLASS_SEEK_BEGINNING_METHOD_STATUS_OK;
+    }
+
+    static bt_message_iterator_class_can_seek_ns_from_origin_method_status
+    canSeekNsFromOrigin(bt_self_message_iterator * const libSelfMsgIterPtr,
+                        const std::int64_t nsFromOrigin, bt_bool * const canSeek) noexcept
+    {
+        try {
+            *canSeek = static_cast<bt_bool>(userMsgIterFromLibSelfMsgIterPtr(libSelfMsgIterPtr)
+                                                .canSeekNsFromOrigin(nsFromOrigin));
+        } catch (const bt2::TryAgain&) {
+            return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_AGAIN;
+        } catch (const std::bad_alloc&) {
+            return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const bt2::Error&) {
+            return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(
+                BT_LOG_WARNING,
+                static_cast<int>(wrap(libSelfMsgIterPtr).component().loggingLevel()),
+                unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_ERROR;
+        }
+
+        return BT_MESSAGE_ITERATOR_CLASS_CAN_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_OK;
+    }
+
+    static bt_message_iterator_class_seek_ns_from_origin_method_status
+    seekNsFromOrigin(bt_self_message_iterator * const libSelfMsgIterPtr,
+                     const std::int64_t nsFromOrigin) noexcept
+    {
+        try {
+            userMsgIterFromLibSelfMsgIterPtr(libSelfMsgIterPtr).seekNsFromOrigin(nsFromOrigin);
+        } catch (const bt2::TryAgain&) {
+            return BT_MESSAGE_ITERATOR_CLASS_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_AGAIN;
+        } catch (const std::bad_alloc&) {
+            return BT_MESSAGE_ITERATOR_CLASS_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_MEMORY_ERROR;
+        } catch (const bt2::Error&) {
+            return BT_MESSAGE_ITERATOR_CLASS_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_ERROR;
+        } catch (...) {
+            BT_LOG_WRITE_CUR_LVL(
+                BT_LOG_WARNING,
+                static_cast<int>(wrap(libSelfMsgIterPtr).component().loggingLevel()),
+                unhandledExcLogTag(), unhandledExcLogStr());
+            return BT_MESSAGE_ITERATOR_CLASS_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_ERROR;
+        }
+
+        return BT_MESSAGE_ITERATOR_CLASS_SEEK_NS_FROM_ORIGIN_METHOD_STATUS_OK;
+    }
+};
+
+} /* namespace internal */
+
+template <typename UserMessageIteratorT, typename UserComponentT>
+class UserMessageIterator;
+
+/*
+ * Base class of any user component.
+ *
+ * See the specific `bt2::UserSourceComponent`,
+ * `bt2::UserFilterComponent`, and `bt2::UserSinkComponent`.
+ */
+template <typename SelfCompT>
+class UserComponent
+{
+    /* Give a related message iterator access to this logger */
+    template <typename, typename>
+    friend class UserMessageIterator;
+
+protected:
+    explicit UserComponent(const SelfCompT selfComp, const std::string& logTag) :
+        _mLogger {selfComp, fmt::format("{}/[{}]", logTag, selfComp.name())}, _mSelfComp {selfComp}
+    {
+    }
+
+protected:
+    const char *_name() const noexcept
+    {
+        return _mSelfComp.name();
+    }
+
+    LoggingLevel _loggingLevel() const noexcept
+    {
+        return _mSelfComp.loggingLevel();
+    }
+
+    std::uint64_t _graphMipVersion() const noexcept
+    {
+        return _mSelfComp.graphMipVersion();
+    }
+
+    SelfCompT _selfComp() noexcept
+    {
+        return _mSelfComp;
+    }
+
+    bt2c::Logger _mLogger;
+
+private:
+    SelfCompT _mSelfComp;
+};
+
+/*
+ * Base class of a user source component `UserComponentT` (CRTP).
+ *
+ * `UserComponentT::UserComponentT()` must accept a
+ * `bt2::SelfSourceComponent` parameter, which it needs to forward to
+ * bt2::UserSourceComponent::UserSourceComponent(), and a
+ * `bt2::ConstValue` parameter (initialization parameters).
+ */
+template <typename UserComponentT>
+class UserSourceComponent : public UserComponent<SelfSourceComponent>
+{
+protected:
+    using _OutputPorts = SelfSourceComponent::OutputPorts;
+
+    explicit UserSourceComponent(const SelfSourceComponent selfComp, const std::string& logTag) :
+        UserComponent<SelfSourceComponent> {selfComp, logTag}
+    {
+    }
+
+public:
+    static Value::Shared query(const SelfComponentClass selfCompCls,
+                               const PrivateQueryExecutor privQueryExec, const char * const obj,
+                               const ConstValue params)
+    {
+        return UserComponentT::_query(selfCompCls, privQueryExec, obj, params);
+    }
+
+    static void getSupportedMipVersions(const SelfComponentClass selfCompCls,
+                                        const ConstValue params, const LoggingLevel loggingLevel,
+                                        const UnsignedIntegerRangeSet ranges)
+    {
+        UserComponentT::_getSupportedMipVersions(selfCompCls, params, loggingLevel, ranges);
+    }
+
+    void outputPortConnected(const SelfComponentOutputPort outputPort,
+                             const ConstInputPort inputPort)
+    {
+        static_cast<UserComponentT&>(*this).outputPortConnected(outputPort, inputPort);
+    }
+
+protected:
+    /* Overloadable */
+    static Value::Shared _query(SelfComponentClass, PrivateQueryExecutor, const char *, ConstValue)
+    {
+        throw UnknownObject {};
+    }
+
+    /* Overloadable */
+    static void _getSupportedMipVersions(SelfComponentClass, ConstValue, LoggingLevel,
+                                         const UnsignedIntegerRangeSet ranges)
+    {
+        ranges.addRange(0, 0);
+    }
+
+    /* Overloadable */
+    void _outputPortConnected(SelfComponentOutputPort, ConstInputPort)
+    {
+    }
+
+    template <typename DataT>
+    _OutputPorts::Port _addOutputPort(const char * const name, DataT * const data)
+    {
+        return this->_selfComp().addOutputPort(name, data);
+    }
+
+    _OutputPorts::Port _addOutputPort(const char *name)
+    {
+        return this->_selfComp().addOutputPort(name);
+    }
+
+    template <typename DataT>
+    _OutputPorts::Port _addOutputPort(const std::string& name, DataT * const data)
+    {
+        return this->_selfComp().addOutputPort(name, data);
+    }
+
+    _OutputPorts::Port _addOutputPort(const std::string& name)
+    {
+        return this->_selfComp().addOutputPort(name);
+    }
+
+    _OutputPorts _outputPorts() noexcept
+    {
+        return this->_selfComp().outputPorts();
+    }
+};
+
+/*
+ * Base class of a user filter component `UserComponentT` (CRTP).
+ *
+ * `UserComponentT::UserComponentT()` must accept a
+ * `bt2::SelfFilterComponent` parameter, which it needs to forward to
+ * bt2::UserFilterComponent::UserFilterComponent(), and a
+ * `bt2::ConstValue` parameter (initialization parameters).
+ */
+template <typename UserComponentT>
+class UserFilterComponent : public UserComponent<SelfFilterComponent>
+{
+protected:
+    using _InputPorts = SelfFilterComponent::InputPorts;
+    using _OutputPorts = SelfFilterComponent::OutputPorts;
+
+    explicit UserFilterComponent(const SelfFilterComponent selfComp, const std::string& logTag) :
+        UserComponent<SelfFilterComponent> {selfComp, logTag}
+    {
+    }
+
+public:
+    static Value::Shared query(const SelfComponentClass selfCompCls,
+                               const PrivateQueryExecutor privQueryExec, const char * const obj,
+                               const ConstValue params)
+    {
+        return UserComponentT::_query(selfCompCls, privQueryExec, obj, params);
+    }
+
+    static void getSupportedMipVersions(const SelfComponentClass selfCompCls,
+                                        const ConstValue params, const LoggingLevel loggingLevel,
+                                        const UnsignedIntegerRangeSet ranges)
+    {
+        UserComponentT::_getSupportedMipVersions(selfCompCls, params, loggingLevel, ranges);
+    }
+
+    void inputPortConnected(const SelfComponentInputPort inputPort,
+                            const ConstOutputPort outputPort)
+    {
+        static_cast<UserComponentT&>(*this)._inputPortConnected(inputPort, outputPort);
+    }
+
+    void outputPortConnected(const SelfComponentOutputPort outputPort,
+                             const ConstInputPort inputPort)
+    {
+        static_cast<UserComponentT&>(*this)._outputPortConnected(outputPort, inputPort);
+    }
+
+protected:
+    /* Overloadable */
+    static Value::Shared _query(SelfComponentClass, PrivateQueryExecutor, const char *, ConstValue)
+    {
+        throw UnknownObject {};
+    }
+
+    /* Overloadable */
+    static void _getSupportedMipVersions(SelfComponentClass, ConstValue, LoggingLevel,
+                                         const UnsignedIntegerRangeSet ranges)
+    {
+        ranges.addRange(0, 0);
+    }
+
+    /* Overloadable */
+    void _inputPortConnected(SelfComponentInputPort, ConstOutputPort)
+    {
+    }
+
+    /* Overloadable */
+    void _outputPortConnected(SelfComponentOutputPort, ConstInputPort)
+    {
+    }
+
+    template <typename DataT>
+    _OutputPorts::Port _addInputPort(const char * const name, DataT * const data)
+    {
+        return this->_selfComp().addInputPort(name, data);
+    }
+
+    _InputPorts::Port _addInputPort(const char *name)
+    {
+        return this->_selfComp().addInputPort(name);
+    }
+
+    template <typename DataT>
+    _InputPorts::Port _addInputPort(const std::string& name, DataT * const data)
+    {
+        return this->_selfComp().addInputPort(name, data);
+    }
+
+    _InputPorts::Port _addInputPort(const std::string& name)
+    {
+        return this->_selfComp().addInputPort(name);
+    }
+
+    _InputPorts _inputPorts() noexcept
+    {
+        return this->_selfComp().inputPorts();
+    }
+
+    template <typename DataT>
+    _OutputPorts::Port _addOutputPort(const char * const name, DataT * const data)
+    {
+        return this->_selfComp().addOutputPort(name, data);
+    }
+
+    _OutputPorts::Port _addOutputPort(const char *name)
+    {
+        return this->_selfComp().addOutputPort(name);
+    }
+
+    template <typename DataT>
+    _OutputPorts::Port _addOutputPort(const std::string& name, DataT * const data)
+    {
+        return this->_selfComp().addOutputPort(name, data);
+    }
+
+    _OutputPorts::Port _addOutputPort(const std::string& name)
+    {
+        return this->_selfComp().addOutputPort(name);
+    }
+
+    _OutputPorts _outputPorts() noexcept
+    {
+        return this->_selfComp().outputPorts();
+    }
+};
+
+/*
+ * Base class of a user sink component `UserComponentT` (CRTP).
+ *
+ * `UserComponentT::UserComponentT()` must accept a
+ * `bt2::SelfSinkComponent` parameter, which it needs to forward to
+ * bt2::UserSinkComponent::UserSinkComponent(), and a `bt2::ConstValue`
+ * parameter (initialization parameters).
+ *
+ * `UserComponentT` must implement:
+ *
+ *     bool _consume();
+ *
+ * This method returns `true` if the sink component still needs to
+ * consume, or `false` if it's finished.
+ */
+template <typename UserComponentT>
+class UserSinkComponent : public UserComponent<SelfSinkComponent>
+{
+protected:
+    using _InputPorts = SelfSinkComponent::InputPorts;
+
+    explicit UserSinkComponent(const SelfSinkComponent selfComp, const std::string& logTag) :
+        UserComponent<SelfSinkComponent> {selfComp, logTag}
+    {
+    }
+
+public:
+    static Value::Shared query(const SelfComponentClass selfCompCls,
+                               const PrivateQueryExecutor privQueryExec, const char * const obj,
+                               const ConstValue params)
+    {
+        return UserComponentT::_query(selfCompCls, privQueryExec, obj, params);
+    }
+
+    static void getSupportedMipVersions(const SelfComponentClass selfCompCls,
+                                        const ConstValue params, const LoggingLevel loggingLevel,
+                                        const UnsignedIntegerRangeSet ranges)
+    {
+        UserComponentT::_getSupportedMipVersions(selfCompCls, params, loggingLevel, ranges);
+    }
+
+    void graphIsConfigured()
+    {
+        static_cast<UserComponentT&>(*this)._graphIsConfigured();
+    }
+
+    void inputPortConnected(const SelfComponentInputPort inputPort,
+                            const ConstOutputPort outputPort)
+    {
+        static_cast<UserComponentT&>(*this)._inputPortConnected(inputPort, outputPort);
+    }
+
+    bool consume()
+    {
+        return static_cast<UserComponentT&>(*this)._consume();
+    }
+
+protected:
+    /* Overloadable */
+    static Value::Shared _query(SelfComponentClass, PrivateQueryExecutor, const char *, ConstValue)
+    {
+        throw UnknownObject {};
+    }
+
+    /* Overloadable */
+    static void _getSupportedMipVersions(SelfComponentClass, ConstValue, LoggingLevel,
+                                         const UnsignedIntegerRangeSet ranges)
+    {
+        ranges.addRange(0, 0);
+    }
+
+    /* Overloadable */
+    void _graphIsConfigured()
+    {
+    }
+
+    /* Overloadable */
+    void _inputPortConnected(SelfComponentInputPort, ConstOutputPort)
+    {
+    }
+
+    MessageIterator::Shared _createMessageIterator(const _InputPorts::Port port)
+    {
+        return this->_selfComp().createMessageIterator(port);
+    }
+
+    template <typename DataT>
+    _InputPorts::Port _addInputPort(const char * const name, DataT * const data)
+    {
+        return this->_selfComp().addInputPort(name, data);
+    }
+
+    _InputPorts::Port _addInputPort(const char *name)
+    {
+        return this->_selfComp().addInputPort(name);
+    }
+
+    template <typename DataT>
+    _InputPorts::Port _addInputPort(const std::string& name, DataT * const data)
+    {
+        return this->_selfComp().addInputPort(name, data);
+    }
+
+    _InputPorts::Port _addInputPort(const std::string& name)
+    {
+        return this->_selfComp().addInputPort(name);
+    }
+
+    _InputPorts _inputPorts() noexcept
+    {
+        return this->_selfComp().inputPorts();
+    }
+};
+
+/*
+ * Base class of a user message iterator `UserMessageIteratorT` (CRTP)
+ * of which the parent user component class is `UserComponentT`.
+ *
+ * `UserMessageIteratorT::UserMessageIteratorT()` must accept a
+ * `bt2::SelfMessageIterator` parameter, which it needs to forward to
+ * bt2::UserMessageIterator::UserMessageIterator().
+ *
+ * The public next() method below (called by the bridge) implements the
+ * very common pattern of appending messages into the output array, and,
+ * meanwhile:
+ *
+ * If it catches a `bt2::TryAgain` exception:
+ *     If the message array isn't empty, transform this into a success
+ *     (don't throw).
+ *
+ *     Otherwise rethrow.
+ *
+ * If it catches an error:
+ *     If the message array isn't empty, transform this into a success
+ *     (don't throw), but save the error of the current thread and the
+ *     type of error to throw the next time the user calls next().
+ *
+ *     Otherwise rethrow.
+ *
+ * `UserMessageIteratorT` must implement:
+ *
+ *     void _next(bt2::ConstMessageArray& messages);
+ *
+ * This method fills `messages` with at most `messages.capacity()`
+ * messages and may throw `bt2::TryAgain` or a valid error whenever.
+ * Leaving an empty `messages` means the end of iteration.
+ */
+template <typename UserMessageIteratorT, typename UserComponentT>
+class UserMessageIterator
+{
+private:
+    /* Type of `_mExcToThrowType` */
+    enum class _ExcToThrowType
+    {
+        NONE,
+        ERROR,
+        MEM_ERROR,
+    };
+
+protected:
+    explicit UserMessageIterator(const SelfMessageIterator selfMsgIter,
+                                 const std::string& logTagSuffix) :
+        _mSelfMsgIter {selfMsgIter},
+        _mLogger {selfMsgIter,
+                  fmt::format("{}/{}", this->_component()._mLogger.tag(), logTagSuffix)}
+    {
+    }
+
+public:
+    ~UserMessageIterator()
+    {
+        this->_resetError();
+    }
+
+    void next(bt2::ConstMessageArray& messages)
+    {
+        /* Any saved error? Now is the time to throw */
+        if (G_UNLIKELY(_mExcToThrowType != _ExcToThrowType::NONE)) {
+            /* Move `_mSavedLibError`, if any, as current thread error */
+            if (_mSavedLibError) {
+                BT_CURRENT_THREAD_MOVE_ERROR_AND_RESET(_mSavedLibError);
+            }
+
+            /* Throw the corresponding exception */
+            if (_mExcToThrowType == _ExcToThrowType::ERROR) {
+                throw bt2::Error {};
+            } else {
+                BT_ASSERT(_mExcToThrowType == _ExcToThrowType::MEM_ERROR);
+                throw bt2::MemoryError {};
+            }
+        }
+
+        /*
+         * When catching some exception below, if our message array
+         * isn't empty, then return immediately before throwing to
+         * provide those messages to downstream.
+         *
+         * When catching an error, also save the current thread error,
+         * if any, so that we can restore it later (see the beginning of
+         * this method).
+         */
+        BT_ASSERT_DBG(_mExcToThrowType == _ExcToThrowType::NONE);
+
+        try {
+            this->_userObj()._next(messages);
+
+            /* We're done: everything below is exception handling */
+            return;
+        } catch (const bt2::TryAgain&) {
+            if (messages.isEmpty()) {
+                throw;
+            }
+        } catch (const std::bad_alloc&) {
+            if (messages.isEmpty()) {
+                throw;
+            }
+
+            _mExcToThrowType = _ExcToThrowType::MEM_ERROR;
+        } catch (const bt2::Error&) {
+            if (messages.isEmpty()) {
+                throw;
+            }
+
+            _mExcToThrowType = _ExcToThrowType::ERROR;
+        }
+
+        if (_mExcToThrowType != _ExcToThrowType::NONE) {
+            BT_CPPLOGE(
+                "An error occurred, but there are {} messages to return: delaying the error reporting.",
+                messages.length());
+            BT_ASSERT(!_mSavedLibError);
+            _mSavedLibError = bt_current_thread_take_error();
+        }
+    }
+
+    bool canSeekBeginning()
+    {
+        this->_resetError();
+        return this->_userObj()._canSeekBeginning();
+    }
+
+    void seekBeginning()
+    {
+        this->_resetError();
+        return this->_userObj()._seekBeginning();
+    }
+
+    bool canSeekNsFromOrigin(const std::int64_t nsFromOrigin)
+    {
+        this->_resetError();
+        return this->_userObj()._canSeekNsFromOrigin(nsFromOrigin);
+    }
+
+    void seekNsFromOrigin(const std::int64_t nsFromOrigin)
+    {
+        this->_resetError();
+        this->_userObj()._seekNsFromOrigin(nsFromOrigin);
+    }
+
+protected:
+    /* Overloadable */
+    bool _canSeekBeginning() noexcept
+    {
+        return false;
+    }
+
+    /* Overloadable */
+    void _seekBeginning() noexcept
+    {
+    }
+
+    /* Overloadable */
+    bool _canSeekNsFromOrigin(std::int64_t) noexcept
+    {
+        return false;
+    }
+
+    /* Overloadable */
+    void _seekNsFromOrigin(std::int64_t) noexcept
+    {
+    }
+
+    MessageIterator::Shared _createMessageIterator(const SelfComponentInputPort port)
+    {
+        return _mSelfMsgIter.createMessageIterator(port);
+    }
+
+    UserComponentT& _component() noexcept
+    {
+        return _mSelfMsgIter.component().template data<UserComponentT>();
+    }
+
+    SelfComponentOutputPort _port() noexcept
+    {
+        return _mSelfMsgIter.port();
+    }
+
+    bool _isInterrupted() const noexcept
+    {
+        return _mSelfMsgIter.isInterrupted();
+    }
+
+private:
+    UserMessageIteratorT& _userObj() noexcept
+    {
+        return static_cast<UserMessageIteratorT&>(*this);
+    }
+
+    void _resetError() noexcept
+    {
+        _mExcToThrowType = _ExcToThrowType::NONE;
+
+        if (_mSavedLibError) {
+            bt_error_release(_mSavedLibError);
+        }
+    }
+
+    SelfMessageIterator _mSelfMsgIter;
+
+    /*
+     * next() may accumulate messages, and then catch an error before
+     * returning. In that case, it saves the error of the current thread
+     * here so that it can return its accumulated messages and throw the
+     * next time.
+     *
+     * It also saves the type of the exception to throw the next time.
+     */
+    _ExcToThrowType _mExcToThrowType = _ExcToThrowType::NONE;
+    const bt_error *_mSavedLibError = nullptr;
+
+protected:
+    bt2c::Logger _mLogger;
+};
+
+} /* namespace bt2 */
+
+#define BT_CPP_PLUGIN_SOURCE_COMPONENT_CLASS_WITH_ID(                                              \
+    _pluginId, _componentClassId, _name, _userComponentClass, _userMessageIteratorClass)           \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_WITH_ID(                                                      \
+        _pluginId, _componentClassId, _name,                                                       \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::next);                         \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_INITIALIZE_METHOD_WITH_ID(                                    \
+        _pluginId, _componentClassId, bt2::internal::SrcCompClsBridge<_userComponentClass>::init); \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_FINALIZE_METHOD_WITH_ID(                                      \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SrcCompClsBridge<_userComponentClass>::finalize);                           \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_GET_SUPPORTED_MIP_VERSIONS_METHOD_WITH_ID(                    \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SrcCompClsBridge<_userComponentClass>::getSupportedMipVersions);            \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_OUTPUT_PORT_CONNECTED_METHOD_WITH_ID(                         \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SrcCompClsBridge<_userComponentClass>::outputPortConnected);                \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_QUERY_METHOD_WITH_ID(                                         \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SrcCompClsBridge<_userComponentClass>::query);                              \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD_WITH_ID(             \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::init);                         \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_FINALIZE_METHOD_WITH_ID(               \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::finalize);                     \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_SEEK_BEGINNING_METHODS_WITH_ID(        \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::seekBeginning,                 \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::canSeekBeginning);             \
+    BT_PLUGIN_SOURCE_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_SEEK_NS_FROM_ORIGIN_METHODS_WITH_ID(   \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::seekNsFromOrigin,              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::canSeekNsFromOrigin);
+
+#define BT_CPP_PLUGIN_FILTER_COMPONENT_CLASS_WITH_ID(                                              \
+    _pluginId, _componentClassId, _name, _userComponentClass, _userMessageIteratorClass)           \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_WITH_ID(                                                      \
+        _pluginId, _componentClassId, _name,                                                       \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::next);                         \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_INITIALIZE_METHOD_WITH_ID(                                    \
+        _pluginId, _componentClassId, bt2::internal::FltCompClsBridge<_userComponentClass>::init); \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_FINALIZE_METHOD_WITH_ID(                                      \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::FltCompClsBridge<_userComponentClass>::finalize);                           \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_GET_SUPPORTED_MIP_VERSIONS_METHOD_WITH_ID(                    \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::FltCompClsBridge<_userComponentClass>::getSupportedMipVersions);            \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_INPUT_PORT_CONNECTED_METHOD_WITH_ID(                          \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::FltCompClsBridge<_userComponentClass>::inputPortConnected);                 \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_OUTPUT_PORT_CONNECTED_METHOD_WITH_ID(                         \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::FltCompClsBridge<_userComponentClass>::outputPortConnected);                \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_QUERY_METHOD_WITH_ID(                                         \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::FltCompClsBridge<_userComponentClass>::query);                              \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_INITIALIZE_METHOD_WITH_ID(             \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::init);                         \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_FINALIZE_METHOD_WITH_ID(               \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::finalize);                     \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_SEEK_BEGINNING_METHODS_WITH_ID(        \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::seekBeginning,                 \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::canSeekBeginning);             \
+    BT_PLUGIN_FILTER_COMPONENT_CLASS_MESSAGE_ITERATOR_CLASS_SEEK_NS_FROM_ORIGIN_METHODS_WITH_ID(   \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::seekNsFromOrigin,              \
+        bt2::internal::MsgIterClsBridge<_userMessageIteratorClass>::canSeekNsFromOrigin);
+
+#define BT_CPP_PLUGIN_SINK_COMPONENT_CLASS_WITH_ID(_pluginId, _componentClassId, _name,            \
+                                                   _userComponentClass)                            \
+    BT_PLUGIN_SINK_COMPONENT_CLASS_WITH_ID(                                                        \
+        _pluginId, _componentClassId, _name,                                                       \
+        bt2::internal::SinkCompClsBridge<_userComponentClass>::consume);                           \
+    BT_PLUGIN_SINK_COMPONENT_CLASS_INITIALIZE_METHOD_WITH_ID(                                      \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SinkCompClsBridge<_userComponentClass>::init);                              \
+    BT_PLUGIN_SINK_COMPONENT_CLASS_FINALIZE_METHOD_WITH_ID(                                        \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SinkCompClsBridge<_userComponentClass>::finalize);                          \
+    BT_PLUGIN_SINK_COMPONENT_CLASS_GET_SUPPORTED_MIP_VERSIONS_METHOD_WITH_ID(                      \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SinkCompClsBridge<_userComponentClass>::getSupportedMipVersions);           \
+    BT_PLUGIN_SINK_COMPONENT_CLASS_INPUT_PORT_CONNECTED_METHOD_WITH_ID(                            \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SinkCompClsBridge<_userComponentClass>::inputPortConnected);                \
+    BT_PLUGIN_SINK_COMPONENT_CLASS_GRAPH_IS_CONFIGURED_METHOD_WITH_ID(                             \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SinkCompClsBridge<_userComponentClass>::graphIsConfigured);                 \
+    BT_PLUGIN_SINK_COMPONENT_CLASS_QUERY_METHOD_WITH_ID(                                           \
+        _pluginId, _componentClassId,                                                              \
+        bt2::internal::SinkCompClsBridge<_userComponentClass>::query);
+
+#define BT_CPP_PLUGIN_SOURCE_COMPONENT_CLASS(_name, _userComponentClass,                           \
+                                             _userMessageIteratorClass)                            \
+    BT_CPP_PLUGIN_SOURCE_COMPONENT_CLASS_WITH_ID(auto, _name, #_name, _userComponentClass,         \
+                                                 _userMessageIteratorClass)
+
+#define BT_CPP_PLUGIN_FILTER_COMPONENT_CLASS(_name, _userComponentClass,                           \
+                                             _userMessageIteratorClass)                            \
+    BT_CPP_PLUGIN_FILTER_COMPONENT_CLASS_WITH_ID(auto, _name, #_name, _userComponentClass,         \
+                                                 _userMessageIteratorClass)
+
+#define BT_CPP_PLUGIN_SINK_COMPONENT_CLASS(_name, _userComponentClass)                             \
+    BT_CPP_PLUGIN_SINK_COMPONENT_CLASS_WITH_ID(auto, _name, #_name, _userComponentClass)
+
+#endif /* BABELTRACE_CPP_COMMON_BT2_PLUGIN_DEV_HPP */
This page took 0.033819 seconds and 4 git commands to generate.