nic_drv: read MAC and tap device from config

E.g.:
<nic mac="12:23:34:45:56:67" tap="tap1"/>

Fixes #1165.
This commit is contained in:
Johannes Schlatow 2014-03-25 15:30:36 +01:00 committed by Christian Helmuth
parent 0f4d525eda
commit b52da1045e
2 changed files with 42 additions and 11 deletions

View File

@ -4,11 +4,15 @@
* \author Christian Helmuth * \author Christian Helmuth
* \date 2011-08-08 * \date 2011-08-08
* *
* Currently, we connect to tap0 only. Possbile candidates for configuration * Configuration options are:
* options are:
* *
* - TAP device to connect to (default is tap0) * - TAP device to connect to (default is tap0)
* - MAC address (default is 02-00-00-00-00-01) * - MAC address (default is 02-00-00-00-00-01)
*
* These can be set in the config section as follows:
* <config>
* <nic mac="12:23:34:45:56:67" tap="tap1"/>
* </config>
*/ */
/* /*
@ -22,6 +26,8 @@
#include <base/sleep.h> #include <base/sleep.h>
#include <cap_session/connection.h> #include <cap_session/connection.h>
#include <nic/component.h> #include <nic/component.h>
#include <os/config.h>
#include <nic/xml_node.h>
/* Linux */ /* Linux */
#include <unistd.h> #include <unistd.h>
@ -79,9 +85,21 @@ class Linux_driver : public Nic::Driver
/* this error is fatal */ /* this error is fatal */
throw Genode::Exception(); throw Genode::Exception();
} }
Genode::memset(&ifr, 0, sizeof(ifr)); Genode::memset(&ifr, 0, sizeof(ifr));
ifr.ifr_flags = IFF_TAP | IFF_NO_PI; ifr.ifr_flags = IFF_TAP | IFF_NO_PI;
/* get tap device from config */
try {
Genode::Xml_node nic_node = Genode::config()->xml_node().sub_node("nic");
nic_node.attribute("tap").value(ifr.ifr_name, sizeof(ifr.ifr_name));
PINF("Using tap device \"%s\"", ifr.ifr_name);
} catch (...) {
/* use tap0 if no config has been provided */
Genode::strncpy(ifr.ifr_name, "tap0", sizeof(ifr.ifr_name)); Genode::strncpy(ifr.ifr_name, "tap0", sizeof(ifr.ifr_name));
PINF("No config provided, using tap0");
}
ret = ioctl(fd, TUNSETIFF, (void *) &ifr); ret = ioctl(fd, TUNSETIFF, (void *) &ifr);
if (ret != 0) { if (ret != 0) {
PERR("could not configure /dev/net/tun: no virtual network emulation"); PERR("could not configure /dev/net/tun: no virtual network emulation");
@ -98,13 +116,26 @@ class Linux_driver : public Nic::Driver
Linux_driver(Nic::Rx_buffer_alloc &alloc) Linux_driver(Nic::Rx_buffer_alloc &alloc)
: _alloc(alloc), _tap_fd(_setup_tap_fd()), _rx_thread(_tap_fd, *this) : _alloc(alloc), _tap_fd(_setup_tap_fd()), _rx_thread(_tap_fd, *this)
{ {
/* fake MAC address (unicast, locally managed) */ /* try using configured MAC address */
try {
Genode::Xml_node nic_config = Genode::config()->xml_node().sub_node("nic");
nic_config.attribute("mac").value(&_mac_addr);
PINF("Using configured MAC address \"%02x:%02x:%02x:%02x:%02x:%02x\"",
_mac_addr.addr[0],
_mac_addr.addr[1],
_mac_addr.addr[2],
_mac_addr.addr[3],
_mac_addr.addr[4],
_mac_addr.addr[5] );
} catch (...) {
/* fall back to fake MAC address (unicast, locally managed) */
_mac_addr.addr[0] = 0x02; _mac_addr.addr[0] = 0x02;
_mac_addr.addr[1] = 0x00; _mac_addr.addr[1] = 0x00;
_mac_addr.addr[2] = 0x00; _mac_addr.addr[2] = 0x00;
_mac_addr.addr[3] = 0x00; _mac_addr.addr[3] = 0x00;
_mac_addr.addr[4] = 0x00; _mac_addr.addr[4] = 0x00;
_mac_addr.addr[5] = 0x01; _mac_addr.addr[5] = 0x01;
}
_rx_thread.start(); _rx_thread.start();
} }

View File

@ -1,4 +1,4 @@
TARGET = nic_drv TARGET = nic_drv
REQUIRES = linux REQUIRES = linux
LIBS = lx_hybrid LIBS = lx_hybrid config
SRC_CC = main.cc SRC_CC = main.cc