2016-08-17 16:48:36 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2016 Intel Corporation.
|
|
|
|
*
|
2017-01-19 02:01:01 +01:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2016-08-17 16:48:36 +02:00
|
|
|
*/
|
|
|
|
|
2017-01-10 14:24:01 +01:00
|
|
|
#define SYS_LOG_LEVEL CONFIG_SYS_LOG_IEEE802154_DRIVER_LEVEL
|
2016-08-17 16:48:36 +02:00
|
|
|
#define SYS_LOG_DOMAIN "net/ieee802154/upipe/"
|
2016-12-17 18:56:56 +01:00
|
|
|
#include <logging/sys_log.h>
|
2016-08-17 16:48:36 +02:00
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
2016-11-09 16:53:16 +01:00
|
|
|
#include <kernel.h>
|
2016-08-17 16:48:36 +02:00
|
|
|
#include <arch/cpu.h>
|
|
|
|
|
|
|
|
#include <board.h>
|
|
|
|
#include <device.h>
|
|
|
|
#include <init.h>
|
|
|
|
#include <net/net_if.h>
|
|
|
|
#include <net/nbuf.h>
|
|
|
|
|
|
|
|
#include <console/uart_pipe.h>
|
|
|
|
#include <net/ieee802154_radio.h>
|
|
|
|
|
|
|
|
#include "ieee802154_uart_pipe.h"
|
|
|
|
|
|
|
|
/** Singleton device used in uart pipe callback */
|
|
|
|
static struct device *upipe_dev;
|
|
|
|
|
|
|
|
static uint8_t *upipe_rx(uint8_t *buf, size_t *off)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = upipe_dev->driver_data;
|
|
|
|
struct net_buf *pkt_buf = NULL;
|
|
|
|
struct net_buf *nbuf = NULL;
|
|
|
|
|
|
|
|
if (!upipe_dev) {
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!upipe->rx && *buf == UART_PIPE_RADIO_15_4_FRAME_TYPE) {
|
|
|
|
upipe->rx = true;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!upipe->rx_len) {
|
|
|
|
if (*buf > 127) {
|
|
|
|
goto flush;
|
|
|
|
}
|
|
|
|
|
|
|
|
upipe->rx_len = *buf;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
upipe->rx_buf[upipe->rx_off++] = *buf;
|
|
|
|
|
|
|
|
if (upipe->rx_len == upipe->rx_off) {
|
2017-02-03 14:56:15 +01:00
|
|
|
nbuf = net_nbuf_get_reserve_rx(0, K_NO_WAIT);
|
2016-08-17 16:48:36 +02:00
|
|
|
if (!nbuf) {
|
|
|
|
SYS_LOG_DBG("No buf available");
|
|
|
|
goto flush;
|
|
|
|
}
|
|
|
|
|
2017-02-24 12:32:56 +01:00
|
|
|
pkt_buf = net_nbuf_get_frag(nbuf, K_NO_WAIT);
|
2016-08-17 16:48:36 +02:00
|
|
|
if (!pkt_buf) {
|
|
|
|
SYS_LOG_DBG("No fragment available");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
net_buf_frag_insert(nbuf, pkt_buf);
|
|
|
|
|
2016-11-15 15:30:23 +01:00
|
|
|
memcpy(pkt_buf->data, upipe->rx_buf, upipe->rx_len - 2);
|
|
|
|
net_buf_add(pkt_buf, upipe->rx_len - 2);
|
2016-08-17 16:48:36 +02:00
|
|
|
|
|
|
|
if (ieee802154_radio_handle_ack(upipe->iface, nbuf) == NET_OK) {
|
|
|
|
SYS_LOG_DBG("ACK packet handled");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2016-11-03 15:45:26 +01:00
|
|
|
SYS_LOG_DBG("Caught a packet (%u)", upipe->rx_len);
|
2016-08-17 16:48:36 +02:00
|
|
|
if (net_recv_data(upipe->iface, nbuf) < 0) {
|
|
|
|
SYS_LOG_DBG("Packet dropped by NET stack");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
goto flush;
|
|
|
|
out:
|
2017-02-20 16:37:21 +01:00
|
|
|
net_nbuf_unref(nbuf);
|
2016-08-17 16:48:36 +02:00
|
|
|
flush:
|
|
|
|
upipe->rx = false;
|
|
|
|
upipe->rx_len = 0;
|
|
|
|
upipe->rx_off = 0;
|
|
|
|
}
|
|
|
|
done:
|
|
|
|
*off = 0;
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_cca(struct device *dev)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
if (upipe->stopped) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_set_channel(struct device *dev, uint16_t channel)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
if (upipe->stopped) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_set_pan_id(struct device *dev, uint16_t pan_id)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
if (upipe->stopped) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_set_short_addr(struct device *dev, uint16_t short_addr)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
if (upipe->stopped) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_set_ieee_addr(struct device *dev, const uint8_t *ieee_addr)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
if (upipe->stopped) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_set_txpower(struct device *dev, int16_t dbm)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
if (upipe->stopped) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
net/ieee802154: Modify radio TX function signature
The cause for this change is TCP. Until now, the radio strategy driver
(ALOHA or CSMA) was providing the actual nbuf, and not the buffer
fragment, counting on the fact that the loop was using
net_buf_frag_del() which made so, iteration after iteration, buffer
framgent to be always buf->frags. The problem with this logic is loosing
the fragments that might be still referenced by TCP, in case the whole
buffer did not make it so TCP can retry later and so on.
Instead, TX now takes the nbuf and the actual frag to send. It could
have been working with just a pointer on the data, and the whole length
of the frame. But it has been avoided due to possible future devices,
that will be smarter and run CSMA directly in the hw, thus it will
require to access the whole buffer list through the nbuf.
Change-Id: I8d77b1e13b648c0ec3645cb2d55d1910d00381ea
Signed-off-by: Tomasz Bursztyka <tomasz.bursztyka@linux.intel.com>
2017-01-26 14:31:30 +01:00
|
|
|
static int upipe_tx(struct device *dev,
|
|
|
|
struct net_buf *buf,
|
|
|
|
struct net_buf *frag)
|
2016-08-17 16:48:36 +02:00
|
|
|
{
|
net/ieee802154: Modify radio TX function signature
The cause for this change is TCP. Until now, the radio strategy driver
(ALOHA or CSMA) was providing the actual nbuf, and not the buffer
fragment, counting on the fact that the loop was using
net_buf_frag_del() which made so, iteration after iteration, buffer
framgent to be always buf->frags. The problem with this logic is loosing
the fragments that might be still referenced by TCP, in case the whole
buffer did not make it so TCP can retry later and so on.
Instead, TX now takes the nbuf and the actual frag to send. It could
have been working with just a pointer on the data, and the whole length
of the frame. But it has been avoided due to possible future devices,
that will be smarter and run CSMA directly in the hw, thus it will
require to access the whole buffer list through the nbuf.
Change-Id: I8d77b1e13b648c0ec3645cb2d55d1910d00381ea
Signed-off-by: Tomasz Bursztyka <tomasz.bursztyka@linux.intel.com>
2017-01-26 14:31:30 +01:00
|
|
|
uint8_t *pkt_buf = frag->data - net_nbuf_ll_reserve(buf);
|
|
|
|
uint8_t len = net_nbuf_ll_reserve(buf) + frag->len;
|
2016-08-17 16:48:36 +02:00
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
uint8_t i, data;
|
|
|
|
|
net/ieee802154: Modify radio TX function signature
The cause for this change is TCP. Until now, the radio strategy driver
(ALOHA or CSMA) was providing the actual nbuf, and not the buffer
fragment, counting on the fact that the loop was using
net_buf_frag_del() which made so, iteration after iteration, buffer
framgent to be always buf->frags. The problem with this logic is loosing
the fragments that might be still referenced by TCP, in case the whole
buffer did not make it so TCP can retry later and so on.
Instead, TX now takes the nbuf and the actual frag to send. It could
have been working with just a pointer on the data, and the whole length
of the frame. But it has been avoided due to possible future devices,
that will be smarter and run CSMA directly in the hw, thus it will
require to access the whole buffer list through the nbuf.
Change-Id: I8d77b1e13b648c0ec3645cb2d55d1910d00381ea
Signed-off-by: Tomasz Bursztyka <tomasz.bursztyka@linux.intel.com>
2017-01-26 14:31:30 +01:00
|
|
|
SYS_LOG_DBG("%p (%u)", frag, len);
|
2016-08-17 16:48:36 +02:00
|
|
|
|
|
|
|
if (upipe->stopped) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = UART_PIPE_RADIO_15_4_FRAME_TYPE;
|
|
|
|
uart_pipe_send(&data, 1);
|
|
|
|
|
|
|
|
data = len + 2;
|
|
|
|
uart_pipe_send(&data, 1);
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
uart_pipe_send(pkt_buf+i, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The 2 dummy bytes representing FCS */
|
|
|
|
data = 0xFF;
|
|
|
|
uart_pipe_send(&data, 1);
|
|
|
|
uart_pipe_send(&data, 1);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_start(struct device *dev)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
if (!upipe->stopped) {
|
|
|
|
return -EALREADY;
|
|
|
|
}
|
|
|
|
|
|
|
|
upipe->stopped = false;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_stop(struct device *dev)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
if (upipe->stopped) {
|
|
|
|
return -EALREADY;
|
|
|
|
}
|
|
|
|
|
|
|
|
upipe->stopped = true;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int upipe_init(struct device *dev)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
memset(upipe, 0, sizeof(struct upipe_context));
|
|
|
|
|
|
|
|
uart_pipe_register(upipe->uart_pipe_buf, 1, upipe_rx);
|
|
|
|
|
|
|
|
upipe_stop(dev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint8_t *get_mac(struct device *dev)
|
|
|
|
{
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
|
|
|
|
upipe->mac_addr[0] = 0x00;
|
|
|
|
upipe->mac_addr[1] = 0x10;
|
|
|
|
upipe->mac_addr[2] = 0x20;
|
|
|
|
upipe->mac_addr[3] = 0x30;
|
|
|
|
|
|
|
|
UNALIGNED_PUT(sys_cpu_to_be32(sys_rand32_get()),
|
|
|
|
(uint32_t *) ((void *)upipe->mac_addr+4));
|
|
|
|
|
|
|
|
return upipe->mac_addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void upipe_iface_init(struct net_if *iface)
|
|
|
|
{
|
|
|
|
struct device *dev = net_if_get_device(iface);
|
|
|
|
struct upipe_context *upipe = dev->driver_data;
|
|
|
|
uint8_t *mac = get_mac(dev);
|
|
|
|
|
|
|
|
SYS_LOG_DBG("");
|
|
|
|
|
2017-02-15 12:20:31 +01:00
|
|
|
net_if_set_link_addr(iface, mac, 8, NET_LINK_IEEE802154);
|
2016-08-17 16:48:36 +02:00
|
|
|
|
|
|
|
upipe_dev = dev;
|
|
|
|
upipe->iface = iface;
|
|
|
|
|
|
|
|
ieee802154_init(iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct upipe_context upipe_context_data;
|
|
|
|
|
|
|
|
static struct ieee802154_radio_api upipe_radio_api = {
|
|
|
|
.iface_api.init = upipe_iface_init,
|
|
|
|
.iface_api.send = ieee802154_radio_send,
|
|
|
|
|
|
|
|
.cca = upipe_cca,
|
|
|
|
.set_channel = upipe_set_channel,
|
|
|
|
.set_pan_id = upipe_set_pan_id,
|
|
|
|
.set_short_addr = upipe_set_short_addr,
|
|
|
|
.set_ieee_addr = upipe_set_ieee_addr,
|
|
|
|
.set_txpower = upipe_set_txpower,
|
|
|
|
.tx = upipe_tx,
|
|
|
|
.start = upipe_start,
|
|
|
|
.stop = upipe_stop,
|
|
|
|
};
|
|
|
|
|
2017-02-06 11:49:16 +01:00
|
|
|
NET_DEVICE_INIT(upipe_15_4, CONFIG_IEEE802154_UPIPE_DRV_NAME,
|
2016-08-17 16:48:36 +02:00
|
|
|
upipe_init, &upipe_context_data, NULL,
|
|
|
|
CONFIG_KERNEL_INIT_PRIORITY_DEFAULT,
|
|
|
|
&upipe_radio_api, IEEE802154_L2,
|
2016-11-16 11:48:40 +01:00
|
|
|
NET_L2_GET_CTX_TYPE(IEEE802154_L2), 125);
|