mirror of
https://github.com/servalproject/serval-dna.git
synced 2024-12-29 17:38:59 +00:00
1661 lines
50 KiB
C
1661 lines
50 KiB
C
/*
|
|
Serval DNA overlay network interfaces
|
|
Copyright (C) 2010 Paul Gardner-Stephen
|
|
Copyright (C) 2012-2013 Serval Project Inc.
|
|
Copyright (C) 2016-2018 Flinders University
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <dirent.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/socket.h>
|
|
#include <arpa/inet.h>
|
|
#include <assert.h>
|
|
#include <time.h>
|
|
#include <fnmatch.h>
|
|
|
|
#include "serval.h"
|
|
#include "conf.h"
|
|
#include "net.h"
|
|
#include "socket.h"
|
|
#include "overlay_interface.h"
|
|
#include "strbuf.h"
|
|
#include "strbuf_helpers.h"
|
|
#include "overlay_buffer.h"
|
|
#include "overlay_packet.h"
|
|
#include "str.h"
|
|
#include "radio_link.h"
|
|
#include "server.h"
|
|
#include "route_link.h"
|
|
#include "debug.h"
|
|
|
|
|
|
// The size of the receive buffer. This effectively sets the MRU for packet radio interfaces where
|
|
// we have to buffer packets on the receive side.
|
|
#define OVERLAY_INTERFACE_RX_BUFFER_SIZE 2048
|
|
|
|
int overlay_ready=0;
|
|
overlay_interface overlay_interfaces[OVERLAY_MAX_INTERFACES];
|
|
int overlay_last_interface_number=-1;
|
|
|
|
struct profile_total interface_poll_stats;
|
|
|
|
struct sched_ent sock_any;
|
|
struct socket_address sock_any_addr;
|
|
struct profile_total sock_any_stats;
|
|
|
|
static void overlay_interface_poll(struct sched_ent *alarm);
|
|
static int inet_up_count=0;
|
|
static void rescan_soon(time_ms_t run_at);
|
|
|
|
void overlay_interface_close(overlay_interface *interface)
|
|
{
|
|
if (interface->alarm.poll.fd>=0){
|
|
if (interface->address.addr.sa_family == AF_UNIX)
|
|
unlink(interface->address.local.sun_path);
|
|
if (is_watching(&interface->alarm))
|
|
unwatch(&interface->alarm);
|
|
|
|
if (interface->address.addr.sa_family == AF_INET &&
|
|
inet_up_count>0 &&
|
|
--inet_up_count==0 &&
|
|
sock_any.poll.fd>0){
|
|
|
|
unwatch(&sock_any);
|
|
close(sock_any.poll.fd);
|
|
sock_any.poll.fd=-1;
|
|
}
|
|
|
|
close(interface->alarm.poll.fd);
|
|
interface->alarm.poll.fd=-1;
|
|
}
|
|
|
|
unschedule(&interface->alarm);
|
|
if (interface->radio_link_state)
|
|
radio_link_free(interface);
|
|
|
|
set_destination_ref(&interface->destination, NULL);
|
|
|
|
interface->state=INTERFACE_STATE_DOWN;
|
|
|
|
INFOF("Interface %s is down: type %s addr %s",
|
|
interface->name,
|
|
interface_type_name(interface->interface_type),
|
|
alloca_socket_address(&interface->address));
|
|
|
|
unsigned i, count=0;
|
|
for (i=0;i<OVERLAY_MAX_INTERFACES;i++){
|
|
if (overlay_interfaces[i].state == INTERFACE_STATE_UP)
|
|
count++;
|
|
}
|
|
CALL_TRIGGER(iupdown, interface, count);
|
|
}
|
|
|
|
static void overlay_interface_close_all()
|
|
{
|
|
unsigned i;
|
|
for (i=0;i<OVERLAY_MAX_INTERFACES;i++){
|
|
if (overlay_interfaces[i].state == INTERFACE_STATE_UP)
|
|
overlay_interface_close(&overlay_interfaces[i]);
|
|
}
|
|
}
|
|
DEFINE_TRIGGER(shutdown, overlay_interface_close_all);
|
|
|
|
void interface_state_html(struct strbuf *b, struct overlay_interface *interface)
|
|
{
|
|
switch(interface->state){
|
|
case INTERFACE_STATE_UP:
|
|
strbuf_sprintf(b, "Interface %s is Up<br>", interface->name);
|
|
break;
|
|
default:
|
|
strbuf_puts(b, "Interface Down");
|
|
return;
|
|
}
|
|
switch(interface->ifconfig.type){
|
|
case OVERLAY_INTERFACE_PACKETRADIO:
|
|
strbuf_puts(b, "Type: Packet Radio<br>");
|
|
radio_link_state_html(b, interface);
|
|
break;
|
|
case OVERLAY_INTERFACE_ETHERNET:
|
|
strbuf_puts(b, "Type: Ethernet<br>");
|
|
break;
|
|
case OVERLAY_INTERFACE_WIFI:
|
|
strbuf_puts(b, "Type: Wifi<br>");
|
|
break;
|
|
default:
|
|
case OVERLAY_INTERFACE_UNKNOWN:
|
|
strbuf_puts(b, "Type: Unknown<br>");
|
|
}
|
|
switch(interface->ifconfig.socket_type){
|
|
case SOCK_STREAM:
|
|
strbuf_puts(b, "Socket: Stream<br>");
|
|
break;
|
|
case SOCK_DGRAM:
|
|
{
|
|
strbuf_puts(b, "Socket: DGram<br>");
|
|
strbuf_sprintf(b, "Address: %s<br>", alloca_socket_address(&interface->address));
|
|
strbuf_sprintf(b, "Broadcast Address: %s<br>", alloca_socket_address(&interface->destination->address));
|
|
}
|
|
break;
|
|
case SOCK_FILE:
|
|
strbuf_puts(b, "Socket: File<br>");
|
|
break;
|
|
case SOCK_EXT:
|
|
{
|
|
strbuf_puts(b, "Socket: External<br>");
|
|
strbuf_sprintf(b, "Client: %s<br>", alloca_socket_address(&interface->address));
|
|
}
|
|
break;
|
|
}
|
|
strbuf_sprintf(b, "TX: %d<br>", interface->tx_count);
|
|
strbuf_sprintf(b, "RX: %d<br>", interface->recv_count);
|
|
}
|
|
|
|
static int
|
|
form_dummy_file_path(char *buf, size_t bufsiz, const char *configured_file_path) {
|
|
assert(configured_file_path);
|
|
assert(configured_file_path[0]);
|
|
return formf_serval_tmp_path(buf, bufsiz, "%s/%s", config.server.interface_path, configured_file_path);
|
|
}
|
|
|
|
static int
|
|
form_dgram_file_socket_address(struct socket_address *addr, const char *file_path) {
|
|
assert(file_path[0]);
|
|
if (!FORMF_SERVAL_RUN_PATH(addr->local.sun_path, "%s/%s", config.server.interface_path, file_path))
|
|
return 0;
|
|
addr->local.sun_family = AF_UNIX;
|
|
addr->addrlen = offsetof(struct sockaddr_un, sun_path) + strlen(addr->local.sun_path) + 1;
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
form_broadcast_file_socket_address(struct socket_address *broadcast, struct socket_address *addr) {
|
|
assert(addr->local.sun_family == AF_UNIX);
|
|
assert((size_t)addr->addrlen >= offsetof(struct sockaddr_un, sun_path) + 2);
|
|
assert((size_t)addr->addrlen <= sizeof(addr->local));
|
|
*broadcast = *addr;
|
|
size_t len = broadcast->addrlen - offsetof(struct sockaddr_un, sun_path) - 1;
|
|
while (len && broadcast->local.sun_path[len] != '/')
|
|
broadcast->local.sun_path[len--] = '\0';
|
|
broadcast->addrlen = offsetof(struct sockaddr_un, sun_path) + len + 2;
|
|
}
|
|
|
|
// create a socket with options common to all our UDP sockets
|
|
static int
|
|
overlay_bind_socket(const struct socket_address *addr){
|
|
int fd;
|
|
int reuseP = 1;
|
|
int broadcastP = 1;
|
|
int protocol;
|
|
|
|
switch(addr->addr.sa_family){
|
|
case AF_INET:
|
|
protocol = PF_INET;
|
|
break;
|
|
case AF_UNIX:
|
|
protocol = PF_UNIX;
|
|
break;
|
|
default:
|
|
return WHYF("Unsupported address %s", alloca_socket_address(addr));
|
|
}
|
|
|
|
fd = esocket(protocol, SOCK_DGRAM, 0);
|
|
if (fd < 0)
|
|
return -1;
|
|
|
|
if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &reuseP, sizeof(reuseP)) < 0)
|
|
WARN_perror("setsockopt(SO_REUSEADR)");
|
|
|
|
#ifdef SO_REUSEPORT
|
|
if (setsockopt(fd, SOL_SOCKET, SO_REUSEPORT, &reuseP, sizeof(reuseP)) < 0)
|
|
WARN_perror("setsockopt(SO_REUSEPORT)");
|
|
#endif
|
|
|
|
if (setsockopt(fd, SOL_SOCKET, SO_BROADCAST, &broadcastP, sizeof(broadcastP)) < 0)
|
|
WARN_perror("setsockopt(SO_BROADCAST)");
|
|
|
|
/* Automatically close socket on calls to exec().
|
|
This makes life easier when we restart with an exec after receiving
|
|
a bad signal. */
|
|
fcntl(fd, F_SETFL, fcntl(fd, F_GETFL, NULL) |
|
|
#ifdef FD_CLOEXEC
|
|
FD_CLOEXEC
|
|
#else
|
|
O_CLOEXEC
|
|
#endif
|
|
);
|
|
|
|
if (socket_bind(fd, addr))
|
|
goto error;
|
|
|
|
set_nonblock(fd);
|
|
|
|
return fd;
|
|
|
|
error:
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
|
|
static int overlay_bind_interface(overlay_interface *interface){
|
|
if (interface->alarm.poll.fd>=0)
|
|
return 0;
|
|
|
|
// We might hear about an interface coming up, after the address is assigned,
|
|
// but before the routing table is updated.
|
|
// So this bind might fail, which is ok. We just need to try again.
|
|
interface->alarm.poll.fd = overlay_bind_socket(&interface->address);
|
|
if (interface->alarm.poll.fd<0)
|
|
return -1;
|
|
|
|
DEBUGF2(packetrx, io, "Bound to %s", alloca_socket_address(&interface->address));
|
|
interface->alarm.poll.events=POLLIN;
|
|
watch(&interface->alarm);
|
|
return 0;
|
|
}
|
|
|
|
// find an interface marked for use as a default internet route
|
|
overlay_interface * overlay_interface_get_default(){
|
|
int i;
|
|
for (i=0;i<OVERLAY_MAX_INTERFACES;i++){
|
|
if (overlay_interfaces[i].state==INTERFACE_STATE_UP && overlay_interfaces[i].ifconfig.default_route)
|
|
return &overlay_interfaces[i];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// find an interface that can send a packet to this IPv4 address
|
|
overlay_interface * overlay_interface_find(struct in_addr addr, int return_default){
|
|
int i;
|
|
overlay_interface *ret = NULL;
|
|
|
|
|
|
for (i=0;i<OVERLAY_MAX_INTERFACES;i++){
|
|
if (overlay_interfaces[i].state!=INTERFACE_STATE_UP)
|
|
continue;
|
|
|
|
if (overlay_interfaces[i].address.addr.sa_family == AF_INET
|
|
&& (overlay_interfaces[i].netmask.s_addr & addr.s_addr) == (overlay_interfaces[i].netmask.s_addr & overlay_interfaces[i].address.inet.sin_addr.s_addr)){
|
|
|
|
DEBUGF(overlayinterfaces, "Found interface #%d for in_addr=0x%08x, interface mask=0x%08x, interface addr=0x%08x\n",
|
|
i,
|
|
addr.s_addr,
|
|
overlay_interfaces[i].netmask.s_addr,
|
|
overlay_interfaces[i].address.inet.sin_addr.s_addr
|
|
);
|
|
|
|
return &overlay_interfaces[i];
|
|
} else {
|
|
DEBUGF(overlayinterfaces, "in_addr=0x%08x is not from interface #%d (interface mask=0x%08x, interface addr=0x%08x)\n",
|
|
addr.s_addr,i,
|
|
overlay_interfaces[i].netmask.s_addr,
|
|
overlay_interfaces[i].address.inet.sin_addr.s_addr
|
|
);
|
|
}
|
|
|
|
// check if this is a default interface
|
|
if (return_default && overlay_interfaces[i].ifconfig.default_route) {
|
|
ret=&overlay_interfaces[i];
|
|
DEBUGF(overlayinterfaces, "in_addr=0x%08x is being deemed to default-route interface #%d (interface mask=0x%08x, interface addr=0x%08x)\n",
|
|
addr.s_addr,i,
|
|
overlay_interfaces[i].netmask.s_addr,
|
|
overlay_interfaces[i].address.inet.sin_addr.s_addr
|
|
);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
// find an interface by name and/or exact address
|
|
overlay_interface * overlay_interface_find_name_file_addr(const char *name, const char *file_path, struct socket_address *addr) {
|
|
assert(name || file_path || addr);
|
|
assert(!name || name[0]); // name is non-empty if given
|
|
assert(!file_path || file_path[0]); // file_path is non-empty if given
|
|
unsigned i;
|
|
for (i = 0; i < OVERLAY_MAX_INTERFACES; ++i) {
|
|
struct overlay_interface *interface = &overlay_interfaces[i];
|
|
if (interface->state == INTERFACE_STATE_UP
|
|
&& (!addr || cmp_sockaddr(addr, &interface->address) == 0)
|
|
&& (!file_path || strcasecmp(file_path, interface->file_path) == 0)
|
|
&& (!name || strcasecmp(name, interface->name) == 0))
|
|
return interface;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// find an interface by name and socket type
|
|
overlay_interface * overlay_interface_find_name_type(const char *name, int socket_type){
|
|
int i;
|
|
for(i = 0; i < OVERLAY_MAX_INTERFACES; i++){
|
|
if (overlay_interfaces[i].state==INTERFACE_STATE_DOWN)
|
|
continue;
|
|
|
|
if (strcasecmp(overlay_interfaces[i].name, name)==0 &&
|
|
overlay_interfaces[i].ifconfig.socket_type == socket_type)
|
|
return &overlay_interfaces[i];
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
// Which interface is better for routing packets?
|
|
// returns -1 to indicate the first is better, 0 for equal, 1 for the second
|
|
int overlay_interface_compare(overlay_interface *one, overlay_interface *two)
|
|
{
|
|
if (one==two)
|
|
return 0;
|
|
if (one->ifconfig.type<two->ifconfig.type)
|
|
return -1;
|
|
if (two->ifconfig.type<one->ifconfig.type)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
// OSX doesn't recieve broadcast packets on sockets bound to an interface's address
|
|
// So we have to bind a socket to INADDR_ANY to receive these packets.
|
|
static void
|
|
overlay_interface_read_any(struct sched_ent *alarm)
|
|
{
|
|
if (alarm->poll.revents & POLLIN) {
|
|
int plen=0;
|
|
int recvttl=1;
|
|
unsigned char packet[16384];
|
|
overlay_interface *interface=NULL;
|
|
struct socket_address recvaddr;
|
|
recvaddr.addrlen = sizeof recvaddr.store;
|
|
|
|
/* Read only one UDP packet per call to share resources more fairly, and also
|
|
enable stats to accurately count packets received */
|
|
plen = recv_message(alarm->poll.fd, &recvaddr, &recvttl, packet, sizeof(packet));
|
|
if (plen == -1) {
|
|
unwatch(alarm);
|
|
close(alarm->poll.fd);
|
|
return;
|
|
}
|
|
|
|
/* Try to identify the real interface that the packet arrived on */
|
|
interface = overlay_interface_find(recvaddr.inet.sin_addr, 0);
|
|
|
|
/* Drop the packet if we don't find a match */
|
|
if (!interface){
|
|
DEBUGF(overlayinterfaces, "Could not find matching interface for packet received from %s", inet_ntoa(recvaddr.inet.sin_addr));
|
|
return;
|
|
}
|
|
packetOkOverlay(interface, packet, plen, &recvaddr);
|
|
}
|
|
if (alarm->poll.revents & (POLLHUP | POLLERR)) {
|
|
INFO("Closing broadcast socket due to error");
|
|
unwatch(alarm);
|
|
close(alarm->poll.fd);
|
|
alarm->poll.fd=-1;
|
|
}
|
|
}
|
|
|
|
/*
|
|
bind a socket to INADDR_ANY:port
|
|
|
|
On linux you can bind to the broadcast address to receive broadcast packets per interface [or subnet],
|
|
but then you can't receive unicast packets on the same socket.
|
|
|
|
On osx, you can only receive broadcast packets if you bind to INADDR_ANY.
|
|
|
|
So the most portable way to do this is to bind to each interface's IP address for sending broadcasts
|
|
and receiving unicasts, and bind a separate socket to INADDR_ANY just for receiving broadcast packets.
|
|
|
|
Sending packets from INADDR_ANY would probably work, but gives us less control over which interfaces are sending packets.
|
|
But there may be some platforms that need some other combination for everything to work.
|
|
*/
|
|
static int overlay_interface_init_any(int port)
|
|
{
|
|
if (sock_any.poll.fd>0){
|
|
// Check the port number matches
|
|
if (sock_any_addr.inet.sin_port != htons(port))
|
|
return WHYF("Unable to listen to broadcast packets for ports %d & %d",
|
|
port, ntohs(sock_any_addr.inet.sin_port));
|
|
|
|
return 0;
|
|
}
|
|
sock_any_addr.addrlen = sizeof(sock_any_addr.inet);
|
|
sock_any_addr.inet.sin_family = AF_INET;
|
|
sock_any_addr.inet.sin_port = htons(port);
|
|
sock_any_addr.inet.sin_addr.s_addr = INADDR_ANY;
|
|
|
|
sock_any.poll.fd = overlay_bind_socket(&sock_any_addr);
|
|
if (sock_any.poll.fd<0)
|
|
return -1;
|
|
|
|
sock_any.poll.events=POLLIN;
|
|
sock_any.function = overlay_interface_read_any;
|
|
|
|
sock_any_stats.name="overlay_interface_read_any";
|
|
sock_any.stats=&sock_any_stats;
|
|
watch(&sock_any);
|
|
return 0;
|
|
}
|
|
|
|
static void calc_next_tick(struct overlay_interface *interface)
|
|
{
|
|
int interval;
|
|
if (interface->state!=INTERFACE_STATE_UP)
|
|
return;
|
|
if (link_interface_has_neighbours(interface))
|
|
interval = interface->destination->ifconfig.tick_ms;
|
|
else
|
|
// tick slower if we have no neighbours
|
|
interval = interface->ifconfig.idle_tick_ms;
|
|
|
|
time_ms_t next_tick = interface->destination->last_tx+interval;
|
|
if (!interface->destination->ifconfig.tick_ms){
|
|
next_tick=TIME_MS_NEVER_WILL;
|
|
}
|
|
time_ms_t next_allowed = limit_next_allowed(&interface->destination->transfer_limit);
|
|
if (next_tick < next_allowed)
|
|
next_tick = next_allowed;
|
|
|
|
if (interface->ifconfig.socket_type==SOCK_FILE){
|
|
time_ms_t next_read = gettime_ms()+10;
|
|
if (next_tick > next_read)
|
|
next_tick = next_read;
|
|
}else if(interface->ifconfig.socket_type==SOCK_DGRAM && interface->alarm.poll.fd<0){
|
|
time_ms_t bind_again = gettime_ms()+50;
|
|
if (next_tick > bind_again)
|
|
next_tick = bind_again;
|
|
}
|
|
|
|
interface->alarm.alarm = next_tick;
|
|
interface->alarm.deadline=interface->alarm.alarm+interface->destination->ifconfig.tick_ms/2;
|
|
}
|
|
|
|
int overlay_destination_configure(struct network_destination *dest, const struct config_mdp_iftype *ifconfig)
|
|
{
|
|
dest->ifconfig = *ifconfig;
|
|
// How often do we announce ourselves on this interface?
|
|
int tick_ms=-1;
|
|
int packet_interval=-1;
|
|
|
|
// hard coded defaults:
|
|
switch(dest->interface->ifconfig.type){
|
|
case OVERLAY_INTERFACE_PACKETRADIO:
|
|
tick_ms = 15000;
|
|
packet_interval = 1000;
|
|
break;
|
|
case OVERLAY_INTERFACE_ETHERNET:
|
|
tick_ms = 500;
|
|
packet_interval = 100;
|
|
break;
|
|
case OVERLAY_INTERFACE_WIFI:
|
|
case OVERLAY_INTERFACE_OTHER:
|
|
tick_ms = 500;
|
|
packet_interval = 800;
|
|
break;
|
|
}
|
|
|
|
if (dest->ifconfig.tick_ms<0)
|
|
dest->ifconfig.tick_ms = tick_ms;
|
|
if (dest->ifconfig.packet_interval<0)
|
|
dest->ifconfig.packet_interval = packet_interval;
|
|
|
|
if (dest->ifconfig.packet_interval<0)
|
|
return WHYF("Invalid packet interval %d specified for destination",
|
|
dest->ifconfig.packet_interval);
|
|
if (dest->ifconfig.packet_interval==0){
|
|
INFOF("Destination is not sending any traffic!");
|
|
dest->ifconfig.tick_ms=0;
|
|
}else if (!dest->ifconfig.send){
|
|
INFOF("Destination is not sending any traffic!");
|
|
dest->ifconfig.tick_ms=0;
|
|
}else if (dest->ifconfig.tick_ms==0)
|
|
INFOF("Destination is running tickless");
|
|
|
|
if (dest->ifconfig.tick_ms<0)
|
|
return WHYF("No tick interval specified for destination");
|
|
|
|
if (dest->ifconfig.reachable_timeout_ms<0)
|
|
dest->ifconfig.reachable_timeout_ms = dest->ifconfig.tick_ms>0 ? dest->ifconfig.tick_ms * 5 : 2500;
|
|
DEBUGF(linkstate, "Configured destination, tick %d, reachable %d",
|
|
dest->ifconfig.tick_ms,
|
|
dest->ifconfig.reachable_timeout_ms);
|
|
|
|
limit_init(&dest->transfer_limit, dest->ifconfig.packet_interval);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int overlay_interface_configure(struct overlay_interface *interface, const struct config_network_interface *ifconfig)
|
|
{
|
|
// copy ifconfig values
|
|
interface->ifconfig = *ifconfig;
|
|
overlay_destination_configure(interface->destination, &interface->ifconfig.broadcast);
|
|
|
|
if (ifconfig->socket_type==SOCK_STREAM)
|
|
interface->ifconfig.unicast.send=0;
|
|
|
|
// schedule the first tick asap
|
|
unschedule(&interface->alarm);
|
|
calc_next_tick(interface);
|
|
schedule(&interface->alarm);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Returns 0 if interface is successfully added.
|
|
* Returns 1 if interface is not added (eg, dummy file does not exist).
|
|
* Returns -1 in case of error (misconfiguration or system error).
|
|
*/
|
|
int
|
|
overlay_interface_init(const char *name,
|
|
const char *file_path,
|
|
short interface_type,
|
|
const struct socket_address *addr,
|
|
const struct socket_address *netmask,
|
|
const struct socket_address *broadcast,
|
|
const struct config_network_interface *ifconfig)
|
|
{
|
|
int cleanup_ret = -1;
|
|
|
|
int interface_id=-1;
|
|
int i;
|
|
for (i=0; i<OVERLAY_MAX_INTERFACES; i++){
|
|
if (overlay_interfaces[i].state==INTERFACE_STATE_DOWN){
|
|
interface_id=i;
|
|
break;
|
|
}
|
|
}
|
|
if (interface_id==-1)
|
|
return WHY("Too many interfaces -- Increase OVERLAY_MAX_INTERFACES");
|
|
|
|
overlay_interface *const interface = &overlay_interfaces[interface_id];
|
|
|
|
bzero(interface, sizeof(overlay_interface));
|
|
interface->state=INTERFACE_STATE_DOWN;
|
|
|
|
buf_strncpy_nul(interface->name, name);
|
|
|
|
if (file_path)
|
|
buf_strncpy_nul(interface->file_path, file_path);
|
|
else
|
|
interface->file_path[0]=0;
|
|
|
|
interface->interface_type = interface_type;
|
|
interface->destination = new_destination(interface);
|
|
|
|
interface->alarm.poll.fd=-1;
|
|
interface->tx_count=0;
|
|
interface->recv_count=0;
|
|
|
|
if (addr)
|
|
interface->address = *addr;
|
|
if (broadcast)
|
|
interface->destination->address = *broadcast;
|
|
interface->alarm.function = overlay_interface_poll;
|
|
interface_poll_stats.name="overlay_interface_poll";
|
|
interface->alarm.stats=&interface_poll_stats;
|
|
|
|
switch(ifconfig->socket_type){
|
|
case SOCK_DGRAM:
|
|
if (ifconfig->broadcast.drop || ifconfig->unicast.drop || ifconfig->drop_packets)
|
|
FATALF("Invalid interface definition. We only support dropping packets on dummy file interfaces");
|
|
interface->netmask = netmask->inet.sin_addr;
|
|
interface->local_echo = 1;
|
|
overlay_bind_interface(interface);
|
|
break;
|
|
|
|
case SOCK_EXT:
|
|
interface->local_echo = 0;
|
|
break;
|
|
|
|
case SOCK_STREAM:
|
|
case SOCK_FILE:
|
|
{
|
|
assert(interface->file_path[0]);
|
|
interface->local_echo = ifconfig->point_to_point?0:1;
|
|
if ((interface->alarm.poll.fd = open(interface->file_path, O_APPEND|O_RDWR)) == -1) {
|
|
if (errno == ENOENT && ifconfig->socket_type == SOCK_FILE) {
|
|
cleanup_ret = 1;
|
|
WARNF("dummy interface %s not enabled: %s does not exist", interface->name, alloca_str_toprint(interface->file_path));
|
|
} else {
|
|
cleanup_ret = WHYF_perror("dummy interface %s not enabled: open(%s, O_APPEND|O_RDWR)", interface->name, alloca_str_toprint(interface->file_path));
|
|
}
|
|
goto cleanup;
|
|
}
|
|
|
|
if (ifconfig->type==OVERLAY_INTERFACE_PACKETRADIO)
|
|
overlay_packetradio_setup_port(interface);
|
|
|
|
switch (ifconfig->socket_type) {
|
|
case SOCK_STREAM:
|
|
radio_link_init(interface);
|
|
interface->alarm.poll.events=POLLIN|POLLOUT;
|
|
watch(&interface->alarm);
|
|
break;
|
|
case SOCK_FILE:
|
|
/* Seek to end of file as initial reading point */
|
|
interface->recv_offset = lseek(interface->alarm.poll.fd,0,SEEK_END);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (overlay_interface_configure(interface, ifconfig)==-1)
|
|
return -1;
|
|
|
|
interface->state=INTERFACE_STATE_UP;
|
|
INFOF("Interface %s is up: type %s addr %s",
|
|
interface->name,
|
|
interface_type_name(interface->interface_type),
|
|
alloca_socket_address(addr));
|
|
INFOF("Allowing a maximum of %d packets every %"PRId64"ms",
|
|
interface->destination->transfer_limit.burst_size,
|
|
interface->destination->transfer_limit.burst_length);
|
|
|
|
unsigned count=0;
|
|
for (i=0;i<OVERLAY_MAX_INTERFACES;i++){
|
|
if (overlay_interfaces[i].state == INTERFACE_STATE_UP)
|
|
count++;
|
|
}
|
|
CALL_TRIGGER(iupdown, interface, count);
|
|
return 0;
|
|
|
|
cleanup:
|
|
if (interface->alarm.poll.fd>=0){
|
|
unwatch(&interface->alarm);
|
|
close(interface->alarm.poll.fd);
|
|
interface->alarm.poll.fd=-1;
|
|
}
|
|
interface->state=INTERFACE_STATE_DOWN;
|
|
return cleanup_ret;
|
|
}
|
|
|
|
static void interface_read_dgram(struct overlay_interface *interface)
|
|
{
|
|
int plen=0;
|
|
unsigned char packet[8096];
|
|
|
|
struct socket_address recvaddr;
|
|
recvaddr.addrlen = sizeof recvaddr.store;
|
|
|
|
/* Read only one UDP packet per call to share resources more fairly, and also
|
|
enable stats to accurately count packets received */
|
|
int recvttl=1;
|
|
plen = recv_message(interface->alarm.poll.fd, &recvaddr, &recvttl, packet, sizeof(packet));
|
|
if (plen == -1) {
|
|
overlay_interface_close(interface);
|
|
return;
|
|
}
|
|
packetOkOverlay(interface, packet, plen, &recvaddr);
|
|
}
|
|
|
|
struct file_packet{
|
|
struct socket_address src_addr;
|
|
struct socket_address dst_addr;
|
|
int pid;
|
|
int payload_length;
|
|
|
|
/* TODO ? ;
|
|
half-power beam height (uint16)
|
|
half-power beam width (uint16)
|
|
range in metres, centre beam (uint32)
|
|
latitude (uint32)
|
|
longitude (uint32)
|
|
X/Z direction (uint16)
|
|
Y direction (uint16)
|
|
speed in metres per second (uint16)
|
|
TX frequency in Hz, uncorrected for doppler (which must be done at the receiving end to take into account
|
|
relative motion)
|
|
coding method (use for doppler response etc) null terminated string
|
|
*/
|
|
|
|
unsigned char payload[1400];
|
|
};
|
|
|
|
static int should_drop(struct overlay_interface *interface, struct socket_address *addr){
|
|
if (interface->ifconfig.drop_packets>=100)
|
|
return 1;
|
|
|
|
if (cmp_sockaddr(addr, &interface->address)==0){
|
|
if (interface->ifconfig.unicast.drop)
|
|
return 1;
|
|
}else if (cmp_sockaddr(addr, &interface->destination->address)==0){
|
|
if (interface->ifconfig.broadcast.drop)
|
|
return 1;
|
|
}else
|
|
return 1;
|
|
|
|
if (interface->ifconfig.drop_packets <= 0)
|
|
return 0;
|
|
if (rand()%100 >= interface->ifconfig.drop_packets)
|
|
return 0;
|
|
return 1;
|
|
}
|
|
|
|
static void interface_read_file(struct overlay_interface *interface)
|
|
{
|
|
IN();
|
|
/* Grab packets, unpackage and dispatch frames to consumers */
|
|
struct file_packet packet;
|
|
|
|
/* Read from interface file */
|
|
off_t length = lseek(interface->alarm.poll.fd, (off_t)0, SEEK_END);
|
|
if (interface->recv_offset > length)
|
|
FATALF("File shrunk? It shouldn't shrink! Ever");
|
|
|
|
if (interface->recv_offset<length){
|
|
int new_packets = (length - interface->recv_offset) / sizeof packet;
|
|
if (new_packets > 20)
|
|
WARNF("Getting behind, there are %d unread packets (%"PRId64" vs %"PRId64")",
|
|
new_packets, (int64_t)interface->recv_offset, (int64_t)length);
|
|
|
|
if (lseek(interface->alarm.poll.fd,interface->recv_offset,SEEK_SET) == -1){
|
|
WHY_perror("lseek");
|
|
OUT();
|
|
return;
|
|
}
|
|
|
|
ssize_t nread = read(interface->alarm.poll.fd, &packet, sizeof packet);
|
|
if (nread == -1){
|
|
WHY_perror("read");
|
|
OUT();
|
|
return;
|
|
}
|
|
|
|
if (nread == sizeof packet) {
|
|
DEBUGF(overlayinterfaces, "Read from interface %s (filesize=%"PRId64") at offset=%"PRId64": src_addr=%s dst_addr=%s pid=%d length=%d",
|
|
interface->name, (int64_t)length, (int64_t)interface->recv_offset,
|
|
alloca_socket_address(&packet.src_addr),
|
|
alloca_socket_address(&packet.dst_addr),
|
|
packet.pid,
|
|
packet.payload_length
|
|
);
|
|
interface->recv_offset += nread;
|
|
if (should_drop(interface, &packet.dst_addr) || (packet.pid == getpid() && !interface->local_echo)){
|
|
DEBUGF(overlayinterfaces, "Ignoring packet from pid=%d src_addr=%s dst_addr=%s",
|
|
packet.pid,
|
|
alloca_socket_address(&packet.src_addr),
|
|
alloca_socket_address(&packet.dst_addr)
|
|
);
|
|
}else{
|
|
packetOkOverlay(interface, packet.payload, packet.payload_length, &packet.src_addr);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* if there's no input, while we want to check for more soon,
|
|
we need to allow all other low priority alarms to fire first,
|
|
otherwise we'll dominate the scheduler without accomplishing anything */
|
|
time_ms_t now = gettime_ms();
|
|
if (interface->recv_offset>=length){
|
|
if (now + 5 < interface->alarm.alarm){
|
|
interface->alarm.alarm = now + 5;
|
|
interface->alarm.deadline = interface->alarm.alarm + 500;
|
|
}
|
|
}else{
|
|
/* keep reading new packets as fast as possible,
|
|
but don't completely prevent other high priority alarms */
|
|
if (now < interface->alarm.alarm){
|
|
interface->alarm.alarm = now;
|
|
interface->alarm.deadline = interface->alarm.alarm + 100;
|
|
}
|
|
}
|
|
OUT();
|
|
}
|
|
|
|
static void interface_read_stream(struct overlay_interface *interface){
|
|
IN();
|
|
unsigned char buffer[OVERLAY_INTERFACE_RX_BUFFER_SIZE];
|
|
ssize_t nread = read(interface->alarm.poll.fd, buffer, OVERLAY_INTERFACE_RX_BUFFER_SIZE);
|
|
if (nread == -1){
|
|
WHY_perror("read");
|
|
OUT();
|
|
return;
|
|
}
|
|
|
|
|
|
int i;
|
|
for (i=0;i<nread;i++)
|
|
radio_link_decode(interface, buffer[i]);
|
|
|
|
OUT();
|
|
}
|
|
|
|
static void overlay_interface_poll(struct sched_ent *alarm)
|
|
{
|
|
struct overlay_interface *interface = (overlay_interface *)alarm;
|
|
time_ms_t now = gettime_ms();
|
|
|
|
if (alarm->poll.revents==0){
|
|
alarm->alarm=TIME_MS_NEVER_WILL;
|
|
|
|
if (interface->state==INTERFACE_STATE_UP && !radio_link_is_busy(interface)){
|
|
|
|
// if we couldn't initially bind to our dgram socket, try again now
|
|
if (interface->ifconfig.socket_type!=SOCK_DGRAM
|
|
|| overlay_bind_interface(interface)==0){
|
|
|
|
if (interface->destination->ifconfig.tick_ms>0
|
|
&& interface->destination->ifconfig.send
|
|
&& now >= interface->destination->last_tx+interface->destination->ifconfig.tick_ms)
|
|
overlay_send_tick_packet(interface->destination);
|
|
|
|
}else{
|
|
// If we can't bind the interface now, make sure we don't keep trying to send a tick packet.
|
|
interface->destination->last_tx = now;
|
|
}
|
|
calc_next_tick(interface);
|
|
}
|
|
|
|
switch(interface->ifconfig.socket_type){
|
|
case SOCK_STREAM:
|
|
radio_link_tx(interface);
|
|
return;
|
|
case SOCK_DGRAM:
|
|
case SOCK_EXT:
|
|
break;
|
|
case SOCK_FILE:
|
|
interface_read_file(interface);
|
|
now = gettime_ms();
|
|
break;
|
|
}
|
|
|
|
unschedule(alarm);
|
|
if (alarm->alarm!=TIME_MS_NEVER_WILL && interface->state==INTERFACE_STATE_UP) {
|
|
if (alarm->alarm < now) {
|
|
alarm->alarm = now;
|
|
alarm->deadline = alarm->alarm + interface->destination->ifconfig.tick_ms / 2;
|
|
}
|
|
schedule(alarm);
|
|
}
|
|
}
|
|
|
|
if (alarm->poll.revents & POLLOUT){
|
|
switch(interface->ifconfig.socket_type){
|
|
case SOCK_STREAM:
|
|
radio_link_tx(interface);
|
|
return;
|
|
case SOCK_EXT:
|
|
case SOCK_DGRAM:
|
|
case SOCK_FILE:
|
|
//XXX error? fatal?
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (alarm->poll.revents & POLLIN) {
|
|
switch(interface->ifconfig.socket_type){
|
|
case SOCK_DGRAM:
|
|
interface_read_dgram(interface);
|
|
break;
|
|
case SOCK_STREAM:
|
|
interface_read_stream(interface);
|
|
// if we read a valid heartbeat packet, we may be able to write more bytes now.
|
|
if (interface->state==INTERFACE_STATE_UP){
|
|
radio_link_tx(interface);
|
|
return;
|
|
}
|
|
break;
|
|
case SOCK_FILE:
|
|
interface_read_file(interface);
|
|
break;
|
|
case SOCK_EXT:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (alarm->poll.revents & (POLLHUP | POLLERR)) {
|
|
overlay_interface_close(interface);
|
|
}
|
|
}
|
|
|
|
static int send_local_packet(int fd, const uint8_t *bytes, size_t len, const char *folder, const char *file)
|
|
{
|
|
struct socket_address addr;
|
|
|
|
strbuf d = strbuf_local_buf(addr.local.sun_path);
|
|
strbuf_path_join(d, folder, file, NULL);
|
|
if (strbuf_overrun(d))
|
|
return WHYF("interface file name overrun: %s", alloca_str_toprint(strbuf_str(d)));
|
|
|
|
struct stat st;
|
|
if (lstat(addr.local.sun_path, &st))
|
|
return 1;
|
|
if (!S_ISSOCK(st.st_mode))
|
|
return 1;
|
|
|
|
addr.local.sun_family = AF_UNIX;
|
|
addr.addrlen = offsetof(struct sockaddr_un, sun_path) + strlen(addr.local.sun_path)+1;
|
|
|
|
ssize_t sent = sendto(fd, bytes, len, 0,
|
|
&addr.addr, addr.addrlen);
|
|
if (sent == -1){
|
|
if (errno != EAGAIN && errno != EWOULDBLOCK)
|
|
return WHYF_perror("sendto(%d, %zu, %s)", fd, len, alloca_socket_address(&addr));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int send_local_broadcast(int fd, const uint8_t *bytes, size_t len, const char *folder)
|
|
{
|
|
if (send_local_packet(fd, bytes, len, folder, "broadcast")==0)
|
|
return 0;
|
|
|
|
DIR *dir;
|
|
struct dirent *dp;
|
|
if ((dir = opendir(folder)) == NULL) {
|
|
WARNF_perror("opendir(%s)", alloca_str_toprint(folder));
|
|
return -1;
|
|
}
|
|
while ((dp = readdir(dir)) != NULL) {
|
|
send_local_packet(fd, bytes, len, folder, dp->d_name);
|
|
}
|
|
closedir(dir);
|
|
return 0;
|
|
}
|
|
|
|
int overlay_broadcast_ensemble(struct network_destination *destination, struct overlay_buffer *buffer)
|
|
{
|
|
assert(destination && destination->interface);
|
|
const unsigned char *bytes = ob_ptr(buffer);
|
|
size_t len = ob_position(buffer);
|
|
|
|
struct overlay_interface *interface = destination->interface;
|
|
destination->last_tx = gettime_ms();
|
|
|
|
if (IF_DEBUG(packettx)) {
|
|
DEBUGF(packettx, "Sending this packet via interface %s (len=%zu)",interface->name, len);
|
|
DEBUG_packet_visualise(NULL, bytes, len);
|
|
}
|
|
|
|
if (interface->state!=INTERFACE_STATE_UP){
|
|
ob_free(buffer);
|
|
return WHYF("Cannot send to interface %s as it is down", interface->name);
|
|
}
|
|
|
|
if (IF_DEBUG(overlayinterfaces) || interface->ifconfig.debug)
|
|
_DEBUGF_TAG("overlayinterfaces", "Sending %zu byte overlay frame on %s to %s [%s]",
|
|
(size_t)len, interface->name, alloca_socket_address(&destination->address),
|
|
alloca_tohex(bytes, len>64?64:len)
|
|
);
|
|
|
|
switch(interface->ifconfig.socket_type){
|
|
case SOCK_STREAM:
|
|
interface->tx_count++;
|
|
return radio_link_queue_packet(interface, buffer);
|
|
|
|
case SOCK_FILE:
|
|
{
|
|
struct file_packet packet={
|
|
.src_addr = interface->address,
|
|
.dst_addr = destination->address,
|
|
.pid = getpid(),
|
|
};
|
|
|
|
if (len > sizeof packet.payload) {
|
|
WARN("Truncating long packet to fit within MTU byte limit for dummy interface");
|
|
len = sizeof packet.payload;
|
|
}
|
|
packet.payload_length = len;
|
|
bcopy(bytes, packet.payload, len);
|
|
ob_free(buffer);
|
|
/* This lseek() is unneccessary because the dummy file is opened in O_APPEND mode. It's
|
|
only purpose is to find out the offset to print in the DEBUG statement. It is vulnerable
|
|
to a race condition with other processes appending to the same file. */
|
|
if (IF_DEBUG(overlayinterfaces)) {
|
|
off_t fsize = lseek(interface->alarm.poll.fd, (off_t) 0, SEEK_END);
|
|
if (fsize == -1) {
|
|
/* Don't complain if the seek fails because we are writing to a pipe or device that does
|
|
not support seeking. */
|
|
if (errno != ESPIPE)
|
|
return WHY_perror("lseek");
|
|
DEBUGF(overlayinterfaces, "Write to interface %s at offset unknown: src_addr=%s dst_addr=%s pid=%d length=%d",
|
|
interface->name,
|
|
alloca_socket_address(&packet.src_addr),
|
|
alloca_socket_address(&packet.dst_addr),
|
|
packet.pid,
|
|
packet.payload_length
|
|
);
|
|
} else
|
|
DEBUGF(overlayinterfaces, "Write to interface %s at offset=%"PRId64": src_addr=%s dst_addr=%s pid=%d length=%d",
|
|
interface->name, (int64_t)fsize,
|
|
alloca_socket_address(&packet.src_addr),
|
|
alloca_socket_address(&packet.dst_addr),
|
|
packet.pid,
|
|
packet.payload_length
|
|
);
|
|
}
|
|
ssize_t nwrite = write(interface->alarm.poll.fd, &packet, sizeof(packet));
|
|
if (nwrite == -1)
|
|
return WHY_perror("write");
|
|
if (nwrite != sizeof(packet))
|
|
return WHYF("only wrote %d of %d bytes", (int)nwrite, (int)sizeof(packet));
|
|
interface->tx_count++;
|
|
return 0;
|
|
}
|
|
case SOCK_EXT:
|
|
{
|
|
mdp_send_external_packet(interface, &destination->address, bytes, (size_t)len);
|
|
ob_free(buffer);
|
|
interface->tx_count++;
|
|
return 0;
|
|
}
|
|
case SOCK_DGRAM:
|
|
{
|
|
// check that we have bound the interface
|
|
if (overlay_bind_interface(interface)==-1){
|
|
ob_free(buffer);
|
|
return -1;
|
|
}
|
|
|
|
set_nonblock(interface->alarm.poll.fd);
|
|
if (destination->address.addr.sa_family == AF_UNIX
|
|
&& !destination->unicast){
|
|
// find all sockets in this folder and send to them
|
|
send_local_broadcast(interface->alarm.poll.fd,
|
|
bytes, (size_t)len, destination->address.local.sun_path);
|
|
}else{
|
|
ssize_t sent = sendto(interface->alarm.poll.fd,
|
|
bytes, (size_t)len, 0,
|
|
&destination->address.addr, destination->address.addrlen);
|
|
if (sent == -1){
|
|
if (errno!=EAGAIN && errno!=EWOULDBLOCK && errno!=ENOENT && errno!=ENOTDIR){
|
|
WHYF_perror("sendto(fd=%d,len=%zu,addr=%s) on interface %s",
|
|
interface->alarm.poll.fd,
|
|
(size_t)len,
|
|
alloca_socket_address(&destination->address),
|
|
interface->name
|
|
);
|
|
|
|
// if we had any error while sending broadcast packets,
|
|
// it could be because the interface is coming down
|
|
// or there might be some socket error that we can't fix.
|
|
// So bring the interface down, and scan for network changes soon
|
|
if (destination == interface->destination){
|
|
overlay_interface_close(interface);
|
|
rescan_soon(gettime_ms()+100);
|
|
}
|
|
}
|
|
ob_free(buffer);
|
|
return -1;
|
|
}
|
|
}
|
|
ob_free(buffer);
|
|
interface->tx_count++;
|
|
return 0;
|
|
}
|
|
|
|
default:
|
|
ob_free(buffer);
|
|
return WHY("Unsupported socket type");
|
|
}
|
|
}
|
|
|
|
static int match_interface_config(const struct config_network_interface *ifconfig,
|
|
int socket_type,
|
|
short interface_type,
|
|
const char *name,
|
|
const char *file_path)
|
|
{
|
|
assert (name || file_path);
|
|
if (ifconfig->socket_type != socket_type)
|
|
return 0;
|
|
if (ifconfig->match_type != OVERLAY_INTERFACE_ANY
|
|
&& ifconfig->match_type != interface_type
|
|
&& interface_type != OVERLAY_INTERFACE_UNKNOWN)
|
|
return 0;
|
|
if (name) {
|
|
unsigned i;
|
|
for (i = 0; i < ifconfig->match.patc && fnmatch(ifconfig->match.patv[i], name, 0) != 0; ++i)
|
|
;
|
|
if (i && i == ifconfig->match.patc)
|
|
return 0;
|
|
}
|
|
if (file_path && ifconfig->file[0]) {
|
|
switch (ifconfig->socket_type) {
|
|
case SOCK_DGRAM:
|
|
{
|
|
struct socket_address addr;
|
|
if (!form_dgram_file_socket_address(&addr, ifconfig->file))
|
|
return 0;
|
|
if (strcmp(addr.local.sun_path, file_path) != 0)
|
|
return 0;
|
|
}
|
|
break;
|
|
case SOCK_STREAM:
|
|
case SOCK_FILE:
|
|
{
|
|
char ifconfig_file_path[256];
|
|
if (!form_dummy_file_path(ifconfig_file_path, sizeof ifconfig_file_path, ifconfig->file))
|
|
return 0;
|
|
if (strcmp(ifconfig_file_path, file_path) != 0)
|
|
return 0;
|
|
}
|
|
break;
|
|
case SOCK_EXT:
|
|
return 0;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static const struct config_network_interface *
|
|
find_first_matching_interface_config(int socket_type, short interface_type, const char *name, const char *file_path)
|
|
{
|
|
unsigned i;
|
|
for (i = 0; i < config.interfaces.ac; ++i) {
|
|
const struct config_network_interface *ifconfig = &config.interfaces.av[i].value;
|
|
if (match_interface_config(ifconfig, socket_type, interface_type, name, file_path))
|
|
return ifconfig;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* Register the real interface, or update the existing interface registration. */
|
|
int
|
|
overlay_interface_register(const char *name,
|
|
struct socket_address *addr,
|
|
const struct socket_address *netmask,
|
|
struct socket_address *broadcast)
|
|
{
|
|
assert(name);
|
|
assert(name[0]);
|
|
|
|
short detected_interface_type = OVERLAY_INTERFACE_UNKNOWN;
|
|
#ifdef linux
|
|
detected_interface_type = OVERLAY_INTERFACE_OTHER;
|
|
// Try to determine the interface type from /sys/class/net/<name>/
|
|
char path[256];
|
|
struct stat file_stat;
|
|
strbuf sb = strbuf_local_buf(path);
|
|
strbuf_sprintf(sb, "/sys/class/net/%s/phy80211", name);
|
|
if (stat(path, &file_stat)==0){
|
|
// This interface has a symlink to a physical wifi device
|
|
detected_interface_type = OVERLAY_INTERFACE_WIFI;
|
|
}else{
|
|
WARNF_perror("stat(%s)", path);
|
|
strbuf_reset(sb);
|
|
strbuf_sprintf(sb, "/sys/class/net/%s/speed", name);
|
|
int fd = open(path, O_RDONLY);
|
|
if (fd >= 0){
|
|
// this interface implements the ethtool get_settings method
|
|
// we *could* read this file to set config based on link speed
|
|
detected_interface_type = OVERLAY_INTERFACE_ETHERNET;
|
|
close(fd);
|
|
}else{
|
|
WARNF_perror("open(%s)", path);
|
|
}
|
|
}
|
|
#endif
|
|
// find the first socket interface config rule that matches the name and detected interface type
|
|
const struct config_network_interface *ifconfig =
|
|
find_first_matching_interface_config(SOCK_DGRAM, detected_interface_type, name, NULL);
|
|
if (!ifconfig) {
|
|
DEBUGF(overlayinterfaces, "Interface %s does not match any rule", name);
|
|
return 0;
|
|
}
|
|
if (ifconfig->exclude) {
|
|
DEBUGF(overlayinterfaces, "Interface %s is explicitly excluded", name);
|
|
return 0;
|
|
}
|
|
|
|
// fill in the configured port number
|
|
if (addr->addr.sa_family==AF_INET)
|
|
addr->inet.sin_port = htons(ifconfig->port);
|
|
if (broadcast->addr.sa_family==AF_INET)
|
|
broadcast->inet.sin_port = htons(ifconfig->port);
|
|
|
|
// nothing to do if a matching interface is already up
|
|
if (overlay_interface_find_name_file_addr(name, NULL, addr))
|
|
return 0;
|
|
|
|
// new interface, so register it
|
|
if (overlay_interface_init(name, NULL, detected_interface_type, addr, netmask, broadcast, ifconfig))
|
|
return WHYF("Could not initialise newly seen interface %s", name);
|
|
|
|
overlay_interface_init_any(ifconfig->port);
|
|
inet_up_count++;
|
|
|
|
return 0;
|
|
}
|
|
|
|
#ifdef HAVE_LINUX_NETLINK_H
|
|
|
|
static void interface_unregister(const char *name, struct socket_address *addr)
|
|
{
|
|
// close all socket interfaces whose names and addresses both match
|
|
unsigned i;
|
|
for (i = 0; i < OVERLAY_MAX_INTERFACES; i++) {
|
|
struct overlay_interface *interface = &overlay_interfaces[i];
|
|
if (interface->state == INTERFACE_STATE_UP
|
|
&& match_interface_config(&interface->ifconfig, SOCK_DGRAM, OVERLAY_INTERFACE_UNKNOWN, name, NULL)) {
|
|
if (addr->addr.sa_family == AF_INET)
|
|
addr->inet.sin_port = htons(interface->ifconfig.port);
|
|
if (cmp_sockaddr(addr, &interface->address) == 0)
|
|
overlay_interface_close(interface);
|
|
}
|
|
}
|
|
}
|
|
|
|
DEFINE_ALARM(netlink_poll);
|
|
static int netlink_send_get()
|
|
{
|
|
struct {
|
|
struct nlmsghdr n;
|
|
struct ifaddrmsg r;
|
|
uint8_t alignment_padding[64];
|
|
} req;
|
|
|
|
memset(&req, 0, sizeof(req));
|
|
req.n.nlmsg_len = NLMSG_LENGTH(sizeof(struct ifaddrmsg));
|
|
req.n.nlmsg_flags = NLM_F_REQUEST | NLM_F_ROOT;
|
|
req.n.nlmsg_type = RTM_GETADDR;
|
|
req.r.ifa_family = AF_INET;
|
|
struct rtattr *rta = (struct rtattr *)(((char *)&req) + NLMSG_ALIGN(req.n.nlmsg_len));
|
|
rta->rta_len = RTA_LENGTH(4);
|
|
|
|
int fd = ALARM_STRUCT(netlink_poll).poll.fd;
|
|
if (fd<0)
|
|
return -1;
|
|
if (send(fd, &req, req.n.nlmsg_len, 0)<0)
|
|
return WHYF_perror("send(%d)", fd);
|
|
|
|
DEBUG(overlayinterfaces, "Sent RTM_GETADDR");
|
|
return 0;
|
|
}
|
|
|
|
void netlink_poll(struct sched_ent *alarm)
|
|
{
|
|
|
|
if (alarm->poll.revents & POLLIN){
|
|
uint8_t buff[4096];
|
|
ssize_t len = recv(alarm->poll.fd, buff, sizeof buff, 0);
|
|
if (len<=0)
|
|
return;
|
|
|
|
DEBUGF(overlayinterfaces, "recv(%d) len %u", alarm->poll.fd, len);
|
|
|
|
struct nlmsghdr *nlh = (struct nlmsghdr *)buff;
|
|
for (nlh = (struct nlmsghdr *)buff; (NLMSG_OK (nlh, (size_t)len)) && (nlh->nlmsg_type != NLMSG_DONE); nlh = NLMSG_NEXT(nlh, len)){
|
|
|
|
switch(nlh->nlmsg_type){
|
|
case RTM_NEWADDR:
|
|
case RTM_DELADDR:
|
|
{
|
|
struct ifaddrmsg *ifa = (struct ifaddrmsg *) NLMSG_DATA (nlh);
|
|
|
|
// ignore loopback addresses
|
|
if (ifa->ifa_scope == RT_SCOPE_HOST)
|
|
continue;
|
|
|
|
struct rtattr *rth = IFA_RTA (ifa);
|
|
int rtl = IFA_PAYLOAD (nlh);
|
|
|
|
// ifa->ifa_family;
|
|
// ifa->ifa_prefixlen;
|
|
const char *name=NULL;
|
|
|
|
struct socket_address addr, broadcast, netmask_addr;
|
|
bzero(&addr, sizeof(addr));
|
|
bzero(&broadcast, sizeof(broadcast));
|
|
bzero(&netmask_addr, sizeof(netmask_addr));
|
|
|
|
addr.addr.sa_family = broadcast.addr.sa_family = netmask_addr.addr.sa_family = ifa->ifa_family;
|
|
|
|
if (ifa->ifa_family == AF_INET){
|
|
addr.addrlen = broadcast.addrlen = netmask_addr.addrlen = sizeof(addr.inet);
|
|
}else{
|
|
DEBUGF(overlayinterfaces, "Ignoring family %d", ifa->ifa_family);
|
|
continue;
|
|
}
|
|
|
|
for (;rtl && RTA_OK (rth, rtl); rth = RTA_NEXT (rth,rtl)){
|
|
void *data = RTA_DATA(rth);
|
|
|
|
switch(rth->rta_type){
|
|
case IFA_LOCAL:
|
|
addr.inet.sin_addr.s_addr = *((uint32_t *)data);
|
|
break;
|
|
case IFA_LABEL:
|
|
name = RTA_DATA(rth);
|
|
break;
|
|
case IFA_BROADCAST:
|
|
broadcast.inet.sin_addr.s_addr = *((uint32_t *)data);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!name){
|
|
WARNF_perror("Interface name not provided by IFA_LABEL");
|
|
continue;
|
|
}
|
|
|
|
{
|
|
//calculate netmask
|
|
unsigned prefix = ifa->ifa_prefixlen;
|
|
if (prefix>32)
|
|
prefix=32;
|
|
char *c = (char *)&netmask_addr.inet.sin_addr.s_addr;
|
|
unsigned i;
|
|
for (i=0;i<(prefix/8);i++)
|
|
*c++ = 0xFF;
|
|
if (prefix %8)
|
|
*c = 0xFF << (8 - (prefix %8));
|
|
}
|
|
|
|
if (nlh->nlmsg_type==RTM_NEWADDR){
|
|
DEBUGF(overlayinterfaces, "New addr %s, %s, %s, %s",
|
|
name,
|
|
alloca_socket_address(&addr),
|
|
alloca_socket_address(&broadcast),
|
|
alloca_socket_address(&netmask_addr)
|
|
);
|
|
overlay_interface_register(name, &addr, &netmask_addr, &broadcast);
|
|
}else if (nlh->nlmsg_type==RTM_DELADDR){
|
|
DEBUGF(overlayinterfaces, "Del addr %s, %s",
|
|
name,
|
|
alloca_socket_address(&addr)
|
|
);
|
|
interface_unregister(name, &addr);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (alarm->poll.revents == 0){
|
|
netlink_send_get();
|
|
}
|
|
}
|
|
|
|
static int netlink_socket()
|
|
{
|
|
int sock = esocket(AF_NETLINK, SOCK_DGRAM, NETLINK_ROUTE);
|
|
if (sock<0)
|
|
return -1;
|
|
|
|
struct sockaddr_nl addr;
|
|
memset (&addr,0,sizeof(addr));
|
|
addr.nl_family = AF_NETLINK;
|
|
addr.nl_groups = RTMGRP_IPV4_IFADDR;
|
|
|
|
if (bind(sock, (struct sockaddr *)&addr, sizeof(addr)) == -1){
|
|
close(sock);
|
|
return WHYF_perror("bind(%d,AF_NETLINK,%lu)", sock, (unsigned long)sizeof(addr));
|
|
}
|
|
|
|
DEBUGF(overlayinterfaces, "bind(%d,AF_NETLINK,%lu)", sock, (unsigned long)sizeof(addr));
|
|
|
|
return sock;
|
|
}
|
|
|
|
// send a request to the kernel to get all interface addresses now
|
|
// eg on config change
|
|
static int netlink_init()
|
|
{
|
|
struct sched_ent *alarm=&ALARM_STRUCT(netlink_poll);
|
|
if (!is_watching(alarm)){
|
|
alarm->poll.fd = netlink_socket();
|
|
if (alarm->poll.fd<0)
|
|
return -1;
|
|
|
|
alarm->poll.events = POLLIN;
|
|
watch(alarm);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void netlink_shutdown()
|
|
{
|
|
struct sched_ent *alarm=&ALARM_STRUCT(netlink_poll);
|
|
if (alarm->poll.fd != -1){
|
|
unwatch(alarm);
|
|
close(alarm->poll.fd);
|
|
DEBUGF(overlayinterfaces, "close(%d)", alarm->poll.fd);
|
|
alarm->poll.fd = -1;
|
|
}
|
|
}
|
|
DEFINE_TRIGGER(shutdown, netlink_shutdown);
|
|
|
|
#else
|
|
|
|
// poll the OS's network interfaces
|
|
DEFINE_ALARM(overlay_interface_discover);
|
|
void overlay_interface_discover(struct sched_ent *alarm)
|
|
{
|
|
// Register new real interfaces
|
|
int no_route = 1;
|
|
#ifdef HAVE_IFADDRS_H
|
|
if (no_route != 0)
|
|
no_route = doifaddrs();
|
|
#endif
|
|
#ifdef SIOCGIFCONF
|
|
if (no_route != 0)
|
|
no_route = lsif();
|
|
#endif
|
|
#ifdef linux
|
|
if (no_route != 0)
|
|
no_route = scrapeProcNetRoute();
|
|
#endif
|
|
if (no_route != 0) {
|
|
FATAL("Unable to get any interface information");
|
|
}
|
|
|
|
alarm->alarm = gettime_ms()+5000;
|
|
alarm->deadline = alarm->alarm + 10000;
|
|
schedule(alarm);
|
|
}
|
|
|
|
#endif
|
|
|
|
static void file_interface_init(const struct config_network_interface *ifconfig)
|
|
{
|
|
assert(ifconfig->file[0]);
|
|
|
|
char file_path[256];
|
|
|
|
struct socket_address addr, netmask, broadcast;
|
|
bzero(&addr, sizeof addr);
|
|
bzero(&netmask, sizeof addr);
|
|
bzero(&broadcast, sizeof broadcast);
|
|
|
|
switch(ifconfig->socket_type){
|
|
case SOCK_FILE:
|
|
// use a fake inet address
|
|
addr.addrlen=sizeof addr.inet;
|
|
addr.inet.sin_family=AF_INET;
|
|
addr.inet.sin_port=htons(ifconfig->port);
|
|
addr.inet.sin_addr=ifconfig->dummy_address;
|
|
|
|
netmask.addrlen=sizeof addr.inet;
|
|
netmask.inet.sin_family=AF_INET;
|
|
netmask.inet.sin_port=htons(ifconfig->port);
|
|
netmask.inet.sin_addr=ifconfig->dummy_netmask;
|
|
|
|
broadcast.addrlen=sizeof addr.inet;
|
|
broadcast.inet.sin_family=AF_INET;
|
|
broadcast.inet.sin_port=htons(ifconfig->port);
|
|
broadcast.inet.sin_addr.s_addr=ifconfig->dummy_address.s_addr | ~ifconfig->dummy_netmask.s_addr;
|
|
|
|
FALLTHROUGH;
|
|
case SOCK_STREAM:
|
|
if (!form_dummy_file_path(file_path, sizeof file_path, ifconfig->file))
|
|
return; // ignore if path is too long
|
|
break;
|
|
|
|
case SOCK_DGRAM:
|
|
{
|
|
// use a local dgram socket
|
|
// no abstract sockets for now
|
|
if (!form_dgram_file_socket_address(&addr, ifconfig->file))
|
|
return;
|
|
form_broadcast_file_socket_address(&broadcast, &addr);
|
|
assert(strlen(addr.local.sun_path) + 1 <= sizeof file_path);
|
|
strcpy(file_path, addr.local.sun_path);
|
|
unlink(addr.local.sun_path);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
return; // ignore
|
|
}
|
|
|
|
// nothing to do if a matching interface is already up
|
|
if (overlay_interface_find_name_file_addr(ifconfig->file, file_path, &addr))
|
|
return;
|
|
|
|
overlay_interface_init(ifconfig->file, file_path, OVERLAY_INTERFACE_UNKNOWN, &addr, &netmask, &broadcast, ifconfig);
|
|
}
|
|
|
|
static void rescan_soon(time_ms_t run_at){
|
|
|
|
#ifdef HAVE_LINUX_NETLINK_H
|
|
struct sched_ent *alarm = &ALARM_STRUCT(netlink_poll);
|
|
|
|
// start listening for network changes & request current interface addresses soon
|
|
if (netlink_init()<0)
|
|
return;
|
|
|
|
#else
|
|
// re-check all interfaces periodically
|
|
struct sched_ent *alarm = &ALARM_STRUCT(overlay_interface_discover);
|
|
|
|
#endif
|
|
|
|
RESCHEDULE(alarm, run_at, run_at, run_at);
|
|
}
|
|
|
|
static void overlay_interface_config_change()
|
|
{
|
|
if (serverMode == SERVER_NOT_RUNNING)
|
|
return;
|
|
|
|
// bring down all interfaces that no longer match any configuration
|
|
unsigned i;
|
|
for (i = 0; i < OVERLAY_MAX_INTERFACES; i++) {
|
|
struct overlay_interface *interface = &overlay_interfaces[i];
|
|
if (interface->state == INTERFACE_STATE_UP) {
|
|
switch (interface->ifconfig.socket_type) {
|
|
case SOCK_EXT:
|
|
// leave external interfaces alone... they do not get configured from the config file, but
|
|
// from packets received from MDP clients
|
|
INFOF("Interface %s remains up: external %s", interface->name, alloca_socket_address(&interface->address));
|
|
break;
|
|
|
|
case SOCK_DGRAM:
|
|
{
|
|
const struct config_network_interface *ifconfig = NULL;
|
|
int remain = 0;
|
|
switch (interface->address.local.sun_family) {
|
|
case AF_UNIX:
|
|
ifconfig = find_first_matching_interface_config(SOCK_DGRAM, interface->interface_type, NULL, interface->address.local.sun_path);
|
|
remain = ifconfig && !ifconfig->exclude;
|
|
break;
|
|
case AF_INET:
|
|
ifconfig = find_first_matching_interface_config(SOCK_DGRAM, interface->interface_type, interface->name, NULL);
|
|
remain = ifconfig && !ifconfig->exclude;
|
|
break;
|
|
default:
|
|
remain = 1; // leave interfaces with unknown addresses alone
|
|
break;
|
|
}
|
|
if (remain)
|
|
INFOF("Interface %s remains up: type %s addr %s",
|
|
interface->name,
|
|
interface_type_name(interface->interface_type),
|
|
alloca_socket_address(&interface->address));
|
|
else
|
|
overlay_interface_close(interface);
|
|
}
|
|
break;
|
|
|
|
case SOCK_STREAM:
|
|
case SOCK_FILE:
|
|
// find a config interface rule whose absolute path matches the current interface's absolute
|
|
// file path
|
|
{
|
|
assert(interface->file_path[0]);
|
|
const struct config_network_interface *ifconfig = find_first_matching_interface_config(
|
|
interface->ifconfig.socket_type,
|
|
interface->interface_type,
|
|
NULL,
|
|
interface->file_path);
|
|
if (ifconfig && !ifconfig->exclude)
|
|
INFOF("Interface %s remains up: type %s path %s",
|
|
interface->name,
|
|
interface_type_name(interface->interface_type),
|
|
interface->file_path);
|
|
else
|
|
overlay_interface_close(interface);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
// leave interfaces with an unrecognised socket type alone
|
|
INFOF("Interface %s remains up: unrecognised socket_type=%#02x",
|
|
interface->name,
|
|
interface->ifconfig.socket_type);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// create dummy file or AF_UNIX interfaces
|
|
int real_interface = 0;
|
|
for (i = 0; i < config.interfaces.ac; ++i) {
|
|
const struct config_network_interface *ifconfig = &config.interfaces.av[i].value;
|
|
if (ifconfig->exclude)
|
|
continue;
|
|
|
|
// ignore real interfaces, we'll deal with them later
|
|
if (!ifconfig->file[0]) {
|
|
real_interface = 1;
|
|
continue;
|
|
}
|
|
|
|
overlay_interface *interface = overlay_interface_find_name_type(ifconfig->file, ifconfig->socket_type);
|
|
// ignore interfaces that are already up
|
|
if (interface)
|
|
continue;
|
|
|
|
// New file interface, so register it.
|
|
file_interface_init(ifconfig);
|
|
}
|
|
|
|
if (real_interface)
|
|
rescan_soon(gettime_ms());
|
|
}
|
|
DEFINE_TRIGGER(conf_change, overlay_interface_config_change);
|
|
|
|
void logServalPacket(int level, struct __sourceloc __whence, const char *message, const unsigned char *packet, size_t len) {
|
|
struct mallocbuf mb = STRUCT_MALLOCBUF_NULL;
|
|
if (!message) message="<no message>";
|
|
if (serval_packetvisualise_xpf(XPRINTF_MALLOCBUF(&mb), message, packet, len) == -1)
|
|
WHY("serval_packetvisualise() failed");
|
|
else if (mb.buffer == NULL)
|
|
WHYF("serval_packetvisualise() output buffer missing, message=%s packet=%p len=%lu", alloca_toprint(-1, message, strlen(message)), packet, (long unsigned int)len);
|
|
else
|
|
serval_log_multiline(level, __whence, mb.buffer);
|
|
if (mb.buffer)
|
|
free(mb.buffer);
|
|
}
|