tinc/src/net.c

574 lines
15 KiB
C
Raw Normal View History

2000-03-26 00:33:07 +00:00
/*
net.c -- most of the network code
Copyright (C) 1998-2005 Ivo Timmermans,
2000-2015 Guus Sliepen <guus@tinc-vpn.org>
2006 Scott Lamb <slamb@slamb.org>
2012-10-10 15:17:49 +00:00
2011 Loïc Grenié <loic.grenie@gmail.com>
2000-03-26 00:33:07 +00:00
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.
2000-03-26 00:33:07 +00:00
*/
#include "system.h"
#include "utils.h"
#include "conf.h"
#include "connection.h"
#include "device.h"
#include "graph.h"
#include "logger.h"
#include "meta.h"
#include "names.h"
#include "net.h"
#include "netutl.h"
#include "protocol.h"
#include "slpd.h"
#include "subnet.h"
2003-07-29 22:59:01 +00:00
#include "xalloc.h"
2000-03-26 00:33:07 +00:00
int contradicting_add_edge = 0;
int contradicting_del_edge = 0;
static int sleeptime = 10;
time_t last_config_check = 0;
static timeout_t pingtimer;
static timeout_t periodictimer;
static struct timeval last_periodic_run_time;
/* Purge edges and subnets of unreachable nodes. Use carefully. */
2002-02-10 21:57:54 +00:00
2007-11-07 02:49:57 +00:00
void purge(void) {
logger(DEBUG_PROTOCOL, LOG_DEBUG, "Purging unreachable nodes");
2002-09-09 21:25:28 +00:00
/* Remove all edges and subnets owned by unreachable nodes. */
2012-10-07 22:35:38 +00:00
for splay_each(node_t, n, node_tree) {
2002-09-09 21:25:28 +00:00
if(!n->status.reachable) {
logger(DEBUG_SCARY_THINGS, LOG_DEBUG, "Purging node %s (%s)", n->name, n->hostname);
2002-09-09 21:25:28 +00:00
2012-10-07 22:35:38 +00:00
for splay_each(subnet_t, s, n->subnet_tree) {
send_del_subnet(everyone, s);
if(!strictsubnets)
subnet_del(n, s);
2002-09-09 21:25:28 +00:00
}
2012-10-07 22:35:38 +00:00
for splay_each(edge_t, e, n->edge_tree) {
if(!tunnelserver)
send_del_edge(everyone, e);
2002-09-09 21:25:28 +00:00
edge_del(e);
}
}
}
/* Check if anyone else claims to have an edge to an unreachable node. If not, delete node. */
2012-10-07 22:35:38 +00:00
for splay_each(node_t, n, node_tree) {
if(!n->status.reachable) {
2012-10-07 22:35:38 +00:00
for splay_each(edge_t, e, edge_weight_tree)
if(e->to == n)
2012-10-07 22:35:38 +00:00
return;
2002-09-09 21:25:28 +00:00
if(!autoconnect && (!strictsubnets || !n->subnet_tree->head))
/* in strictsubnets mode do not delete nodes with subnets */
node_del(n);
2002-09-09 21:25:28 +00:00
}
}
2002-02-10 21:57:54 +00:00
}
2000-03-26 00:33:07 +00:00
/*
Terminate a connection:
- Mark it as inactive
- Remove the edge representing this connection
- Kill it with fire
- Check if we need to retry making an outgoing connection
2000-03-26 00:33:07 +00:00
*/
2007-05-18 10:00:00 +00:00
void terminate_connection(connection_t *c, bool report) {
logger(DEBUG_CONNECTIONS, LOG_NOTICE, "Closing connection with %s (%s)", c->name, c->hostname);
if(c->node) {
if(c->node->connection == c)
c->node->connection = NULL;
if(c->edge) {
if(report && !tunnelserver)
send_del_edge(everyone, c->edge);
2002-02-10 21:57:54 +00:00
edge_del(c->edge);
c->edge = NULL;
/* Run MST and SSSP algorithms */
graph();
/* If the node is not reachable anymore but we remember it had an edge to us, clean it up */
if(report && !c->node->status.reachable) {
edge_t *e;
e = lookup_edge(c->node, myself);
if(e) {
if(!tunnelserver)
send_del_edge(everyone, e);
edge_del(e);
}
}
}
2002-09-09 21:25:28 +00:00
}
outgoing_t *outgoing = c->outgoing;
connection_del(c);
2002-09-09 21:25:28 +00:00
/* Check if this was our outgoing connection */
2007-05-19 13:34:32 +00:00
if(outgoing)
do_outgoing_connection(outgoing);
#ifndef HAVE_MINGW
/* Clean up dead proxy processes */
while(waitpid(-1, NULL, WNOHANG) > 0);
#endif
2000-03-26 00:33:07 +00:00
}
/*
Check if the other end is active.
If we have sent packets, but didn't receive any,
then possibly the other end is dead. We send a
PING request over the meta connection. If the other
end does not reply in time, we consider them dead
and close the connection.
2000-03-26 00:33:07 +00:00
*/
static void timeout_handler(void *data) {
bool close_all_connections = false;
/*
timeout_handler will start after 30 seconds from start of tincd
hold information about the elapsed time since last time the handler
has been run
*/
long sleep_time = now.tv_sec - last_periodic_run_time.tv_sec;
/*
It seems that finding sane default value is harder than expected
Since we send every second a UDP packet to make holepunching work
And default UDP state expire on firewalls is between 15-30 seconds
we drop all connections after 60 Seconds - UDPDiscoveryTimeout=30
by default
*/
if (sleep_time > 2 * udp_discovery_timeout) {
2016-05-19 21:35:20 +00:00
logger(DEBUG_ALWAYS, LOG_ERR, "Back from the dead after %ld seconds of sleep", sleep_time);
/*
Do not send any packets to tinc after we wake up.
The other node probably closed our connection but we still
are holding context information to them. This may happen on
laptops or any other hardware which can be suspended for some time.
Sending any data to node that wasn't expecting it will produce
annoying and misleading errors on the other side about failed signature
verification and or about missing sptps context
*/
close_all_connections = true;
}
last_periodic_run_time = now;
2012-10-07 22:35:38 +00:00
for list_each(connection_t, c, connection_list) {
// control connections (eg. tinc ctl) do not have any timeout
if(c->status.control)
continue;
if(close_all_connections) {
logger(DEBUG_ALWAYS, LOG_ERR, "Forcing connection close after sleep time %s (%s)", c->name, c->hostname);
terminate_connection(c, c->edge);
continue;
}
// Bail out early if we haven't reached the ping timeout for this node yet
2016-05-23 19:27:53 +00:00
if(c->last_ping_time.tv_sec + pingtimeout > now.tv_sec)
continue;
// timeout during connection establishing
if(!c->edge) {
if(c->status.connecting)
logger(DEBUG_CONNECTIONS, LOG_WARNING, "Timeout while connecting to %s (%s)", c->name, c->hostname);
else
logger(DEBUG_CONNECTIONS, LOG_WARNING, "Timeout from %s (%s) during authentication", c->name, c->hostname);
terminate_connection(c, c->edge);
continue;
}
// helps in UDP holepunching
try_tx(c->node, false);
// timeout during ping
if(c->status.pinged) {
2017-04-28 08:40:50 +00:00
logger(DEBUG_CONNECTIONS, LOG_INFO, "%s (%s) didn't respond to PING in %ld seconds", c->name, c->hostname, (long)(now.tv_sec - c->last_ping_time.tv_sec));
terminate_connection(c, c->edge);
continue;
2002-09-09 21:25:28 +00:00
}
// check whether we need to send a new ping
2016-05-23 19:27:53 +00:00
if(c->last_ping_time.tv_sec + pinginterval <= now.tv_sec)
send_ping(c);
2002-09-09 21:25:28 +00:00
}
timeout_set(data, &(struct timeval){1, rand() % 100000});
}
static void periodic_handler(void *data) {
/* Check if there are too many contradicting ADD_EDGE and DEL_EDGE messages.
This usually only happens when another node has the same Name as this node.
If so, sleep for a short while to prevent a storm of contradicting messages.
*/
if(contradicting_del_edge > 100 && contradicting_add_edge > 100) {
logger(DEBUG_ALWAYS, LOG_WARNING, "Possible node with same Name as us! Sleeping %d seconds.", sleeptime);
2016-04-14 15:20:36 +00:00
nanosleep(&(struct timespec){sleeptime, 0}, NULL);
sleeptime *= 2;
if(sleeptime < 0)
sleeptime = 3600;
} else {
sleeptime /= 2;
if(sleeptime < 10)
sleeptime = 10;
}
2002-09-09 21:25:28 +00:00
contradicting_add_edge = 0;
contradicting_del_edge = 0;
/* If AutoConnect is set, check if we need to make or break connections. */
if(autoconnect && node_tree->count > 1) {
/* Count number of active connections */
int nc = 0;
for list_each(connection_t, c, connection_list) {
if(c->edge)
nc++;
}
if(nc < 3) {
/* Not enough active connections, try to add one.
Choose a random node, if we don't have a connection to it,
and we are not already trying to make one, create an
outgoing connection to this node.
*/
int count = 0;
for splay_each(node_t, n, node_tree) {
if(n == myself || n->connection || !(n->status.has_address || n->status.reachable))
continue;
count++;
}
2016-05-14 21:21:36 +00:00
if(!count)
goto end;
2016-05-14 21:21:36 +00:00
logger(DEBUG_STATUS, LOG_INFO, "%d active connections - autoconnecting to one of %d available nodes",
2016-05-08 14:46:29 +00:00
nc, count);
int r = rand() % count;
for splay_each(node_t, n, node_tree) {
if(n == myself || n->connection || !(n->status.has_address || n->status.reachable))
continue;
if(r--)
continue;
bool found = false;
for list_each(outgoing_t, outgoing, outgoing_list) {
if(!strcmp(outgoing->name, n->name)) {
found = true;
break;
}
}
2016-05-08 14:46:29 +00:00
if(!found) {
logger(DEBUG_CONNECTIONS, LOG_INFO, "Autoconnecting to %s", n->name);
outgoing_t *outgoing = xzalloc(sizeof *outgoing);
outgoing->name = xstrdup(n->name);
list_insert_tail(outgoing_list, outgoing);
setup_outgoing_connection(outgoing);
}
break;
}
} else if(nc > 3) {
/* Too many active connections, try to remove one.
Choose a random outgoing connection to a node
that has at least one other connection.
*/
int r = rand() % nc;
int i = 0;
for list_each(connection_t, c, connection_list) {
if(!c->edge)
continue;
if(i++ != r)
continue;
if(!c->outgoing || c->outgoing->keep_it || !c->node || c->node->edge_tree->count < 2)
break;
logger(DEBUG_CONNECTIONS, LOG_INFO, "Autodisconnecting from %s", c->name);
list_delete(outgoing_list, c->outgoing);
c->outgoing = NULL;
terminate_connection(c, c->edge);
break;
}
}
if(nc >= 3) {
/* If we have enough active connections,
remove any pending outgoing connections.
*/
for list_each(outgoing_t, o, outgoing_list) {
bool found = false;
for list_each(connection_t, c, connection_list) {
if(c->outgoing == o) {
found = true;
break;
}
}
if(!found && !o->keep_it) {
logger(DEBUG_CONNECTIONS, LOG_INFO, "Cancelled outgoing connection to %s", o->name);
list_delete_node(outgoing_list, node);
}
}
}
}
// call SLPD periodic handler
periodic_slpd_handler();
end:
timeout_set(data, &(struct timeval){5, rand() % 100000});
2000-03-26 00:33:07 +00:00
}
void handle_meta_connection_data(connection_t *c) {
if (!receive_meta(c)) {
terminate_connection(c, c->edge);
return;
2002-09-09 21:25:28 +00:00
}
}
2013-02-20 13:39:24 +00:00
#ifndef HAVE_MINGW
static void sigterm_handler(void *data) {
logger(DEBUG_ALWAYS, LOG_NOTICE, "Got %s signal", strsignal(((signal_t *)data)->signum));
event_exit();
}
static void sighup_handler(void *data) {
logger(DEBUG_ALWAYS, LOG_NOTICE, "Got %s signal", strsignal(((signal_t *)data)->signum));
reopenlogger();
if(reload_configuration())
exit(1);
}
static void sigalrm_handler(void *data) {
logger(DEBUG_ALWAYS, LOG_NOTICE, "Got %s signal", strsignal(((signal_t *)data)->signum));
2011-06-25 18:20:07 +00:00
retry();
}
2013-02-20 13:39:24 +00:00
#endif
2011-06-25 18:20:07 +00:00
int reload_configuration(void) {
char fname[PATH_MAX];
2007-05-17 21:34:58 +00:00
/* Reread our own configuration file */
exit_configuration(&config_tree);
init_configuration(&config_tree);
if(!read_server_config()) {
logger(DEBUG_ALWAYS, LOG_ERR, "Unable to reread configuration file.");
return EINVAL;
2007-05-17 21:34:58 +00:00
}
read_config_options(config_tree, NULL);
snprintf(fname, sizeof fname, "%s" SLASH "hosts" SLASH "%s", confbase, myself->name);
2012-10-10 15:17:49 +00:00
read_config_file(config_tree, fname);
/* Parse some options that are allowed to be changed while tinc is running */
setup_myself_reloadable();
/* If StrictSubnet is set, expire deleted Subnets and read new ones in */
if(strictsubnets) {
2012-10-07 22:35:38 +00:00
for splay_each(subnet_t, subnet, subnet_tree)
if (subnet->owner)
subnet->expires = 1;
}
2002-09-09 21:25:28 +00:00
for splay_each(node_t, n, node_tree)
n->status.has_address = false;
load_all_nodes();
if(strictsubnets) {
2012-10-07 22:35:38 +00:00
for splay_each(subnet_t, subnet, subnet_tree) {
if (!subnet->owner)
continue;
if(subnet->expires == 1) {
send_del_subnet(everyone, subnet);
if(subnet->owner->status.reachable)
subnet_update(subnet->owner, subnet, false);
subnet_del(subnet->owner, subnet);
} else if(subnet->expires == -1) {
subnet->expires = 0;
} else {
send_add_subnet(everyone, subnet);
if(subnet->owner->status.reachable)
subnet_update(subnet->owner, subnet, true);
}
2002-09-09 21:25:28 +00:00
}
} else { /* Only read our own subnets back in */
2012-10-07 22:35:38 +00:00
for splay_each(subnet_t, subnet, myself->subnet_tree)
if(!subnet->expires)
subnet->expires = 1;
config_t *cfg = lookup_config(config_tree, "Subnet");
while(cfg) {
subnet_t *subnet, *s2;
if(!get_config_subnet(cfg, &subnet))
continue;
if((s2 = lookup_subnet(myself, subnet))) {
if(s2->expires == 1)
s2->expires = 0;
free_subnet(subnet);
} else {
subnet_add(myself, subnet);
send_add_subnet(everyone, subnet);
subnet_update(myself, subnet, true);
}
cfg = lookup_config_next(config_tree, cfg);
}
2012-10-07 22:35:38 +00:00
for splay_each(subnet_t, subnet, myself->subnet_tree) {
if(subnet->expires == 1) {
send_del_subnet(everyone, subnet);
subnet_update(myself, subnet, false);
subnet_del(myself, subnet);
}
}
}
2002-09-09 21:25:28 +00:00
2007-05-17 21:34:58 +00:00
/* Try to make outgoing connections */
2012-10-10 15:17:49 +00:00
2007-05-17 21:34:58 +00:00
try_outgoing_connections();
/* Close connections to hosts that have a changed or deleted host config file */
2012-10-07 22:35:38 +00:00
for list_each(connection_t, c, connection_list) {
if(c->status.control)
continue;
snprintf(fname, sizeof fname, "%s" SLASH "hosts" SLASH "%s", confbase, c->name);
struct stat s;
if(stat(fname, &s) || s.st_mtime > last_config_check) {
logger(DEBUG_CONNECTIONS, LOG_INFO, "Host config file of %s has been changed", c->name);
terminate_connection(c, c->edge);
}
}
last_config_check = now.tv_sec;
return 0;
2007-05-17 21:34:58 +00:00
}
void retry(void) {
/* Reset the reconnection timers for all outgoing connections */
for list_each(outgoing_t, outgoing, outgoing_list) {
outgoing->timeout = 0;
if(outgoing->ev.cb)
timeout_set(&outgoing->ev, &(struct timeval){0, 0});
}
/* Check for outgoing connections that are in progress, and reset their ping timers */
2012-10-07 22:35:38 +00:00
for list_each(connection_t, c, connection_list) {
if(c->outgoing && !c->node) {
c->last_ping_time.tv_sec = 0;
c->last_ping_time.tv_usec = 0;
}
2007-05-17 23:04:02 +00:00
}
/* Kick the ping timeout handler */
timeout_set(&pingtimer, &(struct timeval){0, 0});
2007-05-17 23:04:02 +00:00
}
/*
Initialize timers
*/
void init_timers(void) {
timeout_add(&pingtimer, timeout_handler, &pingtimer, &(struct timeval){pingtimeout, rand() % 100000});
timeout_add(&periodictimer, periodic_handler, &periodictimer, &(struct timeval){0, 0});
}
/*
Release timers
*/
void exit_timers(void) {
timeout_del(&periodictimer);
timeout_del(&pingtimer);
}
2000-03-26 00:33:07 +00:00
/*
this is where it all happens...
2000-03-26 00:33:07 +00:00
*/
2007-05-18 10:00:00 +00:00
int main_loop(void) {
last_periodic_run_time = now;
init_timers();
2011-06-25 18:20:07 +00:00
#ifndef HAVE_MINGW
2015-06-30 17:14:54 +00:00
signal_t sighup;
signal_t sigterm;
signal_t sigquit;
signal_t sigint;
signal_t sigalrm;
memset(&sighup, 0x0, sizeof(signal_t));
memset(&sigterm, 0x0, sizeof(signal_t));
memset(&sigquit, 0x0, sizeof(signal_t));
memset(&sigint, 0x0, sizeof(signal_t));
memset(&sigalrm, 0x0, sizeof(signal_t));
signal_add(&sighup, sighup_handler, &sighup, SIGHUP);
signal_add(&sigterm, sigterm_handler, &sigterm, SIGTERM);
signal_add(&sigquit, sigterm_handler, &sigquit, SIGQUIT);
2013-01-17 13:14:17 +00:00
signal_add(&sigint, sigterm_handler, &sigint, SIGINT);
signal_add(&sigalrm, sigalrm_handler, &sigalrm, SIGALRM);
#endif
2007-05-17 21:34:58 +00:00
if(!event_loop()) {
logger(DEBUG_ALWAYS, LOG_ERR, "Error while waiting for input: %s", sockstrerror(sockerrno));
return 1;
2002-09-09 21:25:28 +00:00
}
2011-06-25 18:20:07 +00:00
#ifndef HAVE_MINGW
signal_del(&sighup);
signal_del(&sigterm);
2013-08-02 18:50:19 +00:00
signal_del(&sigquit);
signal_del(&sigint);
signal_del(&sigalrm);
2011-06-25 18:20:07 +00:00
#endif
exit_timers();
return 0;
2000-03-26 00:33:07 +00:00
}