Fix MTU as soon as possible.

If a probe reply is received that makes minmtu equal to maxmtu, we
have to wait until try_mtu() runs to realize that. Since try_mtu()
runs after a packet is sent, this means there is at least one packet
(possibly more, depending on timing) that won't benefit from the
fixed MTU. This also happens when maxmtu is updated from the send()
path.

This commit fixes that by making sure we check whether the MTU can be
fixed every time minmtu or maxmtu is touched.
This commit is contained in:
Etienne Dechamps 2015-01-01 10:32:14 +00:00
parent 97cf478318
commit df6f678957

View file

@ -65,6 +65,21 @@ int udp_discovery_timeout = 30;
#define MAX_SEQNO 1073741824 #define MAX_SEQNO 1073741824
static void try_fix_mtu(node_t *n) {
if(n->mtuprobes > 30)
return;
if(n->mtuprobes == 30 || n->minmtu >= n->maxmtu) {
if(n->minmtu > n->maxmtu)
n->minmtu = n->maxmtu;
else
n->maxmtu = n->minmtu;
n->mtu = n->minmtu;
logger(DEBUG_TRAFFIC, LOG_INFO, "Fixing MTU of %s (%s) to %d after %d probes", n->name, n->hostname, n->mtu, n->mtuprobes);
n->mtuprobes = 31;
}
}
static void udp_probe_timeout_handler(void *data) { static void udp_probe_timeout_handler(void *data) {
node_t *n = data; node_t *n = data;
if(!n->status.udp_confirmed) if(!n->status.udp_confirmed)
@ -137,8 +152,10 @@ static void udp_probe_h(node_t *n, vpn_packet_t *packet, length_t len) {
if(probelen > n->maxmtu) if(probelen > n->maxmtu)
probelen = n->maxmtu; probelen = n->maxmtu;
if(n->minmtu < probelen) if(n->minmtu < probelen) {
n->minmtu = probelen; n->minmtu = probelen;
try_fix_mtu(n);
}
/* Calculate RTT and bandwidth. /* Calculate RTT and bandwidth.
The RTT is the time between the MTU probe burst was sent and the first The RTT is the time between the MTU probe burst was sent and the first
@ -658,6 +675,7 @@ static void send_udppacket(node_t *n, vpn_packet_t *origpkt) {
n->maxmtu = origlen - 1; n->maxmtu = origlen - 1;
if(n->mtu >= origlen) if(n->mtu >= origlen)
n->mtu = origlen - 1; n->mtu = origlen - 1;
try_fix_mtu(n);
} else } else
logger(DEBUG_TRAFFIC, LOG_WARNING, "Error sending packet to %s (%s): %s", n->name, n->hostname, sockstrerror(sockerrno)); logger(DEBUG_TRAFFIC, LOG_WARNING, "Error sending packet to %s (%s): %s", n->name, n->hostname, sockstrerror(sockerrno));
} }
@ -722,6 +740,7 @@ static bool send_sptps_data_priv(node_t *to, node_t *from, int type, const void
relay->maxmtu = len - 1; relay->maxmtu = len - 1;
if(relay->mtu >= len) if(relay->mtu >= len)
relay->mtu = len - 1; relay->mtu = len - 1;
try_fix_mtu(relay);
} else { } else {
logger(DEBUG_TRAFFIC, LOG_WARNING, "Error sending UDP SPTPS packet to %s (%s): %s", relay->name, relay->hostname, sockstrerror(sockerrno)); logger(DEBUG_TRAFFIC, LOG_WARNING, "Error sending UDP SPTPS packet to %s (%s): %s", relay->name, relay->hostname, sockstrerror(sockerrno));
return false; return false;
@ -900,15 +919,7 @@ static void try_mtu(node_t *n) {
return; return;
} }
if(n->mtuprobes == 30 || (n->mtuprobes < 30 && n->minmtu >= n->maxmtu)) { try_fix_mtu(n);
if(n->minmtu > n->maxmtu)
n->minmtu = n->maxmtu;
else
n->maxmtu = n->minmtu;
n->mtu = n->minmtu;
logger(DEBUG_TRAFFIC, LOG_INFO, "Fixing MTU of %s (%s) to %d after %d probes", n->name, n->hostname, n->mtu, n->mtuprobes);
n->mtuprobes = 31;
}
int timeout; int timeout;
if(n->mtuprobes == 31) { if(n->mtuprobes == 31) {
@ -926,6 +937,8 @@ static void try_mtu(node_t *n) {
send_udp_probe_packet(n, MAX(len, 64)); send_udp_probe_packet(n, MAX(len, 64));
} }
if(n->mtuprobes >= 0)
n->mtuprobes++; n->mtuprobes++;
} }