1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/tests/unittests/netdev_dummy/netdev_dummy.c
Martine Lenders cf3fdaabb6 unittests: add test network device
The reasoning behind this commit is twofold:
1. test the "lower" part (the device part) of netdev
2. provide a well defined testing infrastructure for modules that utilise
   netdev
The actual tests that represent 1. are provided in the following commit
2014-10-30 20:37:20 +01:00

702 lines
19 KiB
C

/*
* Copyright (C) 2014 Martin Lenders <mlenders@inf.fu-berlin.de>
*
* 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.
*/
/**
* @addtogroup
* @{
*
* @file netdev_dummy.c
*
* @author Martine Lenders <mlenders@inf.fu-berlin.de>
*/
#include <stdlib.h>
#include <stdint.h>
#include <string.h>
#include "netdev/base.h"
#include "netdev_dummy.h"
#define _NETDEV_MORE(dev) ((_ut_dev_internal *)(dev->more))
typedef struct {
char dst[UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN];
size_t dst_len;
char src[UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN];
size_t src_len;
char data[UNITTESTS_NETDEV_DUMMY_MAX_PACKET];
size_t data_len;
} _unittest_test_buffer;
typedef struct {
unsigned int channel;
char short_address[UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN];
unsigned int nid;
char long_address[UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN];
int tx_power;
netdev_state_t state;
netdev_rcv_data_cb_t callbacks[UNITTESTS_NETDEV_DUMMY_MAX_CB];
_unittest_test_buffer rx_buffer;
_unittest_test_buffer tx_buffer;
uint32_t last_event;
} _ut_dev_internal;
netdev_t unittest_netdev_dummy_devs[UNITTESTS_NETDEV_DUMMY_MAX_DEV];
static _ut_dev_internal _netdevs_internal[UNITTESTS_NETDEV_DUMMY_MAX_DEV];
static int _find_dev(netdev_t *dev)
{
if (dev == NULL || dev < (&(unittest_netdev_dummy_devs[0])) ||
dev > (&(unittest_netdev_dummy_devs[UNITTESTS_NETDEV_DUMMY_MAX_DEV - 1]))) {
return -1;
}
for (int i = 0; i < UNITTESTS_NETDEV_DUMMY_MAX_DEV; i++) {
if (&(unittest_netdev_dummy_devs[i]) == dev) {
return i;
}
}
return -1;
}
static int _init(netdev_t *dev)
{
if (_find_dev(dev) < 0) {
return -ENODEV;
}
_NETDEV_MORE(dev)->channel = 0;
_NETDEV_MORE(dev)->nid = 0;
_NETDEV_MORE(dev)->state = 0;
memset(_NETDEV_MORE(dev)->short_address, 0,
UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN);
memset(_NETDEV_MORE(dev)->long_address, 0,
UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN);
memset(&(_NETDEV_MORE(dev)->rx_buffer), 0, sizeof(_unittest_test_buffer));
memset(&(_NETDEV_MORE(dev)->tx_buffer), 0, sizeof(_unittest_test_buffer));
_NETDEV_MORE(dev)->last_event = 0;
for (int j = 0; j < UNITTESTS_NETDEV_DUMMY_MAX_CB; j++) {
_NETDEV_MORE(dev)->callbacks[j] = NULL;
}
return 0;
}
static int _send_data(netdev_t *dev, void *dest, size_t dest_len,
netdev_hlist_t *upper_layer_hdrs, void *data,
size_t data_len)
{
netdev_hlist_t *ptr = upper_layer_hdrs;
size_t tx_ptr = 0;
if (dest == NULL || data == NULL) {
return -EFAULT;
}
if (_find_dev(dev) < 0) {
return -ENODEV;
}
if ((data_len + netdev_get_hlist_len(upper_layer_hdrs)) >
UNITTESTS_NETDEV_DUMMY_MAX_PACKET) {
return -EMSGSIZE;
}
if (dest_len > UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN) {
return -EAFNOSUPPORT;
}
memcpy(_NETDEV_MORE(dev)->tx_buffer.dst, dest, dest_len);
_NETDEV_MORE(dev)->tx_buffer.dst_len = dest_len;
_NETDEV_MORE(dev)->tx_buffer.data_len = 0;
if (upper_layer_hdrs) {
do {
memcpy(&(_NETDEV_MORE(dev)->tx_buffer.data[tx_ptr]),
ptr->header, ptr->header_len);
_NETDEV_MORE(dev)->tx_buffer.data_len += ptr->header_len;
tx_ptr += ptr->header_len;
netdev_hlist_advance(&ptr);
} while (ptr != upper_layer_hdrs);
}
memcpy(&(_NETDEV_MORE(dev)->tx_buffer.data[tx_ptr]), data,
data_len);
_NETDEV_MORE(dev)->tx_buffer.data_len += data_len;
return _NETDEV_MORE(dev)->tx_buffer.data_len;
}
static int _add_receive_data_callback(netdev_t *dev, netdev_rcv_data_cb_t cb)
{
if (_find_dev(dev) < 0) {
return -ENODEV;
}
for (int j = 0; j < UNITTESTS_NETDEV_DUMMY_MAX_CB; j++) {
if (_NETDEV_MORE(dev)->callbacks[j] == cb) {
return 0;
}
if (_NETDEV_MORE(dev)->callbacks[j] == NULL) {
_NETDEV_MORE(dev)->callbacks[j] = cb;
return 0;
}
}
return -ENOBUFS;
}
static int _rem_receive_data_callback(netdev_t *dev, netdev_rcv_data_cb_t cb)
{
if (_find_dev(dev) < 0) {
return -ENODEV;
}
for (int j = 0; j < UNITTESTS_NETDEV_DUMMY_MAX_CB; j++) {
if (_NETDEV_MORE(dev)->callbacks[j] == cb) {
_NETDEV_MORE(dev)->callbacks[j] = NULL;
return 0;
}
}
return 0;
}
static int _get_option(netdev_t *dev, netdev_opt_t opt, void *value,
size_t *value_len)
{
if (_find_dev(dev) < 0) {
return -ENODEV;
}
if (value == NULL || value_len == NULL) {
return -EFAULT;
}
switch (opt) {
case NETDEV_OPT_CHANNEL:
if (*value_len < sizeof(unsigned int)) {
return -EOVERFLOW;
}
if (*value_len > sizeof(unsigned int)) {
*value_len = sizeof(unsigned int);
}
*((unsigned int *)value) = _NETDEV_MORE(dev)->channel;
break;
case NETDEV_OPT_ADDRESS:
if (*value_len < UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN) {
return -EOVERFLOW;
}
if (*value_len > UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN) {
*value_len = UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN;
}
memcpy(value, _NETDEV_MORE(dev)->short_address, *value_len);
break;
case NETDEV_OPT_NID:
if (*value_len < sizeof(unsigned int)) {
return -EOVERFLOW;
}
if (*value_len > sizeof(unsigned int)) {
*value_len = sizeof(unsigned int);
}
*((unsigned int *)value) = _NETDEV_MORE(dev)->nid;
break;
case NETDEV_OPT_ADDRESS_LONG:
if (*value_len < UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN) {
return -EOVERFLOW;
}
if (*value_len > UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN) {
*value_len = UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN;
}
memcpy(value, _NETDEV_MORE(dev)->long_address, *value_len);
break;
case NETDEV_OPT_TX_POWER:
if (*value_len < sizeof(int)) {
return -EOVERFLOW;
}
if (*value_len > sizeof(int)) {
*value_len = sizeof(int);
}
*((unsigned int *)value) = _NETDEV_MORE(dev)->tx_power;
break;
case NETDEV_OPT_MAX_PACKET_SIZE:
#if UNITTESTS_NETDEV_DUMMY_MAX_PACKET < 256
if (*value_len == 0) {
return -EOVERFLOW;
}
*value_len = sizeof(uint8_t);
*((uint8_t *)value) = (uint8_t)UNITTESTS_NETDEV_DUMMY_MAX_PACKET;
#elif UNITTESTS_NETDEV_DUMMY_MAX_PACKET < (1 << 16)
if (*value_len < sizeof(uint16_t)) {
return -EOVERFLOW;
}
*value_len = sizeof(uint16_t);
*((uint16_t *)value) = (uint16_t)UNITTESTS_NETDEV_DUMMY_MAX_PACKET;
#elif UNITTESTS_NETDEV_DUMMY_MAX_PACKET < (1 << 32)
if (*value_len < sizeof(uint32_t)) {
return -EOVERFLOW;
}
*value_len = sizeof(uint32_t);
*((uint32_t *)value) = (uint32_t)UNITTESTS_NETDEV_DUMMY_MAX_PACKET;
#else
if (*value_len < sizeof(uint64_t)) {
return -EOVERFLOW;
}
*value_len = sizeof(uint64_t);
*((uint64_t *)value) = (uint64_t)UNITTESTS_NETDEV_DUMMY_MAX_PACKET;
#endif
break;
case NETDEV_OPT_PROTO:
if (*value_len < sizeof(netdev_proto_t)) {
return -EOVERFLOW;
}
if (*value_len > sizeof(netdev_proto_t)) {
*value_len = sizeof(netdev_proto_t);
}
*((netdev_proto_t *)value) = NETDEV_PROTO_UNKNOWN;
break;
default:
return -ENOTSUP;
}
return 0;
}
static int _type_pun_up_unsigned(void *value_out, size_t desired_len,
void *value_in, size_t given_len)
{
if (given_len > desired_len) {
return -EOVERFLOW;
}
/* XXX this is ugly, but bear with me */
switch (given_len) {
case 8:
switch (desired_len) {
case 8:
*((uint64_t *)value_out) = (*((uint64_t *)value_in));
return 0;
default:
return -EINVAL;
}
case 4:
switch (desired_len) {
case 8:
*((uint64_t *)value_out) = (uint64_t)(*((uint32_t *)value_in));
return 0;
case 4:
*((uint32_t *)value_out) = (*((uint32_t *)value_in));
return 0;
default:
return -EINVAL;
}
case 2:
switch (desired_len) {
case 8:
*((uint64_t *)value_out) = (uint64_t)(*((uint16_t *)value_in));
return 0;
case 4:
*((uint32_t *)value_out) = (uint32_t)(*((uint16_t *)value_in));
return 0;
case 2:
*((uint16_t *)value_out) = (*((uint16_t *)value_in));
return 0;
default:
return -EINVAL;
}
case 1:
switch (desired_len) {
case 8:
*((uint64_t *)value_out) = (uint64_t)(*((uint8_t *)value_in));
return 0;
case 4:
*((uint32_t *)value_out) = (uint32_t)(*((uint8_t *)value_in));
return 0;
case 2:
*((uint16_t *)value_out) = (uint16_t)(*((uint8_t *)value_in));
return 0;
case 1:
*((uint8_t *)value_out) = (*((uint8_t *)value_in));
return 0;
default:
return -EINVAL;
}
default:
return -EINVAL;
}
}
static int _type_pun_up_signed(void *value_out, size_t desired_len,
void *value_in, size_t given_len)
{
if (given_len > desired_len) {
return -EOVERFLOW;
}
/* XXX this is ugly, but bear with me */
switch (given_len) {
case 8:
switch (desired_len) {
case 8:
*((int64_t *)value_out) = (*((int64_t *)value_in));
return 0;
default:
return -EINVAL;
}
case 4:
switch (desired_len) {
case 8:
*((int64_t *)value_out) = (int64_t)(*((int32_t *)value_in));
return 0;
case 4:
*((int32_t *)value_out) = (*((int32_t *)value_in));
return 0;
default:
return -EINVAL;
}
case 2:
switch (desired_len) {
case 8:
*((int64_t *)value_out) = (int64_t)(*((int16_t *)value_in));
return 0;
case 4:
*((int32_t *)value_out) = (int32_t)(*((int16_t *)value_in));
return 0;
case 2:
*((int16_t *)value_out) = (*((int16_t *)value_in));
return 0;
default:
return -EINVAL;
}
case 1:
switch (desired_len) {
case 8:
*((int64_t *)value_out) = (int64_t)(*((int8_t *)value_in));
return 0;
case 4:
*((int32_t *)value_out) = (int32_t)(*((int8_t *)value_in));
return 0;
case 2:
*((int16_t *)value_out) = (int16_t)(*((int8_t *)value_in));
return 0;
case 1:
*((int8_t *)value_out) = (*((int8_t *)value_in));
return 0;
default:
return -EINVAL;
}
default:
return -EINVAL;
}
}
static int _set_option(netdev_t *dev, netdev_opt_t opt, void *value,
size_t value_len)
{
uint8_t set_value[UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN];
int res = 0;
if (_find_dev(dev) < 0) {
return -ENODEV;
}
if (value == NULL) {
return -EFAULT;
}
switch (opt) {
case NETDEV_OPT_CHANNEL:
if ((res = _type_pun_up_unsigned(set_value, sizeof(unsigned int),
value, value_len)) == 0) {
_NETDEV_MORE(dev)->channel = *((unsigned int *)value);
}
break;
case NETDEV_OPT_ADDRESS:
if (value_len > UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN) {
return -EOVERFLOW;
}
do {
int start = UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN - value_len;
if (value_len == 0) {
return -EINVAL;
}
if (value_len < UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN) {
memcpy(_NETDEV_MORE(dev)->short_address, value, start);
}
memcpy(&(_NETDEV_MORE(dev)->short_address[start]), value,
value_len);
} while (0);
break;
case NETDEV_OPT_NID:
if ((res = _type_pun_up_unsigned(set_value, sizeof(unsigned int),
value, value_len)) == 0) {
_NETDEV_MORE(dev)->nid = *((unsigned int *)value);
}
break;
case NETDEV_OPT_ADDRESS_LONG:
if (value_len > UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN) {
return -EOVERFLOW;
}
do {
int start = UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN - value_len;
if (value_len == 0) {
return -EINVAL;
}
if (value_len < UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN) {
memcpy(_NETDEV_MORE(dev)->long_address, value, start);
}
memcpy(&(_NETDEV_MORE(dev)->long_address[start]), value,
value_len);
} while (0);
break;
case NETDEV_OPT_TX_POWER:
if ((res = _type_pun_up_signed(set_value, sizeof(int),
value, value_len)) == 0) {
_NETDEV_MORE(dev)->tx_power = *((int *)value);
}
break;
default:
return -ENOTSUP;
}
return res;
}
static int _get_state(netdev_t *dev, netdev_state_t *state)
{
if (_find_dev(dev) < 0) {
return -ENODEV;
}
if (state == NULL) {
return -EFAULT;
}
*state = _NETDEV_MORE(dev)->state;
return 0;
}
static int _set_state(netdev_t *dev, netdev_state_t state)
{
if (_find_dev(dev) < 0) {
return -ENODEV;
}
switch (state) {
case NETDEV_STATE_POWER_OFF:
case NETDEV_STATE_POWER_SLEEP:
case NETDEV_STATE_POWER_IDLE:
case NETDEV_STATE_RX_MODE:
case NETDEV_STATE_PROMISCUOUS_MODE:
_NETDEV_MORE(dev)->state = state;
return 0;
default:
return -ENOTSUP;
}
}
static void _event(netdev_t *dev, uint32_t event_type)
{
if (_find_dev(dev) < 0) {
return;
}
_NETDEV_MORE(dev)->last_event = event_type;
}
const netdev_driver_t unittest_netdev_dummy_driver = {
_init,
_send_data,
_add_receive_data_callback,
_rem_receive_data_callback,
_get_option,
_set_option,
_get_state,
_set_state,
_event,
};
int unittest_netdev_dummy_fire_rcv_event(netdev_t *dev, void *src,
size_t src_len, void *dest, size_t dest_len, void *data,
size_t data_len)
{
if (_find_dev(dev) < 0) {
return -ENODEV;
}
if ((src_len != UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN &&
src_len != UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN) ||
(dest_len != UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN &&
dest_len != UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN)) {
return -EAFNOSUPPORT;
}
if (data_len > UNITTESTS_NETDEV_DUMMY_MAX_PACKET) {
return -EMSGSIZE;
}
if (src == NULL || dest == NULL || (data == NULL && data_len != 0)) {
return -EINVAL;
}
_NETDEV_MORE(dev)->rx_buffer.src_len = src_len;
_NETDEV_MORE(dev)->rx_buffer.dst_len = dest_len;
_NETDEV_MORE(dev)->rx_buffer.data_len = data_len;
memcpy(_NETDEV_MORE(dev)->rx_buffer.dst, dest, dest_len);
memcpy(_NETDEV_MORE(dev)->rx_buffer.src, src, src_len);
memcpy(_NETDEV_MORE(dev)->rx_buffer.data, data, data_len);
for (int j = 0; j < UNITTESTS_NETDEV_DUMMY_MAX_CB; j++) {
if (_NETDEV_MORE(dev)->callbacks[j] != 0) {
int res = _NETDEV_MORE(dev)->callbacks[j](dev,
_NETDEV_MORE(dev)->rx_buffer.src,
_NETDEV_MORE(dev)->rx_buffer.src_len,
_NETDEV_MORE(dev)->rx_buffer.dst,
_NETDEV_MORE(dev)->rx_buffer.dst_len,
_NETDEV_MORE(dev)->rx_buffer.data,
_NETDEV_MORE(dev)->rx_buffer.data_len);
if (res < 0) {
return -ECANCELED;
}
}
}
return 0;
}
int unittest_netdev_dummy_check_transmitted(netdev_t *dev,
void *expected_dest,
size_t expected_dest_len,
void *expected_data,
size_t expected_data_len)
{
if (_find_dev(dev) < 0) {
return -ENODEV;
}
if ((expected_dest == NULL && expected_dest_len != 0) ||
(expected_data == NULL && expected_data_len != 0)) {
return -EINVAL;
}
if (!(expected_dest_len == UNITTESTS_NETDEV_DUMMY_MAX_LONG_ADDR_LEN ||
expected_dest_len == UNITTESTS_NETDEV_DUMMY_MAX_ADDR_LEN) ||
expected_dest_len != _NETDEV_MORE(dev)->tx_buffer.dst_len ||
expected_data_len != _NETDEV_MORE(dev)->tx_buffer.data_len ||
(memcmp(expected_dest, _NETDEV_MORE(dev)->tx_buffer.dst, expected_dest_len) != 0) ||
(memcmp(expected_data, _NETDEV_MORE(dev)->tx_buffer.data, expected_data_len) != 0)) {
return 1;
}
return 0;
}
uint32_t unittest_netdev_dummy_get_last_event(netdev_t *dev)
{
if (_find_dev(dev) < 0) {
return ENODEV;
}
return _NETDEV_MORE(dev)->last_event;
}
void unittest_netdev_dummy_init(void)
{
for (int i = 0; i < UNITTESTS_NETDEV_DUMMY_MAX_DEV; i++) {
netdev_t *dev = &(unittest_netdev_dummy_devs[i]);
dev->type = NETDEV_TYPE_BASE;
dev->driver = &unittest_netdev_dummy_driver;
dev->more = &(_netdevs_internal[i]);
}
}
/**
* @}
*/