2015-08-21 16:06:57 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @{
|
|
|
|
* @ingroup net
|
|
|
|
* @file
|
2017-02-15 13:07:34 +01:00
|
|
|
* @brief gnrc netdev ethernet glue code
|
2015-08-21 16:06:57 +02:00
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "net/gnrc.h"
|
2017-02-15 13:07:34 +01:00
|
|
|
#include "net/gnrc/netdev.h"
|
2015-08-21 16:06:57 +02:00
|
|
|
#include "net/ethernet/hdr.h"
|
|
|
|
|
2016-02-05 21:50:53 +01:00
|
|
|
#ifdef MODULE_GNRC_IPV6
|
|
|
|
#include "net/ipv6/hdr.h"
|
|
|
|
#endif
|
|
|
|
|
2015-08-21 16:06:57 +02:00
|
|
|
#include "od.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static gnrc_pktsnip_t *_recv(gnrc_netdev_t *gnrc_netdev)
|
2015-08-21 16:06:57 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_t *dev = gnrc_netdev->dev;
|
2016-01-29 15:48:10 +01:00
|
|
|
int bytes_expected = dev->driver->recv(dev, NULL, 0, NULL);
|
2015-08-21 16:06:57 +02:00
|
|
|
gnrc_pktsnip_t *pkt = NULL;
|
|
|
|
|
2016-12-14 14:40:11 +01:00
|
|
|
if (bytes_expected > 0) {
|
2015-08-21 16:06:57 +02:00
|
|
|
pkt = gnrc_pktbuf_add(NULL, NULL,
|
|
|
|
bytes_expected,
|
|
|
|
GNRC_NETTYPE_UNDEF);
|
|
|
|
|
|
|
|
if(!pkt) {
|
|
|
|
DEBUG("_recv_ethernet_packet: cannot allocate pktsnip.\n");
|
2015-12-24 06:24:57 +01:00
|
|
|
|
|
|
|
/* drop the packet */
|
|
|
|
dev->driver->recv(dev, NULL, bytes_expected, NULL);
|
|
|
|
|
2015-08-21 16:06:57 +02:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2016-01-29 15:48:10 +01:00
|
|
|
int nread = dev->driver->recv(dev, pkt->data, bytes_expected, NULL);
|
2015-08-21 16:06:57 +02:00
|
|
|
if(nread <= 0) {
|
|
|
|
DEBUG("_recv_ethernet_packet: read error.\n");
|
|
|
|
goto safe_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nread < bytes_expected) {
|
|
|
|
/* we've got less then the expected packet size,
|
|
|
|
* so free the unused space.*/
|
|
|
|
|
|
|
|
DEBUG("_recv_ethernet_packet: reallocating.\n");
|
|
|
|
gnrc_pktbuf_realloc_data(pkt, nread);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* mark ethernet header */
|
|
|
|
gnrc_pktsnip_t *eth_hdr = gnrc_pktbuf_mark(pkt, sizeof(ethernet_hdr_t), GNRC_NETTYPE_UNDEF);
|
|
|
|
if (!eth_hdr) {
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("gnrc_netdev_eth: no space left in packet buffer\n");
|
2015-08-21 16:06:57 +02:00
|
|
|
goto safe_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ethernet_hdr_t *hdr = (ethernet_hdr_t *)eth_hdr->data;
|
|
|
|
|
|
|
|
/* set payload type from ethertype */
|
|
|
|
pkt->type = gnrc_nettype_from_ethertype(byteorder_ntohs(hdr->type));
|
|
|
|
|
|
|
|
/* create netif header */
|
|
|
|
gnrc_pktsnip_t *netif_hdr;
|
|
|
|
netif_hdr = gnrc_pktbuf_add(NULL, NULL,
|
|
|
|
sizeof(gnrc_netif_hdr_t) + (2 * ETHERNET_ADDR_LEN),
|
|
|
|
GNRC_NETTYPE_NETIF);
|
|
|
|
|
|
|
|
if (netif_hdr == NULL) {
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("gnrc_netdev_eth: no space left in packet buffer\n");
|
2015-08-21 16:06:57 +02:00
|
|
|
pkt = eth_hdr;
|
|
|
|
goto safe_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
gnrc_netif_hdr_init(netif_hdr->data, ETHERNET_ADDR_LEN, ETHERNET_ADDR_LEN);
|
|
|
|
gnrc_netif_hdr_set_src_addr(netif_hdr->data, hdr->src, ETHERNET_ADDR_LEN);
|
|
|
|
gnrc_netif_hdr_set_dst_addr(netif_hdr->data, hdr->dst, ETHERNET_ADDR_LEN);
|
|
|
|
((gnrc_netif_hdr_t *)netif_hdr->data)->if_pid = thread_getpid();
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("gnrc_netdev_eth: received packet from %02x:%02x:%02x:%02x:%02x:%02x "
|
2015-11-16 03:43:59 +01:00
|
|
|
"of length %d\n",
|
2015-08-21 16:06:57 +02:00
|
|
|
hdr->src[0], hdr->src[1], hdr->src[2], hdr->src[3], hdr->src[4],
|
|
|
|
hdr->src[5], nread);
|
|
|
|
#if defined(MODULE_OD) && ENABLE_DEBUG
|
|
|
|
od_hex_dump(hdr, nread, OD_WIDTH_DEFAULT);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
gnrc_pktbuf_remove_snip(pkt, eth_hdr);
|
|
|
|
LL_APPEND(pkt, netif_hdr);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return pkt;
|
|
|
|
|
|
|
|
safe_out:
|
|
|
|
gnrc_pktbuf_release(pkt);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _addr_set_broadcast(uint8_t *dst)
|
|
|
|
{
|
|
|
|
memset(dst, 0xff, ETHERNET_ADDR_LEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _addr_set_multicast(uint8_t *dst, gnrc_pktsnip_t *payload)
|
|
|
|
{
|
|
|
|
switch (payload->type) {
|
2016-02-05 21:50:53 +01:00
|
|
|
#ifdef MODULE_GNRC_IPV6
|
2015-08-21 16:06:57 +02:00
|
|
|
case GNRC_NETTYPE_IPV6:
|
2016-02-05 21:50:53 +01:00
|
|
|
/* https://tools.ietf.org/html/rfc2464#section-7 */
|
2015-08-21 16:06:57 +02:00
|
|
|
dst[0] = 0x33;
|
|
|
|
dst[1] = 0x33;
|
2016-02-05 21:50:53 +01:00
|
|
|
ipv6_hdr_t *ipv6 = payload->data;
|
|
|
|
memcpy(dst + 2, ipv6->dst.u8 + 12, 4);
|
2015-08-21 16:06:57 +02:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
_addr_set_broadcast(dst);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
static int _send(gnrc_netdev_t *gnrc_netdev, gnrc_pktsnip_t *pkt)
|
2015-08-21 16:06:57 +02:00
|
|
|
{
|
|
|
|
ethernet_hdr_t hdr;
|
|
|
|
gnrc_netif_hdr_t *netif_hdr;
|
|
|
|
gnrc_pktsnip_t *payload;
|
2016-03-07 21:36:35 +01:00
|
|
|
int res;
|
2015-08-21 16:06:57 +02:00
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
netdev_t *dev = gnrc_netdev->dev;
|
2015-08-21 16:06:57 +02:00
|
|
|
|
|
|
|
if (pkt == NULL) {
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("gnrc_netdev_eth: pkt was NULL\n");
|
2015-08-21 16:06:57 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
payload = pkt->next;
|
|
|
|
|
|
|
|
if (pkt->type != GNRC_NETTYPE_NETIF) {
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("gnrc_netdev_eth: First header was not generic netif header\n");
|
2015-08-21 16:06:57 +02:00
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (payload) {
|
|
|
|
hdr.type = byteorder_htons(gnrc_nettype_to_ethertype(payload->type));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
hdr.type = byteorder_htons(ETHERTYPE_UNKNOWN);
|
|
|
|
}
|
|
|
|
|
|
|
|
netif_hdr = pkt->data;
|
|
|
|
|
|
|
|
/* set ethernet header */
|
|
|
|
if (netif_hdr->src_l2addr_len == ETHERNET_ADDR_LEN) {
|
|
|
|
memcpy(hdr.dst, gnrc_netif_hdr_get_src_addr(netif_hdr),
|
|
|
|
netif_hdr->src_l2addr_len);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dev->driver->get(dev, NETOPT_ADDRESS, hdr.src, ETHERNET_ADDR_LEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (netif_hdr->flags & GNRC_NETIF_HDR_FLAGS_BROADCAST) {
|
|
|
|
_addr_set_broadcast(hdr.dst);
|
|
|
|
}
|
|
|
|
else if (netif_hdr->flags & GNRC_NETIF_HDR_FLAGS_MULTICAST) {
|
2015-09-11 21:30:06 +02:00
|
|
|
if (payload == NULL) {
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("gnrc_netdev_eth: empty multicast packets over Ethernet "\
|
2015-09-11 21:30:06 +02:00
|
|
|
"are not yet supported\n");
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
2015-08-21 16:06:57 +02:00
|
|
|
_addr_set_multicast(hdr.dst, payload);
|
|
|
|
}
|
|
|
|
else if (netif_hdr->dst_l2addr_len == ETHERNET_ADDR_LEN) {
|
|
|
|
memcpy(hdr.dst, gnrc_netif_hdr_get_dst_addr(netif_hdr),
|
|
|
|
ETHERNET_ADDR_LEN);
|
|
|
|
}
|
|
|
|
else {
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("gnrc_netdev_eth: destination address had unexpected format\n");
|
2015-08-21 16:06:57 +02:00
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
DEBUG("gnrc_netdev_eth: send to %02x:%02x:%02x:%02x:%02x:%02x\n",
|
2015-08-21 16:06:57 +02:00
|
|
|
hdr.dst[0], hdr.dst[1], hdr.dst[2],
|
|
|
|
hdr.dst[3], hdr.dst[4], hdr.dst[5]);
|
|
|
|
|
|
|
|
size_t n;
|
2016-03-07 21:36:35 +01:00
|
|
|
payload = gnrc_pktbuf_get_iovec(pkt, &n); /* use payload as temporary
|
|
|
|
* variable */
|
|
|
|
res = -ENOBUFS;
|
|
|
|
if (payload != NULL) {
|
|
|
|
pkt = payload; /* reassign for later release; vec_snip is prepended to pkt */
|
|
|
|
struct iovec *vector = (struct iovec *)pkt->data;
|
|
|
|
vector[0].iov_base = (char*)&hdr;
|
|
|
|
vector[0].iov_len = sizeof(ethernet_hdr_t);
|
2016-02-11 23:06:07 +01:00
|
|
|
#ifdef MODULE_NETSTATS_L2
|
|
|
|
if ((netif_hdr->flags & GNRC_NETIF_HDR_FLAGS_BROADCAST) ||
|
|
|
|
(netif_hdr->flags & GNRC_NETIF_HDR_FLAGS_MULTICAST)) {
|
2017-02-15 13:07:34 +01:00
|
|
|
gnrc_netdev->dev->stats.tx_mcast_count++;
|
2016-02-11 23:06:07 +01:00
|
|
|
}
|
|
|
|
else {
|
2017-02-15 13:07:34 +01:00
|
|
|
gnrc_netdev->dev->stats.tx_unicast_count++;
|
2016-02-11 23:06:07 +01:00
|
|
|
}
|
|
|
|
#endif
|
2016-03-07 21:36:35 +01:00
|
|
|
res = dev->driver->send(dev, vector, n);
|
|
|
|
}
|
2015-08-21 16:06:57 +02:00
|
|
|
|
|
|
|
gnrc_pktbuf_release(pkt);
|
|
|
|
|
2016-03-07 21:36:35 +01:00
|
|
|
return res;
|
2015-08-21 16:06:57 +02:00
|
|
|
}
|
|
|
|
|
2017-02-15 13:07:34 +01:00
|
|
|
int gnrc_netdev_eth_init(gnrc_netdev_t *gnrc_netdev, netdev_t *dev)
|
2015-08-21 16:06:57 +02:00
|
|
|
{
|
2017-02-15 13:07:34 +01:00
|
|
|
gnrc_netdev->send = _send;
|
|
|
|
gnrc_netdev->recv = _recv;
|
|
|
|
gnrc_netdev->dev = dev;
|
2015-08-21 16:06:57 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|