mirror of
https://gitee.com/mirrors_PX4/PX4-Autopilot.git
synced 2026-05-23 10:17:35 +08:00
Subscriber with simple test
This commit is contained in:
@@ -0,0 +1,48 @@
|
||||
/*
|
||||
* Copyright (C) 2014 Pavel Kirienko <pavel.kirienko@gmail.com>
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <cassert>
|
||||
#include <uavcan/internal/transport/transfer_listener.hpp>
|
||||
|
||||
namespace uavcan
|
||||
{
|
||||
|
||||
template <typename DataType_>
|
||||
class ReceivedDataStructure : public DataType_
|
||||
{
|
||||
const IncomingTransfer* transfer_;
|
||||
|
||||
template <typename Ret, Ret (IncomingTransfer::*Fun)() const>
|
||||
Ret safeget() const
|
||||
{
|
||||
if (!transfer_)
|
||||
{
|
||||
assert(0);
|
||||
return Ret();
|
||||
}
|
||||
return (transfer_->*Fun)();
|
||||
}
|
||||
|
||||
protected:
|
||||
ReceivedDataStructure() : transfer_(NULL) { }
|
||||
|
||||
void setTransfer(const IncomingTransfer* transfer)
|
||||
{
|
||||
assert(transfer);
|
||||
transfer_ = transfer;
|
||||
}
|
||||
|
||||
public:
|
||||
typedef DataType_ DataType;
|
||||
|
||||
uint64_t getMonotonicTimestamp() const { return safeget<uint64_t, &IncomingTransfer::getMonotonicTimestamp>(); }
|
||||
uint64_t getUtcTimestamp() const { return safeget<uint64_t, &IncomingTransfer::getUtcTimestamp>(); }
|
||||
TransferType getTransferType() const { return safeget<TransferType, &IncomingTransfer::getTransferType>(); }
|
||||
TransferID getTransferID() const { return safeget<TransferID, &IncomingTransfer::getTransferID>(); }
|
||||
NodeID getSrcNodeID() const { return safeget<NodeID, &IncomingTransfer::getSrcNodeID>(); }
|
||||
};
|
||||
|
||||
}
|
||||
@@ -0,0 +1,158 @@
|
||||
/*
|
||||
* Copyright (C) 2014 Pavel Kirienko <pavel.kirienko@gmail.com>
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <uavcan/scheduler.hpp>
|
||||
#include <uavcan/data_type.hpp>
|
||||
#include <uavcan/global_data_type_registry.hpp>
|
||||
#include <uavcan/received_data_structure.hpp>
|
||||
#include <uavcan/internal/debug.hpp>
|
||||
#include <uavcan/internal/util.hpp>
|
||||
#include <uavcan/internal/lazy_constructor.hpp>
|
||||
#include <uavcan/internal/transport/transfer_listener.hpp>
|
||||
#include <uavcan/internal/marshal/scalar_codec.hpp>
|
||||
#include <uavcan/internal/marshal/types.hpp>
|
||||
|
||||
namespace uavcan
|
||||
{
|
||||
|
||||
template <typename DataType_,
|
||||
typename Callback = void(*)(const ReceivedDataStructure<DataType_>&),
|
||||
unsigned int NumStaticBufs = 1,
|
||||
unsigned int NumStaticReceivers = NumStaticBufs + 1>
|
||||
class Subscriber : Noncopyable
|
||||
{
|
||||
typedef Subscriber<DataType_, Callback, NumStaticBufs, NumStaticReceivers> SelfType;
|
||||
|
||||
public:
|
||||
typedef DataType_ DataType;
|
||||
|
||||
private:
|
||||
typedef TransferListener<BitLenToByteLen<DataType::MaxBitLen>::Result,
|
||||
NumStaticBufs ? NumStaticBufs : 1, // TODO: add support for zero buffers
|
||||
NumStaticReceivers ? NumStaticReceivers : 1> TransferListenerType;
|
||||
|
||||
// We need to break the inheritance chain here to implement lazy initialization
|
||||
class TransferForwarder : public TransferListenerType
|
||||
{
|
||||
SelfType& obj_;
|
||||
|
||||
void handleIncomingTransfer(IncomingTransfer& transfer)
|
||||
{
|
||||
obj_.handleIncomingTransfer(transfer);
|
||||
}
|
||||
|
||||
public:
|
||||
TransferForwarder(SelfType& obj, const DataTypeDescriptor& data_type, IAllocator& allocator)
|
||||
: TransferListenerType(data_type, allocator)
|
||||
, obj_(obj)
|
||||
{ }
|
||||
};
|
||||
|
||||
struct ReceivedDataStructureSpec : public ReceivedDataStructure<DataType>
|
||||
{
|
||||
using ReceivedDataStructure<DataType>::setTransfer;
|
||||
};
|
||||
|
||||
Scheduler& scheduler_;
|
||||
IAllocator& allocator_;
|
||||
Callback callback_;
|
||||
LazyConstructor<TransferForwarder> forwarder_;
|
||||
ReceivedDataStructureSpec message_;
|
||||
uint32_t failure_count_;
|
||||
|
||||
bool checkInit()
|
||||
{
|
||||
if (forwarder_)
|
||||
return true;
|
||||
|
||||
GlobalDataTypeRegistry::instance().freeze();
|
||||
|
||||
const DataTypeDescriptor* const descr =
|
||||
GlobalDataTypeRegistry::instance().find(DataTypeKindMessage, DataType::getDataTypeFullName());
|
||||
if (!descr)
|
||||
{
|
||||
UAVCAN_TRACE("Subscriber", "Type [%s] is not registered", DataType::getDataTypeFullName());
|
||||
return false;
|
||||
}
|
||||
forwarder_.template construct<SelfType&, const DataTypeDescriptor&, IAllocator&>(*this, *descr, allocator_);
|
||||
return true;
|
||||
}
|
||||
|
||||
void handleIncomingTransfer(IncomingTransfer& transfer)
|
||||
{
|
||||
assert(transfer.getTransferType() == TransferTypeMessageBroadcast ||
|
||||
transfer.getTransferType() == TransferTypeMessageUnicast);
|
||||
|
||||
{
|
||||
BitStream bitstream(transfer);
|
||||
ScalarCodec codec(bitstream);
|
||||
|
||||
const int decode_res = DataType::decode(message_, codec);
|
||||
// We don't need the data anymore, the memory can be reused from the callback:
|
||||
transfer.release();
|
||||
if (decode_res <= 0)
|
||||
{
|
||||
UAVCAN_TRACE("Subscriber", "Unable to decode the message [%i] [%s]",
|
||||
decode_res, DataType::getDataTypeFullName());
|
||||
failure_count_++;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
message_.setTransfer(&transfer);
|
||||
if (try_implicit_cast<bool>(callback_, true))
|
||||
callback_(message_); // Callback can accept non-const message reference and mutate it, that's OK
|
||||
else
|
||||
assert(0);
|
||||
}
|
||||
|
||||
public:
|
||||
Subscriber(Scheduler& scheduler, IAllocator& allocator)
|
||||
: scheduler_(scheduler)
|
||||
, allocator_(allocator)
|
||||
, callback_()
|
||||
, failure_count_(0)
|
||||
{
|
||||
StaticAssert<DataTypeKind(DataType::DataTypeKind) == DataTypeKindMessage>::check();
|
||||
}
|
||||
|
||||
int start(Callback callback)
|
||||
{
|
||||
stop();
|
||||
|
||||
if (!try_implicit_cast<bool>(callback, true))
|
||||
{
|
||||
UAVCAN_TRACE("Subscriber", "Invalid callback");
|
||||
return -1;
|
||||
}
|
||||
callback_ = callback;
|
||||
|
||||
if (!checkInit())
|
||||
{
|
||||
UAVCAN_TRACE("Subscriber", "Initialization failure [%s]", DataType::getDataTypeFullName());
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (!scheduler_.getDispatcher().registerMessageListener(forwarder_))
|
||||
{
|
||||
UAVCAN_TRACE("Subscriber", "Failed to register message listener [%s]", DataType::getDataTypeFullName());
|
||||
return -1;
|
||||
}
|
||||
return 1;
|
||||
}
|
||||
|
||||
void stop()
|
||||
{
|
||||
if (forwarder_)
|
||||
scheduler_.getDispatcher().unregisterMessageListener(forwarder_);
|
||||
}
|
||||
|
||||
Scheduler& getScheduler() const { return scheduler_; }
|
||||
|
||||
uint32_t getFailureCount() const { return failure_count_; }
|
||||
};
|
||||
|
||||
}
|
||||
@@ -0,0 +1,164 @@
|
||||
/*
|
||||
* Copyright (C) 2014 Pavel Kirienko <pavel.kirienko@gmail.com>
|
||||
*/
|
||||
|
||||
#include <gtest/gtest.h>
|
||||
#include <uavcan/subscriber.hpp>
|
||||
#include <uavcan/internal/method_binder.hpp>
|
||||
#include <uavcan/mavlink/Message.hpp>
|
||||
#include "common.hpp"
|
||||
#include "transport/can/iface_mock.hpp"
|
||||
|
||||
|
||||
template <typename DataType>
|
||||
struct SubscriptionListener
|
||||
{
|
||||
typedef uavcan::ReceivedDataStructure<DataType> ReceivedDataStructure;
|
||||
|
||||
struct ReceivedDataStructureCopy
|
||||
{
|
||||
uint64_t ts_monotonic;
|
||||
uint64_t ts_utc;
|
||||
uavcan::TransferType transfer_type;
|
||||
uavcan::TransferID transfer_id;
|
||||
uavcan::NodeID src_node_id;
|
||||
DataType msg;
|
||||
|
||||
ReceivedDataStructureCopy(const ReceivedDataStructure& s)
|
||||
: ts_monotonic(s.getMonotonicTimestamp())
|
||||
, ts_utc(s.getUtcTimestamp())
|
||||
, transfer_type(s.getTransferType())
|
||||
, transfer_id(s.getTransferID())
|
||||
, src_node_id(s.getSrcNodeID())
|
||||
, msg(s)
|
||||
{ }
|
||||
};
|
||||
|
||||
std::vector<DataType> simple;
|
||||
std::vector<ReceivedDataStructureCopy> extended;
|
||||
|
||||
void receiveExtended(ReceivedDataStructure& msg)
|
||||
{
|
||||
extended.push_back(msg);
|
||||
}
|
||||
|
||||
void receiveSimple(DataType& msg)
|
||||
{
|
||||
simple.push_back(msg);
|
||||
}
|
||||
|
||||
typedef SubscriptionListener<DataType> SelfType;
|
||||
typedef uavcan::MethodBinder<SelfType*, void (SelfType::*)(ReceivedDataStructure&)> ExtendedBinder;
|
||||
typedef uavcan::MethodBinder<SelfType*, void (SelfType::*)(DataType&)> SimpleBinder;
|
||||
|
||||
ExtendedBinder bindExtended() { return ExtendedBinder(this, &SelfType::receiveExtended); }
|
||||
SimpleBinder bindSimple() { return SimpleBinder(this, &SelfType::receiveSimple); }
|
||||
};
|
||||
|
||||
// TODO: add autogenerated comparison operators, then remove this
|
||||
static bool operator==(const uavcan::mavlink::Message& a, const uavcan::mavlink::Message& b)
|
||||
{
|
||||
return
|
||||
a.seq == b.seq &&
|
||||
a.sysid == b.sysid &&
|
||||
a.compid == b.compid &&
|
||||
a.msgid == b.msgid &&
|
||||
a.payload == b.payload;
|
||||
}
|
||||
|
||||
TEST(Subscriber, Basic)
|
||||
{
|
||||
uavcan::PoolAllocator<uavcan::MemPoolBlockSize * 8, uavcan::MemPoolBlockSize> pool;
|
||||
uavcan::PoolManager<1> poolmgr;
|
||||
poolmgr.addPool(&pool);
|
||||
|
||||
// Manual type registration - we can't rely on the GDTR state
|
||||
uavcan::GlobalDataTypeRegistry::instance().reset();
|
||||
uavcan::DefaultDataTypeRegistrator<uavcan::mavlink::Message> _registrator;
|
||||
|
||||
SystemClockDriver clock_driver;
|
||||
CanDriverMock can_driver(2, clock_driver);
|
||||
|
||||
uavcan::OutgoingTransferRegistry<8> out_trans_reg(poolmgr);
|
||||
|
||||
uavcan::Scheduler sch(can_driver, poolmgr, clock_driver, out_trans_reg, uavcan::NodeID(1));
|
||||
|
||||
typedef SubscriptionListener<uavcan::mavlink::Message> Listener;
|
||||
|
||||
uavcan::Subscriber<uavcan::mavlink::Message, Listener::ExtendedBinder> sub_extended(sch, poolmgr);
|
||||
uavcan::Subscriber<uavcan::mavlink::Message, Listener::SimpleBinder> sub_simple(sch, poolmgr);
|
||||
|
||||
// Null binder - will fail
|
||||
ASSERT_EQ(-1, sub_extended.start(Listener::ExtendedBinder(NULL, NULL)));
|
||||
|
||||
Listener listener;
|
||||
|
||||
/*
|
||||
* Message layout:
|
||||
* uint8 seq
|
||||
* uint8 sysid
|
||||
* uint8 compid
|
||||
* uint8 msgid
|
||||
* uint8[<256] payload
|
||||
*/
|
||||
uavcan::mavlink::Message expected_msg;
|
||||
expected_msg.seq = 0x42;
|
||||
expected_msg.sysid = 0x72;
|
||||
expected_msg.compid = 0x08;
|
||||
expected_msg.msgid = 0xa5;
|
||||
expected_msg.payload = "Msg";
|
||||
|
||||
const uint8_t transfer_payload[] = {0x42, 0x72, 0x08, 0xa5, 'M', 's', 'g'};
|
||||
|
||||
/*
|
||||
* RxFrame generation
|
||||
*/
|
||||
std::vector<uavcan::RxFrame> rx_frames;
|
||||
for (int i = 0; i < 4; i++)
|
||||
{
|
||||
uavcan::TransferType tt = (i & 1) ? uavcan::TransferTypeMessageUnicast : uavcan::TransferTypeMessageBroadcast;
|
||||
uavcan::NodeID dni = (tt == uavcan::TransferTypeMessageBroadcast) ?
|
||||
uavcan::NodeID::Broadcast : sch.getDispatcher().getSelfNodeID();
|
||||
// uint_fast16_t data_type_id, TransferType transfer_type, NodeID src_node_id, NodeID dst_node_id,
|
||||
// uint_fast8_t frame_index, TransferID transfer_id, bool last_frame
|
||||
uavcan::Frame frame(uavcan::mavlink::Message::DefaultDataTypeID, tt, uavcan::NodeID(i + 100), dni, 0, i, true);
|
||||
frame.setPayload(transfer_payload, 7);
|
||||
uavcan::RxFrame rx_frame(frame, clock_driver.getMonotonicMicroseconds(), clock_driver.getUtcMicroseconds(), 0);
|
||||
rx_frames.push_back(rx_frame);
|
||||
}
|
||||
|
||||
/*
|
||||
* Reception
|
||||
*/
|
||||
ASSERT_EQ(1, sub_extended.start(listener.bindExtended()));
|
||||
ASSERT_EQ(1, sub_simple.start(listener.bindSimple()));
|
||||
|
||||
for (unsigned int i = 0; i < rx_frames.size(); i++)
|
||||
{
|
||||
can_driver.ifaces[0].pushRx(rx_frames[i]);
|
||||
can_driver.ifaces[1].pushRx(rx_frames[i]);
|
||||
}
|
||||
|
||||
ASSERT_LE(0, sch.spin(clock_driver.getMonotonicMicroseconds() + 10000));
|
||||
|
||||
/*
|
||||
* Validation
|
||||
*/
|
||||
for (unsigned int i = 0; i < rx_frames.size(); i++)
|
||||
{
|
||||
const Listener::ReceivedDataStructureCopy s = listener.extended.at(i);
|
||||
ASSERT_TRUE(s.msg == expected_msg);
|
||||
ASSERT_EQ(rx_frames[i].getSrcNodeID(), s.src_node_id);
|
||||
ASSERT_EQ(rx_frames[i].getTransferID(), s.transfer_id);
|
||||
ASSERT_EQ(rx_frames[i].getTransferType(), s.transfer_type);
|
||||
ASSERT_EQ(rx_frames[i].getMonotonicTimestamp(), s.ts_monotonic);
|
||||
}
|
||||
|
||||
for (unsigned int i = 0; i < rx_frames.size(); i++)
|
||||
{
|
||||
ASSERT_TRUE(listener.simple.at(i) == expected_msg);
|
||||
}
|
||||
|
||||
ASSERT_EQ(0, sub_extended.getFailureCount());
|
||||
ASSERT_EQ(0, sub_simple.getFailureCount());
|
||||
}
|
||||
Reference in New Issue
Block a user