2016-02-16 17:17:58 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* @author Martine Lenders <mlenders@inf.fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
#include "net/netdev_test.h"
|
2016-02-16 17:17:58 +01:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _send(netdev_t *netdev, const struct iovec *vector, unsigned count);
|
|
|
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info);
|
|
|
|
static int _init(netdev_t *dev);
|
|
|
|
static void _isr(netdev_t *dev);
|
|
|
|
static int _get(netdev_t *dev, netopt_t opt, void *value, size_t max_len);
|
2017-08-25 07:34:03 +02:00
|
|
|
static int _set(netdev_t *dev, netopt_t opt, const void *value, size_t value_len);
|
2016-02-16 17:17:58 +01:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static const netdev_driver_t _driver = {
|
2016-02-16 17:17:58 +01:00
|
|
|
.send = _send,
|
|
|
|
.recv = _recv,
|
|
|
|
.init = _init,
|
|
|
|
.isr = _isr,
|
|
|
|
.get = _get,
|
|
|
|
.set = _set,
|
|
|
|
};
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
void netdev_test_setup(netdev_test_t *dev, void *state)
|
2016-02-16 17:17:58 +01:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_t *netdev = (netdev_t *)dev;
|
2016-02-16 17:17:58 +01:00
|
|
|
|
|
|
|
netdev->driver = &_driver;
|
|
|
|
dev->state = state;
|
|
|
|
mutex_init(&dev->mutex);
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_test_reset(dev);
|
2016-02-16 17:17:58 +01:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
void netdev_test_reset(netdev_test_t *dev)
|
2016-02-16 17:17:58 +01:00
|
|
|
{
|
|
|
|
mutex_lock(&dev->mutex);
|
|
|
|
dev->send_cb = NULL;
|
|
|
|
dev->recv_cb = NULL;
|
|
|
|
dev->init_cb = NULL;
|
|
|
|
dev->isr_cb = NULL;
|
|
|
|
memset(dev->get_cbs, 0, sizeof(dev->get_cbs));
|
|
|
|
memset(dev->set_cbs, 0, sizeof(dev->set_cbs));
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _send(netdev_t *netdev, const struct iovec *vector, unsigned count)
|
2016-02-16 17:17:58 +01:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_test_t *dev = (netdev_test_t *)netdev;
|
2016-08-02 16:56:15 +02:00
|
|
|
int res = (int)count; /* assume everything would be fine */
|
2016-02-16 17:17:58 +01:00
|
|
|
|
|
|
|
mutex_lock(&dev->mutex);
|
|
|
|
if (dev->send_cb != NULL) {
|
|
|
|
res = dev->send_cb(netdev, vector, count);
|
|
|
|
}
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _recv(netdev_t *netdev, void *buf, size_t len, void *info)
|
2016-02-16 17:17:58 +01:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_test_t *dev = (netdev_test_t *)netdev;
|
2016-02-16 17:17:58 +01:00
|
|
|
int res = (buf == NULL) ? 0 : len; /* assume everything would be fine */
|
|
|
|
|
|
|
|
mutex_lock(&dev->mutex);
|
|
|
|
if (dev->recv_cb != NULL) {
|
|
|
|
/* could fire context change and call _recv so we need to unlock */
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
res = dev->recv_cb(netdev, buf, len, info);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _init(netdev_t *netdev)
|
2016-02-16 17:17:58 +01:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_test_t *dev = (netdev_test_t *)netdev;
|
2016-02-16 17:17:58 +01:00
|
|
|
int res = 0; /* assume everything would be fine */
|
|
|
|
|
|
|
|
mutex_lock(&dev->mutex);
|
|
|
|
if (dev->init_cb != NULL) {
|
|
|
|
res = dev->init_cb(netdev);
|
|
|
|
}
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static void _isr(netdev_t *netdev)
|
2016-02-16 17:17:58 +01:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_test_t *dev = (netdev_test_t *)netdev;
|
2016-02-16 17:17:58 +01:00
|
|
|
|
|
|
|
mutex_lock(&dev->mutex);
|
|
|
|
if (dev->isr_cb != NULL) {
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
dev->isr_cb(netdev);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _get(netdev_t *netdev, netopt_t opt, void *value, size_t max_len)
|
2016-02-16 17:17:58 +01:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_test_t *dev = (netdev_test_t *)netdev;
|
2016-02-16 17:17:58 +01:00
|
|
|
int res = -ENOTSUP; /* option assumed to be not supported */
|
|
|
|
|
|
|
|
mutex_lock(&dev->mutex);
|
|
|
|
if (dev->get_cbs[opt] != NULL) {
|
|
|
|
res = dev->get_cbs[opt](netdev, value, max_len);
|
|
|
|
}
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-08-25 07:34:03 +02:00
|
|
|
static int _set(netdev_t *netdev, netopt_t opt, const void *value, size_t value_len)
|
2016-02-16 17:17:58 +01:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_test_t *dev = (netdev_test_t *)netdev;
|
2016-02-16 17:17:58 +01:00
|
|
|
int res = -ENOTSUP; /* option assumed to be not supported */
|
|
|
|
|
|
|
|
mutex_lock(&dev->mutex);
|
|
|
|
if (dev->set_cbs[opt] != NULL) {
|
|
|
|
res = dev->set_cbs[opt](netdev, value, value_len);
|
|
|
|
}
|
|
|
|
mutex_unlock(&dev->mutex);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** @} */
|