mirror of
https://gitee.com/mirrors_PX4/PX4-Autopilot.git
synced 2026-05-22 14:47:35 +08:00
221 lines
8.5 KiB
C++
221 lines
8.5 KiB
C++
/*
|
|
* Copyright (C) 2014 Pavel Kirienko <pavel.kirienko@gmail.com>
|
|
*/
|
|
|
|
#include <algorithm>
|
|
#include <gtest/gtest.h>
|
|
#include "transfer_test_helpers.hpp"
|
|
#include "can/can.hpp"
|
|
#include <uavcan/transport/transfer_sender.hpp>
|
|
|
|
|
|
static int sendOne(uavcan::TransferSender& sender, const std::string& data,
|
|
uint64_t monotonic_tx_deadline, uint64_t monotonic_blocking_deadline,
|
|
uavcan::TransferType transfer_type, uavcan::NodeID dst_node_id)
|
|
{
|
|
return sender.send(reinterpret_cast<const uint8_t*>(data.c_str()), data.length(),
|
|
uavcan::MonotonicTime::fromUSec(monotonic_tx_deadline),
|
|
uavcan::MonotonicTime::fromUSec(monotonic_blocking_deadline), transfer_type, dst_node_id);
|
|
}
|
|
|
|
static int sendOne(uavcan::TransferSender& sender, const std::string& data,
|
|
uint64_t monotonic_tx_deadline, uint64_t monotonic_blocking_deadline,
|
|
uavcan::TransferType transfer_type, uavcan::NodeID dst_node_id, uavcan::TransferID tid)
|
|
{
|
|
return sender.send(reinterpret_cast<const uint8_t*>(data.c_str()), data.length(),
|
|
uavcan::MonotonicTime::fromUSec(monotonic_tx_deadline),
|
|
uavcan::MonotonicTime::fromUSec(monotonic_blocking_deadline), transfer_type, dst_node_id, tid);
|
|
}
|
|
|
|
|
|
TEST(TransferSender, Basic)
|
|
{
|
|
uavcan::PoolManager<1> poolmgr;
|
|
|
|
SystemClockMock clockmock(100);
|
|
CanDriverMock driver(2, clockmock);
|
|
|
|
uavcan::OutgoingTransferRegistry<8> out_trans_reg(poolmgr);
|
|
|
|
static const uavcan::NodeID TX_NODE_ID(64);
|
|
static const uavcan::NodeID RX_NODE_ID(65);
|
|
uavcan::Dispatcher dispatcher_tx(driver, poolmgr, clockmock, out_trans_reg);
|
|
uavcan::Dispatcher dispatcher_rx(driver, poolmgr, clockmock, out_trans_reg);
|
|
ASSERT_TRUE(dispatcher_tx.setNodeID(TX_NODE_ID));
|
|
ASSERT_TRUE(dispatcher_rx.setNodeID(RX_NODE_ID));
|
|
|
|
/*
|
|
* Test environment
|
|
*/
|
|
static const uavcan::DataTypeDescriptor TYPES[2] =
|
|
{
|
|
makeDataType(uavcan::DataTypeKindMessage, 1),
|
|
makeDataType(uavcan::DataTypeKindService, 1)
|
|
};
|
|
|
|
uavcan::TransferSender senders[2] =
|
|
{
|
|
uavcan::TransferSender(dispatcher_tx, TYPES[0], uavcan::CanTxQueue::Volatile),
|
|
uavcan::TransferSender(dispatcher_tx, TYPES[1], uavcan::CanTxQueue::Persistent)
|
|
};
|
|
|
|
static const std::string DATA[4] =
|
|
{
|
|
"Don't panic.",
|
|
|
|
"The ships hung in the sky in much the same way that bricks don't.",
|
|
|
|
"Would it save you a lot of time if I just gave up and went mad now?",
|
|
|
|
"If there's anything more important than my ego around, I want it caught and shot now."
|
|
};
|
|
|
|
/*
|
|
* Transmission
|
|
*/
|
|
static const uint64_t TX_DEADLINE = 1000000;
|
|
|
|
sendOne(senders[0], DATA[0], TX_DEADLINE, 0, uavcan::TransferTypeMessageBroadcast, 0);
|
|
sendOne(senders[0], DATA[1], TX_DEADLINE, 0, uavcan::TransferTypeMessageUnicast, RX_NODE_ID);
|
|
sendOne(senders[0], "123", TX_DEADLINE, 0, uavcan::TransferTypeMessageBroadcast, 0);
|
|
sendOne(senders[0], "456", TX_DEADLINE, 0, uavcan::TransferTypeMessageUnicast, RX_NODE_ID);
|
|
|
|
sendOne(senders[1], DATA[2], TX_DEADLINE, 0, uavcan::TransferTypeServiceRequest, RX_NODE_ID);
|
|
sendOne(senders[1], DATA[3], TX_DEADLINE, 0, uavcan::TransferTypeServiceResponse, RX_NODE_ID, 1);
|
|
sendOne(senders[1], "", TX_DEADLINE, 0, uavcan::TransferTypeServiceRequest, RX_NODE_ID);
|
|
sendOne(senders[1], "", TX_DEADLINE, 0, uavcan::TransferTypeServiceResponse, RX_NODE_ID, 2);
|
|
|
|
static const Transfer TRANSFERS[8] =
|
|
{
|
|
Transfer(TX_DEADLINE, 0, uavcan::TransferTypeMessageBroadcast, 0, TX_NODE_ID, 0, DATA[0], TYPES[0]),
|
|
Transfer(TX_DEADLINE, 0, uavcan::TransferTypeMessageUnicast, 0, TX_NODE_ID, RX_NODE_ID, DATA[1], TYPES[0]),
|
|
Transfer(TX_DEADLINE, 0, uavcan::TransferTypeMessageBroadcast, 1, TX_NODE_ID, 0, "123", TYPES[0]),
|
|
Transfer(TX_DEADLINE, 0, uavcan::TransferTypeMessageUnicast, 1, TX_NODE_ID, RX_NODE_ID, "456", TYPES[0]),
|
|
|
|
Transfer(TX_DEADLINE, 0, uavcan::TransferTypeServiceRequest, 0, TX_NODE_ID, RX_NODE_ID, DATA[2], TYPES[1]),
|
|
Transfer(TX_DEADLINE, 0, uavcan::TransferTypeServiceResponse, 1, TX_NODE_ID, RX_NODE_ID, DATA[3], TYPES[1]),
|
|
Transfer(TX_DEADLINE, 0, uavcan::TransferTypeServiceRequest, 1, TX_NODE_ID, RX_NODE_ID, "", TYPES[1]),
|
|
Transfer(TX_DEADLINE, 0, uavcan::TransferTypeServiceResponse, 2, TX_NODE_ID, RX_NODE_ID, "", TYPES[1])
|
|
};
|
|
|
|
/*
|
|
* Receiving on the other side.
|
|
*/
|
|
for (int i = 0; i < driver.getNumIfaces(); i++) // Moving the frames from TX to RX side
|
|
{
|
|
CanIfaceMock& iface = driver.ifaces.at(i);
|
|
std::cout << "Num frames: " << iface.tx.size() << std::endl;
|
|
while (!iface.tx.empty())
|
|
{
|
|
CanIfaceMock::FrameWithTime ft = iface.tx.front();
|
|
iface.tx.pop();
|
|
iface.rx.push(ft);
|
|
}
|
|
}
|
|
|
|
TestListener<512, 2, 2> sub_msg(dispatcher_rx.getTransferPerfCounter(), TYPES[0], poolmgr);
|
|
TestListener<512, 2, 2> sub_srv_req(dispatcher_rx.getTransferPerfCounter(), TYPES[1], poolmgr);
|
|
TestListener<512, 2, 2> sub_srv_resp(dispatcher_rx.getTransferPerfCounter(), TYPES[1], poolmgr);
|
|
|
|
dispatcher_rx.registerMessageListener(&sub_msg);
|
|
dispatcher_rx.registerServiceRequestListener(&sub_srv_req);
|
|
dispatcher_rx.registerServiceResponseListener(&sub_srv_resp);
|
|
|
|
while (true)
|
|
{
|
|
const int res = dispatcher_rx.spin(tsMono(0));
|
|
ASSERT_LE(0, res);
|
|
clockmock.advance(100);
|
|
if (res == 0)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Validation
|
|
*/
|
|
ASSERT_TRUE(sub_msg.matchAndPop(TRANSFERS[0]));
|
|
ASSERT_TRUE(sub_msg.matchAndPop(TRANSFERS[1]));
|
|
ASSERT_TRUE(sub_msg.matchAndPop(TRANSFERS[2]));
|
|
ASSERT_TRUE(sub_msg.matchAndPop(TRANSFERS[3]));
|
|
|
|
ASSERT_TRUE(sub_srv_req.matchAndPop(TRANSFERS[4]));
|
|
ASSERT_TRUE(sub_srv_req.matchAndPop(TRANSFERS[6]));
|
|
|
|
ASSERT_TRUE(sub_srv_resp.matchAndPop(TRANSFERS[5]));
|
|
ASSERT_TRUE(sub_srv_resp.matchAndPop(TRANSFERS[7]));
|
|
|
|
/*
|
|
* Perf counters
|
|
*/
|
|
EXPECT_EQ(0, dispatcher_tx.getTransferPerfCounter().getErrorCount());
|
|
EXPECT_EQ(8, dispatcher_tx.getTransferPerfCounter().getTxTransferCount());
|
|
EXPECT_EQ(0, dispatcher_tx.getTransferPerfCounter().getRxTransferCount());
|
|
|
|
EXPECT_EQ(0, dispatcher_rx.getTransferPerfCounter().getErrorCount());
|
|
EXPECT_EQ(0, dispatcher_rx.getTransferPerfCounter().getTxTransferCount());
|
|
EXPECT_EQ(8, dispatcher_rx.getTransferPerfCounter().getRxTransferCount());
|
|
}
|
|
|
|
|
|
struct TransferSenderTestLoopbackFrameListener : public uavcan::LoopbackFrameListenerBase
|
|
{
|
|
uavcan::RxFrame last_frame;
|
|
unsigned int count;
|
|
|
|
TransferSenderTestLoopbackFrameListener(uavcan::Dispatcher& dispatcher)
|
|
: uavcan::LoopbackFrameListenerBase(dispatcher)
|
|
, count(0)
|
|
{
|
|
startListening();
|
|
}
|
|
|
|
void handleLoopbackFrame(const uavcan::RxFrame& frame)
|
|
{
|
|
last_frame = frame;
|
|
count++;
|
|
}
|
|
};
|
|
|
|
TEST(TransferSender, Loopback)
|
|
{
|
|
uavcan::PoolManager<1> poolmgr;
|
|
|
|
SystemClockMock clockmock(100);
|
|
CanDriverMock driver(2, clockmock);
|
|
|
|
uavcan::OutgoingTransferRegistry<8> out_trans_reg(poolmgr);
|
|
|
|
static const uavcan::NodeID TX_NODE_ID(64);
|
|
uavcan::Dispatcher dispatcher(driver, poolmgr, clockmock, out_trans_reg);
|
|
ASSERT_TRUE(dispatcher.setNodeID(TX_NODE_ID));
|
|
|
|
uavcan::DataTypeDescriptor desc = makeDataType(uavcan::DataTypeKindMessage, 1, "Foobar");
|
|
|
|
uavcan::TransferSender sender(dispatcher, desc, uavcan::CanTxQueue::Volatile);
|
|
|
|
sender.setCanIOFlags(uavcan::CanIOFlagLoopback);
|
|
ASSERT_EQ(uavcan::CanIOFlagLoopback, sender.getCanIOFlags());
|
|
|
|
sender.setIfaceMask(2);
|
|
ASSERT_EQ(2, sender.getIfaceMask());
|
|
|
|
TransferSenderTestLoopbackFrameListener listener(dispatcher);
|
|
|
|
ASSERT_LE(0, sender.send(reinterpret_cast<const uint8_t*>("123"), 3, tsMono(1000), tsMono(0),
|
|
uavcan::TransferTypeMessageBroadcast, 0));
|
|
|
|
ASSERT_EQ(0, listener.count);
|
|
ASSERT_EQ(0, dispatcher.spin(tsMono(1000)));
|
|
ASSERT_EQ(1, listener.count);
|
|
ASSERT_EQ(1, listener.last_frame.getIfaceIndex());
|
|
ASSERT_EQ(3, listener.last_frame.getPayloadLen());
|
|
ASSERT_TRUE(TX_NODE_ID == listener.last_frame.getSrcNodeID());
|
|
ASSERT_TRUE(listener.last_frame.isLast());
|
|
|
|
EXPECT_EQ(0, dispatcher.getTransferPerfCounter().getErrorCount());
|
|
EXPECT_EQ(1, dispatcher.getTransferPerfCounter().getTxTransferCount());
|
|
EXPECT_EQ(0, dispatcher.getTransferPerfCounter().getRxTransferCount());
|
|
}
|