2013-08-29 14:41:55 +02:00
|
|
|
/**
|
|
|
|
* tap.h implementation
|
|
|
|
*
|
2014-05-15 18:07:02 +02:00
|
|
|
* Copyright (C) 2013 Ludwig Ortmann <ludwig.ortmann@fu-berlin.de>
|
2013-08-29 14:41:55 +02:00
|
|
|
*
|
2014-07-31 19:45:27 +02:00
|
|
|
* 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.
|
2013-08-29 14:41:55 +02:00
|
|
|
*
|
|
|
|
* @ingroup native_cpu
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @author Ludwig Ortmann <ludwig.ortmann@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
2013-06-26 23:29:09 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <err.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <arpa/inet.h>
|
2013-08-21 19:22:32 +02:00
|
|
|
#include <inttypes.h>
|
2013-11-08 13:16:08 +01:00
|
|
|
#include <errno.h>
|
2013-06-26 23:29:09 +02:00
|
|
|
|
|
|
|
#ifdef __MACH__
|
|
|
|
#define _POSIX_C_SOURCE
|
|
|
|
#include <net/if.h>
|
|
|
|
#undef _POSIX_C_SOURCE
|
|
|
|
#include <ifaddrs.h>
|
|
|
|
#include <net/if_dl.h>
|
2014-04-10 14:58:12 +02:00
|
|
|
|
|
|
|
#elif defined(__FreeBSD__)
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <net/if.h>
|
|
|
|
#include <ifaddrs.h>
|
|
|
|
#include <net/if_dl.h>
|
|
|
|
|
2013-06-26 23:29:09 +02:00
|
|
|
#else
|
|
|
|
#include <net/if.h>
|
|
|
|
#include <linux/if_tun.h>
|
|
|
|
#include <linux/if_ether.h>
|
|
|
|
#endif
|
|
|
|
|
2013-08-08 11:08:33 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
2013-06-26 23:29:09 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "cpu-conf.h"
|
|
|
|
#include "tap.h"
|
2013-08-08 11:08:33 +02:00
|
|
|
#include "nativenet.h"
|
|
|
|
#include "nativenet_internal.h"
|
2013-11-07 17:23:08 +01:00
|
|
|
#include "native_internal.h"
|
2013-06-26 23:29:09 +02:00
|
|
|
|
2013-11-21 20:01:13 +01:00
|
|
|
#include "hwtimer.h"
|
|
|
|
#include "timex.h"
|
|
|
|
|
2013-08-08 11:08:33 +02:00
|
|
|
#define TAP_BUFFER_LENGTH (ETHER_MAX_LEN)
|
|
|
|
int _native_marshall_ethernet(uint8_t *framebuf, radio_packet_t *packet);
|
2013-06-26 23:29:09 +02:00
|
|
|
|
2014-07-13 16:23:27 +02:00
|
|
|
int _native_tap_fd = -1;
|
2013-06-26 23:29:09 +02:00
|
|
|
unsigned char _native_tap_mac[ETHER_ADDR_LEN];
|
|
|
|
|
2013-12-24 11:22:23 +01:00
|
|
|
#ifdef __MACH__
|
|
|
|
pid_t sigio_child_pid;
|
|
|
|
#endif
|
|
|
|
|
2013-08-08 11:08:33 +02:00
|
|
|
void _native_handle_tap_input(void)
|
2013-06-26 23:29:09 +02:00
|
|
|
{
|
|
|
|
int nread;
|
2013-08-21 19:22:32 +02:00
|
|
|
union eth_frame frame;
|
2013-08-08 11:08:33 +02:00
|
|
|
radio_packet_t p;
|
2013-06-26 23:29:09 +02:00
|
|
|
|
2013-08-08 11:08:33 +02:00
|
|
|
DEBUG("_native_handle_tap_input\n");
|
2013-06-26 23:29:09 +02:00
|
|
|
|
|
|
|
/* TODO: check whether this is an input or an output event
|
|
|
|
TODO: refactor this into general io-signal multiplexer */
|
|
|
|
|
2013-11-07 17:23:08 +01:00
|
|
|
nread = real_read(_native_tap_fd, &frame, sizeof(union eth_frame));
|
2013-08-08 11:08:33 +02:00
|
|
|
DEBUG("_native_handle_tap_input - read %d bytes\n", nread);
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-06-26 23:29:09 +02:00
|
|
|
if (nread > 0) {
|
2013-08-21 19:22:32 +02:00
|
|
|
if (ntohs(frame.field.header.ether_type) == NATIVE_ETH_PROTO) {
|
2013-06-26 23:29:09 +02:00
|
|
|
nread = nread - ETHER_HDR_LEN;
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-06-26 23:29:09 +02:00
|
|
|
if ((nread - 1) <= 0) {
|
2014-01-22 18:23:10 +01:00
|
|
|
DEBUG("_native_handle_tap_input: no payload\n");
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
|
|
|
else {
|
2013-11-21 20:01:13 +01:00
|
|
|
unsigned long t = hwtimer_now();
|
|
|
|
p.processing = 0;
|
2013-08-21 19:22:32 +02:00
|
|
|
p.src = ntohs(frame.field.payload.nn_header.src);
|
2013-11-21 20:01:13 +01:00
|
|
|
p.dst = ntohs(frame.field.payload.nn_header.dst);
|
2013-09-04 21:02:57 +02:00
|
|
|
p.rssi = 0;
|
|
|
|
p.lqi = 0;
|
2014-08-06 22:09:18 +02:00
|
|
|
p.toa.seconds = HWTIMER_TICKS_TO_US(t) / 1000000;
|
|
|
|
p.toa.microseconds = HWTIMER_TICKS_TO_US(t) % 1000000;
|
2013-11-21 20:01:13 +01:00
|
|
|
/* XXX: check overflow */
|
|
|
|
p.length = ntohs(frame.field.payload.nn_header.length);
|
2013-08-21 19:22:32 +02:00
|
|
|
p.data = frame.field.payload.data;
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2014-01-08 14:58:57 +01:00
|
|
|
if (p.length > (nread - sizeof(struct nativenet_header))) {
|
|
|
|
warnx("_native_handle_tap_input: packet with malicious length field received, discarding");
|
|
|
|
}
|
|
|
|
else {
|
2014-08-06 22:09:18 +02:00
|
|
|
DEBUG("_native_handle_tap_input: received packet of length %" PRIu16 " for %" PRIu16 " from %"
|
|
|
|
PRIu16 "\n", p.length, p.dst, p.src);
|
2014-01-08 14:58:57 +01:00
|
|
|
_nativenet_handle_packet(&p);
|
|
|
|
}
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DEBUG("ignoring non-native frame\n");
|
|
|
|
}
|
2013-11-08 13:16:08 +01:00
|
|
|
|
|
|
|
/* work around lost signals */
|
|
|
|
fd_set rfds;
|
|
|
|
struct timeval t;
|
|
|
|
memset(&t, 0, sizeof(t));
|
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_SET(_native_tap_fd, &rfds);
|
|
|
|
|
|
|
|
_native_in_syscall++; // no switching here
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2014-11-18 17:31:26 +01:00
|
|
|
if (real_select(_native_tap_fd + 1, &rfds, NULL, NULL, &t) == 1) {
|
2013-11-08 13:16:08 +01:00
|
|
|
int sig = SIGIO;
|
|
|
|
extern int _sig_pipefd[2];
|
2014-08-06 15:51:46 +02:00
|
|
|
extern ssize_t (*real_write)(int fd, const void *buf, size_t count);
|
2013-11-08 13:16:08 +01:00
|
|
|
real_write(_sig_pipefd[1], &sig, sizeof(int));
|
|
|
|
_native_sigpend++;
|
|
|
|
DEBUG("_native_handle_tap_input: sigpend++\n");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DEBUG("_native_handle_tap_input: no more pending tap data\n");
|
2013-12-24 11:22:23 +01:00
|
|
|
#ifdef __MACH__
|
|
|
|
kill(sigio_child_pid, SIGCONT);
|
|
|
|
#endif
|
2013-11-08 13:16:08 +01:00
|
|
|
}
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-11-08 13:16:08 +01:00
|
|
|
_native_in_syscall--;
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
|
|
|
else if (nread == -1) {
|
2014-08-06 22:09:18 +02:00
|
|
|
if ((errno == EAGAIN) || (errno == EWOULDBLOCK)) {
|
2013-11-08 13:16:08 +01:00
|
|
|
//warn("read");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
err(EXIT_FAILURE, "_native_handle_tap_input: read");
|
|
|
|
}
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
|
|
|
else {
|
2013-08-08 11:08:33 +02:00
|
|
|
errx(EXIT_FAILURE, "internal error _native_handle_tap_input");
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-24 11:22:23 +01:00
|
|
|
#ifdef __MACH__
|
2014-07-30 10:26:05 +02:00
|
|
|
void sigio_child(void)
|
2013-12-24 11:22:23 +01:00
|
|
|
{
|
2014-05-13 16:53:36 +02:00
|
|
|
pid_t parent = _native_pid;
|
2013-12-24 11:22:23 +01:00
|
|
|
|
2014-06-18 20:34:12 +02:00
|
|
|
if ((sigio_child_pid = real_fork()) == -1) {
|
2013-12-24 11:22:23 +01:00
|
|
|
err(EXIT_FAILURE, "sigio_child: fork");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sigio_child_pid > 0) {
|
|
|
|
/* return in parent process */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* watch tap interface and signal parent process if data is
|
|
|
|
* available */
|
|
|
|
|
|
|
|
fd_set rfds;
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-12-24 11:22:23 +01:00
|
|
|
while (1) {
|
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_SET(_native_tap_fd, &rfds);
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2014-11-18 17:31:26 +01:00
|
|
|
if (real_select(_native_tap_fd + 1, &rfds, NULL, NULL, NULL) == 1) {
|
2013-12-24 11:22:23 +01:00
|
|
|
kill(parent, SIGIO);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
kill(parent, SIGKILL);
|
|
|
|
err(EXIT_FAILURE, "osx_sigio_child: select");
|
|
|
|
}
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-12-24 11:22:23 +01:00
|
|
|
pause();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-08-08 11:08:33 +02:00
|
|
|
int _native_marshall_ethernet(uint8_t *framebuf, radio_packet_t *packet)
|
|
|
|
{
|
|
|
|
int data_len;
|
|
|
|
union eth_frame *f;
|
|
|
|
unsigned char addr[ETHER_ADDR_LEN];
|
|
|
|
|
2014-08-06 22:09:18 +02:00
|
|
|
f = (union eth_frame *)framebuf;
|
2013-09-30 15:31:40 +02:00
|
|
|
addr[0] = addr[1] = addr[2] = addr[3] = addr[4] = addr[5] = 0xFF;
|
2013-08-08 11:08:33 +02:00
|
|
|
|
|
|
|
memcpy(f->field.header.ether_dhost, addr, ETHER_ADDR_LEN);
|
|
|
|
memcpy(f->field.header.ether_shost, _native_tap_mac, ETHER_ADDR_LEN);
|
|
|
|
f->field.header.ether_type = htons(NATIVE_ETH_PROTO);
|
|
|
|
|
|
|
|
/* XXX: check overflow */
|
2013-08-21 19:22:32 +02:00
|
|
|
memcpy(f->field.payload.data, packet->data, packet->length);
|
|
|
|
f->field.payload.nn_header.length = htons(packet->length);
|
|
|
|
f->field.payload.nn_header.dst = htons(packet->dst);
|
|
|
|
f->field.payload.nn_header.src = htons(packet->src);
|
|
|
|
|
|
|
|
data_len = packet->length + sizeof(struct nativenet_header);
|
2013-08-08 11:08:33 +02:00
|
|
|
|
2013-09-16 17:48:58 +02:00
|
|
|
/* Pad to minimum payload size.
|
|
|
|
* Linux does this on its own, but it doesn't hurt to do it here.
|
|
|
|
* As of now only tuntaposx needs this. */
|
|
|
|
if (data_len < ETHERMIN) {
|
2014-08-06 15:51:46 +02:00
|
|
|
DEBUG("padding data! (%d -> ", data_len);
|
2013-09-16 17:48:58 +02:00
|
|
|
data_len = ETHERMIN;
|
|
|
|
DEBUG("%d)\n", data_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
return data_len + ETHER_HDR_LEN;
|
2013-08-08 11:08:33 +02:00
|
|
|
}
|
|
|
|
|
2013-12-21 16:56:42 +01:00
|
|
|
int8_t send_buf(radio_packet_t *packet)
|
2013-06-26 23:29:09 +02:00
|
|
|
{
|
|
|
|
uint8_t buf[TAP_BUFFER_LENGTH];
|
2013-08-08 11:08:33 +02:00
|
|
|
int nsent, to_send;
|
2013-06-26 23:29:09 +02:00
|
|
|
|
2013-09-30 15:31:40 +02:00
|
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
|
2014-08-06 22:09:18 +02:00
|
|
|
DEBUG("send_buf: Sending packet of length %" PRIu16 " from %" PRIu16 " to %" PRIu16 "\n",
|
|
|
|
packet->length, packet->src, packet->dst);
|
2013-08-08 11:08:33 +02:00
|
|
|
to_send = _native_marshall_ethernet(buf, packet);
|
2013-06-26 23:29:09 +02:00
|
|
|
|
2013-08-08 11:08:33 +02:00
|
|
|
DEBUG("send_buf: trying to send %d bytes\n", to_send);
|
|
|
|
|
2014-08-06 22:09:18 +02:00
|
|
|
if ((nsent = _native_write(_native_tap_fd, buf, to_send)) == -1) {
|
2013-06-26 23:29:09 +02:00
|
|
|
warn("write");
|
|
|
|
return -1;
|
|
|
|
}
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-12-21 16:56:42 +01:00
|
|
|
return (nsent > INT8_MAX ? INT8_MAX : nsent);
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int tap_init(char *name)
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef __MACH__ /* OSX */
|
|
|
|
char clonedev[255] = "/dev/"; /* XXX bad size */
|
2014-08-06 22:09:18 +02:00
|
|
|
strncpy(clonedev + 5, name, 250);
|
2014-04-10 14:58:12 +02:00
|
|
|
#elif defined(__FreeBSD__)
|
|
|
|
char clonedev[255] = "/dev/"; /* XXX bad size */
|
2014-08-06 22:09:18 +02:00
|
|
|
strncpy(clonedev + 5, name, 250);
|
2013-06-26 23:29:09 +02:00
|
|
|
#else /* Linux */
|
|
|
|
struct ifreq ifr;
|
2014-02-10 08:29:22 +01:00
|
|
|
const char *clonedev = "/dev/net/tun";
|
2013-06-26 23:29:09 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* implicitly create the tap interface */
|
2014-11-18 17:31:26 +01:00
|
|
|
if ((_native_tap_fd = real_open(clonedev , O_RDWR)) == -1) {
|
2013-06-26 23:29:09 +02:00
|
|
|
err(EXIT_FAILURE, "open(%s)", clonedev);
|
|
|
|
}
|
|
|
|
|
2014-04-10 14:58:12 +02:00
|
|
|
#if (defined(__MACH__) || defined(__FreeBSD__)) /* OSX/FreeBSD */
|
2014-08-06 22:09:18 +02:00
|
|
|
struct ifaddrs *iflist;
|
|
|
|
|
2014-11-18 17:31:26 +01:00
|
|
|
if (real_getifaddrs(&iflist) == 0) {
|
2013-06-26 23:29:09 +02:00
|
|
|
for (struct ifaddrs *cur = iflist; cur; cur = cur->ifa_next) {
|
|
|
|
if ((cur->ifa_addr->sa_family == AF_LINK) && (strcmp(cur->ifa_name, name) == 0) && cur->ifa_addr) {
|
2014-08-06 22:09:18 +02:00
|
|
|
struct sockaddr_dl *sdl = (struct sockaddr_dl *)cur->ifa_addr;
|
2013-06-26 23:29:09 +02:00
|
|
|
memcpy(_native_tap_mac, LLADDR(sdl), sdl->sdl_alen);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-18 17:31:26 +01:00
|
|
|
real_freeifaddrs(iflist);
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-06-26 23:29:09 +02:00
|
|
|
#else /* Linux */
|
|
|
|
memset(&ifr, 0, sizeof(ifr));
|
|
|
|
ifr.ifr_flags = IFF_TAP | IFF_NO_PI;
|
|
|
|
strncpy(ifr.ifr_name, name, IFNAMSIZ);
|
|
|
|
|
2014-11-18 17:31:26 +01:00
|
|
|
if (real_ioctl(_native_tap_fd, TUNSETIFF, (void *)&ifr) == -1) {
|
2014-01-29 19:13:49 +01:00
|
|
|
_native_in_syscall++;
|
2014-01-23 17:14:31 +01:00
|
|
|
warn("ioctl TUNSETIFF");
|
2014-01-23 17:25:41 +01:00
|
|
|
warnx("probably the tap interface (%s) does not exist or is already in use", name);
|
2014-11-18 17:31:26 +01:00
|
|
|
real_exit(EXIT_FAILURE);
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: use strncpy */
|
|
|
|
strcpy(name, ifr.ifr_name);
|
|
|
|
|
|
|
|
|
|
|
|
/* get MAC address */
|
2014-08-06 22:09:18 +02:00
|
|
|
memset(&ifr, 0, sizeof(ifr));
|
|
|
|
snprintf(ifr.ifr_name, sizeof(ifr.ifr_name), "%s", name);
|
|
|
|
|
2014-11-18 17:31:26 +01:00
|
|
|
if (real_ioctl(_native_tap_fd, SIOCGIFHWADDR, &ifr) == -1) {
|
2014-01-29 19:13:49 +01:00
|
|
|
_native_in_syscall++;
|
2014-01-23 17:14:31 +01:00
|
|
|
warn("ioctl SIOCGIFHWADDR");
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2014-06-13 23:52:05 +02:00
|
|
|
if (real_close(_native_tap_fd) == -1) {
|
2013-06-26 23:29:09 +02:00
|
|
|
warn("close");
|
|
|
|
}
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2014-11-18 17:31:26 +01:00
|
|
|
real_exit(EXIT_FAILURE);
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-06-26 23:29:09 +02:00
|
|
|
memcpy(_native_tap_mac, ifr.ifr_hwaddr.sa_data, ETHER_ADDR_LEN);
|
|
|
|
#endif
|
2014-08-06 22:09:18 +02:00
|
|
|
DEBUG("_native_tap_mac: %02x:%02x:%02x:%02x:%02x:%02x\n", _native_tap_mac[0], _native_tap_mac[1],
|
|
|
|
_native_tap_mac[2], _native_tap_mac[3], _native_tap_mac[4], _native_tap_mac[5]);
|
2014-02-05 09:46:52 +01:00
|
|
|
|
2014-08-06 15:51:46 +02:00
|
|
|
unsigned char *eui_64 = (unsigned char *)(&(_nativenet_default_dev_more._long_addr));
|
2014-02-05 09:46:52 +01:00
|
|
|
eui_64[0] = _native_tap_mac[0];
|
|
|
|
eui_64[1] = _native_tap_mac[1];
|
|
|
|
eui_64[2] = _native_tap_mac[2];
|
|
|
|
eui_64[3] = 0xff;
|
|
|
|
eui_64[4] = 0xfe;
|
|
|
|
eui_64[5] = _native_tap_mac[3];
|
|
|
|
eui_64[6] = _native_tap_mac[4];
|
|
|
|
eui_64[7] = _native_tap_mac[5];
|
2013-06-26 23:29:09 +02:00
|
|
|
|
|
|
|
/* configure signal handler for fds */
|
2013-08-08 11:08:33 +02:00
|
|
|
register_interrupt(SIGIO, _native_handle_tap_input);
|
2013-06-26 23:29:09 +02:00
|
|
|
|
2013-12-24 11:22:23 +01:00
|
|
|
#ifdef __MACH__
|
|
|
|
/* tuntap signalled IO is not working in OSX,
|
|
|
|
* check http://sourceforge.net/p/tuntaposx/bugs/17/ */
|
|
|
|
sigio_child();
|
|
|
|
#else
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-06-26 23:29:09 +02:00
|
|
|
/* configure fds to send signals on io */
|
2014-05-13 16:53:36 +02:00
|
|
|
if (fcntl(_native_tap_fd, F_SETOWN, _native_pid) == -1) {
|
2013-11-20 15:25:11 +01:00
|
|
|
err(EXIT_FAILURE, "tap_init(): fcntl(F_SETOWN)");
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
|
|
|
|
2014-08-06 15:51:46 +02:00
|
|
|
/* set file access mode to non-blocking */
|
2014-08-06 22:09:18 +02:00
|
|
|
if (fcntl(_native_tap_fd, F_SETFL, O_NONBLOCK | O_ASYNC) == -1) {
|
2013-11-20 15:25:11 +01:00
|
|
|
err(EXIT_FAILURE, "tap_init(): fcntl(F_SETFL)");
|
2013-06-26 23:29:09 +02:00
|
|
|
}
|
2014-08-06 22:09:18 +02:00
|
|
|
|
2013-12-24 11:22:23 +01:00
|
|
|
#endif /* not OSX */
|
2013-06-26 23:29:09 +02:00
|
|
|
|
2013-09-30 15:31:40 +02:00
|
|
|
DEBUG("RIOT native tap initialized.\n");
|
2013-06-26 23:29:09 +02:00
|
|
|
return _native_tap_fd;
|
|
|
|
}
|
2013-08-29 14:41:55 +02:00
|
|
|
/** @} */
|