mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-12-25 06:23:53 +01:00
tests/gnrc_tx_sync: add test for gnrc_tx_sync
This commit is contained in:
parent
9895b840cd
commit
bd37457e14
22
tests/gnrc_tx_sync/Makefile
Normal file
22
tests/gnrc_tx_sync/Makefile
Normal file
@ -0,0 +1,22 @@
|
||||
include ../Makefile.tests_common
|
||||
|
||||
TEST_6LO ?= 1
|
||||
|
||||
ifeq (1,$(TEST_6LO))
|
||||
USEMODULE += gnrc_sixlowpan_default
|
||||
USEMODULE += netdev_ieee802154
|
||||
else
|
||||
USEMODULE += gnrc_ipv6_default
|
||||
USEMODULE += netdev_eth
|
||||
endif
|
||||
|
||||
USEMODULE += gnrc_tx_sync
|
||||
USEMODULE += netdev_test
|
||||
USEMODULE += sock_udp
|
||||
USEMODULE += xtimer
|
||||
USEMODULE += auto_init_gnrc_netif
|
||||
USEMODULE += iolist
|
||||
|
||||
include $(RIOTBASE)/Makefile.include
|
||||
|
||||
CFLAGS += -DDEBUG_ASSERT_VERBOSE=1
|
||||
22
tests/gnrc_tx_sync/Makefile.ci
Normal file
22
tests/gnrc_tx_sync/Makefile.ci
Normal file
@ -0,0 +1,22 @@
|
||||
BOARD_INSUFFICIENT_MEMORY := \
|
||||
bluepill-stm32f030c8 \
|
||||
i-nucleo-lrwan1 \
|
||||
msb-430 \
|
||||
msb-430h \
|
||||
nucleo-f030r8 \
|
||||
nucleo-f031k6 \
|
||||
nucleo-f042k6 \
|
||||
nucleo-f303k8 \
|
||||
nucleo-f334r8 \
|
||||
nucleo-l011k4 \
|
||||
nucleo-l031k6 \
|
||||
nucleo-l053r8 \
|
||||
samd10-xmini \
|
||||
slstk3400a \
|
||||
stk3200 \
|
||||
stm32f030f4-demo \
|
||||
stm32f0discovery \
|
||||
stm32l0538-disco \
|
||||
telosb \
|
||||
z1 \
|
||||
#
|
||||
223
tests/gnrc_tx_sync/main.c
Normal file
223
tests/gnrc_tx_sync/main.c
Normal file
@ -0,0 +1,223 @@
|
||||
/*
|
||||
* Copyright (C) 2020 Otto-von-Guericke-Universität Magdeburg
|
||||
*
|
||||
* This file is subject to the terms and conditions of the GNU Lesser
|
||||
* General Public License v2.1. See the file LICENSE in the top level
|
||||
* directory for more details.
|
||||
*/
|
||||
|
||||
/**
|
||||
* @{
|
||||
*
|
||||
* @file
|
||||
* @brief Text application for gnrc_tx_sync
|
||||
*
|
||||
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
|
||||
*
|
||||
* @}
|
||||
*/
|
||||
|
||||
/* keep include of stdint.h before stdatomic.h for compatibility with broken
|
||||
* toolchains */
|
||||
#include <stdint.h>
|
||||
#include <stdatomic.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#include "net/af.h"
|
||||
#include "net/gnrc/netif/raw.h"
|
||||
#include "net/ipv6/addr.h"
|
||||
#include "net/netdev_test.h"
|
||||
#include "net/sock/udp.h"
|
||||
#include "net/gnrc/pkt.h"
|
||||
#include "test_utils/expect.h"
|
||||
#include "xtimer.h"
|
||||
|
||||
#define NETIF_STACKSIZE THREAD_STACKSIZE_DEFAULT
|
||||
#define NETIF_PRIO (THREAD_PRIORITY_MAIN - 4)
|
||||
#define MAIN_QUEUE_SIZE (8)
|
||||
|
||||
static char netif_stack[NETIF_STACKSIZE];
|
||||
static msg_t main_msg_queue[MAIN_QUEUE_SIZE];
|
||||
|
||||
static atomic_int sends_completed = ATOMIC_VAR_INIT(0);
|
||||
static gnrc_netif_t netif;
|
||||
static netdev_test_t netdev_test;
|
||||
static netdev_t *netdev =
|
||||
#if IS_USED(MODULE_NETDEV_IEEE802154)
|
||||
&netdev_test.netdev.netdev;
|
||||
#else
|
||||
&netdev_test.netdev;
|
||||
#endif
|
||||
|
||||
static bool carries_test_message(const iolist_t *iolist)
|
||||
{
|
||||
/* This is dark magic, don't do this at home.
|
||||
* We rely on two things here:
|
||||
* 1.) As we're testing gnrc_tx_sync, we can be sure that the used network stack is GNRC.
|
||||
* Hence, iolist actually is of type gnrc_pktsnip_t * and we can just cast it back
|
||||
* 2.) There are only two types of messages being send in this test: ICMPv6 messages and
|
||||
* our UDP test message (or fragments of it).
|
||||
*
|
||||
* Detecting (unfragmented) ICMPv6 messages is trivial, so we just do this. All other frames
|
||||
* must be carrying our test message.
|
||||
*/
|
||||
for (const gnrc_pktsnip_t *iter = (const gnrc_pktsnip_t *)iolist; iter; iter = iter->next) {
|
||||
if (iter->type == GNRC_NETTYPE_ICMPV6) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static int netdev_send(netdev_t *dev, const iolist_t *iolist)
|
||||
{
|
||||
(void)dev;
|
||||
if (carries_test_message(iolist)) {
|
||||
/* sending part of UDP datagram */
|
||||
xtimer_msleep(100);
|
||||
atomic_fetch_add(&sends_completed, 1);
|
||||
}
|
||||
return iolist_size(iolist);
|
||||
}
|
||||
|
||||
static int netdev_get_device_type(netdev_t *dev, void *value, size_t max_len)
|
||||
{
|
||||
(void)dev;
|
||||
const uint16_t type_ipv6 = NETDEV_TYPE_ETHERNET;
|
||||
const uint16_t type_6lo = NETDEV_TYPE_IEEE802154;
|
||||
assert(max_len == sizeof(uint16_t));
|
||||
if (IS_USED(MODULE_NETDEV_IEEE802154)) {
|
||||
memcpy(value, &type_6lo, sizeof(type_6lo));
|
||||
}
|
||||
else {
|
||||
memcpy(value, &type_ipv6, sizeof(type_ipv6));
|
||||
}
|
||||
return sizeof(uint16_t);
|
||||
}
|
||||
|
||||
static int netdev_get_max_pdu_size(netdev_t *dev, void *value, size_t max_len)
|
||||
{
|
||||
(void)dev;
|
||||
const uint16_t pdu_size_ethernet = 1500;
|
||||
const uint16_t pdu_size_6lo = 32;
|
||||
assert(max_len == sizeof(uint16_t));
|
||||
if (IS_USED(MODULE_NETDEV_IEEE802154)) {
|
||||
memcpy(value, &pdu_size_6lo, sizeof(pdu_size_6lo));
|
||||
}
|
||||
else {
|
||||
memcpy(value, &pdu_size_ethernet, sizeof(pdu_size_ethernet));
|
||||
}
|
||||
return sizeof(uint16_t);
|
||||
}
|
||||
|
||||
static int netdev_get_proto(netdev_t *dev, void *value, size_t max_len)
|
||||
{
|
||||
(void)dev;
|
||||
const gnrc_nettype_t proto =
|
||||
#if IS_USED(MODULE_NETDEV_IEEE802154)
|
||||
GNRC_NETTYPE_SIXLOWPAN;
|
||||
#else
|
||||
GNRC_NETTYPE_IPV6;
|
||||
#endif
|
||||
assert(max_len == sizeof(proto));
|
||||
memcpy(value, &proto, sizeof(proto));
|
||||
return sizeof(proto);
|
||||
}
|
||||
|
||||
static int netdev_get_address(netdev_t *dev, void *value, size_t max_len)
|
||||
{
|
||||
(void)dev;
|
||||
const uint8_t addr[] = {
|
||||
#if IS_USED(MODULE_NETDEV_IEEE802154)
|
||||
0x13, 0x37
|
||||
#else
|
||||
0x13, 0x37, 0xac, 0xdc, 0xbe, 0xef
|
||||
#endif
|
||||
};
|
||||
assert(max_len >= sizeof(addr));
|
||||
memcpy(value, addr, sizeof(addr));
|
||||
return sizeof(addr);
|
||||
}
|
||||
|
||||
static int netdev_get_address_long(netdev_t *dev, void *value, size_t max_len)
|
||||
{
|
||||
(void)dev;
|
||||
const uint8_t addr[] = {0x13, 0x37, 0x13, 0x37, 0x13, 0x37, 0x13, 0x37};
|
||||
if (!IS_USED(MODULE_NETDEV_IEEE802154)) {
|
||||
return -ENOTSUP;
|
||||
}
|
||||
assert(max_len >= sizeof(addr));
|
||||
memcpy(value, addr, sizeof(addr));
|
||||
return sizeof(addr);
|
||||
}
|
||||
|
||||
static int netdev_get_src_len(netdev_t *dev, void *value, size_t max_len)
|
||||
{
|
||||
(void)dev;
|
||||
const uint16_t src_len = 2;
|
||||
if (!IS_USED(MODULE_NETDEV_IEEE802154)) {
|
||||
return -ENOTSUP;
|
||||
}
|
||||
assert(max_len == sizeof(src_len));
|
||||
memcpy(value, &src_len, sizeof(src_len));
|
||||
return sizeof(src_len);
|
||||
}
|
||||
|
||||
int main(void)
|
||||
{
|
||||
puts(
|
||||
"Test application for gnrc_tx_sync\n"
|
||||
"=================================\n"
|
||||
"\n"
|
||||
"This application sends a single UDP datagram over a virtual Ethernet\n"
|
||||
"or 802.15.4 device. (Set TEST_6LO to 1 for 802.15.4 and to 0 for\n"
|
||||
"Ethernet.) The simulated network devices blocks for 100ms - more than\n"
|
||||
"enough time for sock_upd_send() to return if no synchronization is done.\n"
|
||||
"If an 802.15.4 device is simulated, the test datagram will be transmitted\n"
|
||||
"in two fragments. In that case sock_udp_send() must only return after both\n"
|
||||
"fragments are out. In the Ethernet case, only a single frame is send.\n"
|
||||
);
|
||||
|
||||
if (IS_USED(MODULE_NETDEV_IEEE802154)) {
|
||||
puts("IEEE 802.15.4 mode (TEST_6LO=1), sending 2 6LoWPAN fragments");
|
||||
}
|
||||
else {
|
||||
puts("Ethernet mode (TEST_6LO=0), sending 1 IPv6 packet");
|
||||
}
|
||||
msg_init_queue(main_msg_queue, MAIN_QUEUE_SIZE);
|
||||
netdev_test_setup(&netdev_test, NULL);
|
||||
netdev_test_set_send_cb(&netdev_test, netdev_send);
|
||||
netdev_test_set_get_cb(&netdev_test, NETOPT_DEVICE_TYPE, netdev_get_device_type);
|
||||
netdev_test_set_get_cb(&netdev_test, NETOPT_MAX_PDU_SIZE, netdev_get_max_pdu_size);
|
||||
netdev_test_set_get_cb(&netdev_test, NETOPT_PROTO, netdev_get_proto);
|
||||
netdev_test_set_get_cb(&netdev_test, NETOPT_ADDRESS, netdev_get_address);
|
||||
netdev_test_set_get_cb(&netdev_test, NETOPT_ADDRESS_LONG, netdev_get_address_long);
|
||||
netdev_test_set_get_cb(&netdev_test, NETOPT_SRC_LEN, netdev_get_src_len);
|
||||
gnrc_netif_raw_create(&netif, netif_stack, sizeof(netif_stack), NETIF_PRIO,
|
||||
"netdev_test", netdev);
|
||||
|
||||
sock_udp_t sock;
|
||||
sock_udp_ep_t local = SOCK_IPV6_EP_ANY;
|
||||
sock_udp_ep_t remote = { .family = AF_INET6 };
|
||||
remote.port = 12345;
|
||||
ipv6_addr_set_all_nodes_multicast((ipv6_addr_t *)&remote.addr.ipv6,
|
||||
IPV6_ADDR_MCAST_SCP_LINK_LOCAL);
|
||||
expect(sock_udp_create(&sock, &local, NULL, 0) == 0);
|
||||
/* With 6LoWPAN, This test message needs exactly two fragments to be transmitted due
|
||||
* to the maximum L2 PDU of 32 bytes */
|
||||
static const char test_msg[33] = { 'T', 'e', 's', 't' };
|
||||
/* with gnrc_tx_sync, we expect sock_udp_send() to block until transmission is done */
|
||||
expect(sock_udp_send(&sock, test_msg, sizeof(test_msg), &remote) > 0);
|
||||
/* the virtual netdev device increments sends_completed for each frame carrying the test
|
||||
* payload. If this value has not reached the value of 1 (or 2 for 6LoWPAN fragmentation),
|
||||
* the test has failed. */
|
||||
int sends = atomic_load(&sends_completed);
|
||||
int sends_expected = (IS_USED(MODULE_NETDEV_IEEE802154)) ? 2 : 1;
|
||||
printf("transmissions expected = %d, transmissions completed = %d\n", sends_expected, sends);
|
||||
expect(sends == sends_expected);
|
||||
|
||||
puts("TEST PASSED");
|
||||
|
||||
return 0;
|
||||
}
|
||||
20
tests/gnrc_tx_sync/tests/01-run.py
Executable file
20
tests/gnrc_tx_sync/tests/01-run.py
Executable file
@ -0,0 +1,20 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
# Copyright (C) 2020 Otto-von-Guericke-Universität Magdeburg
|
||||
#
|
||||
# This file is subject to the terms and conditions of the GNU Lesser
|
||||
# General Public License v2.1. See the file LICENSE in the top level
|
||||
# directory for more details.
|
||||
|
||||
# @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
|
||||
|
||||
import sys
|
||||
from testrunner import run
|
||||
|
||||
|
||||
def testfunc(child):
|
||||
child.expect("TEST PASSED")
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
sys.exit(run(testfunc))
|
||||
Loading…
x
Reference in New Issue
Block a user