Send one MTU probe at a time.

Currently, tinc sends MTU probes in batches of three every second. This
commit changes that to send one packet every 333 milliseconds instead.

This change brings two benefits:

 - It makes MTU probing faster, because MTU probe lengths are calculated
   based on minmtu, and minmtu is adjusted based on the replies. When
   sending batches of three packets, all three packets are based on the
   same minmtu estimation; in contrast, by sending one packet more
   frequently, each subsequent packet can benefit from the replies that
   have been received since the last packet was sent. As a result, MTU
   discovery converges much faster (2-3 times as fast, typically).

 - It reduces network spikiness - it's more network-friendly to send
   one packet from time to time as opposed to sending bursts.
This commit is contained in:
Etienne Dechamps 2014-12-30 10:16:32 +00:00
parent 5bdc1f2b82
commit 1b972f2273

View file

@ -69,7 +69,7 @@ static void try_fix_mtu(node_t *n) {
if(n->mtuprobes < 0) if(n->mtuprobes < 0)
return; return;
if(n->mtuprobes == 30 || n->minmtu >= n->maxmtu) { if(n->mtuprobes == 90 || n->minmtu >= n->maxmtu) {
if(n->minmtu > n->maxmtu) if(n->minmtu > n->maxmtu)
n->minmtu = n->maxmtu; n->minmtu = n->maxmtu;
else else
@ -144,7 +144,7 @@ static void udp_probe_h(node_t *n, vpn_packet_t *packet, length_t len) {
if(probelen >= n->maxmtu + 8) { if(probelen >= n->maxmtu + 8) {
logger(DEBUG_TRAFFIC, LOG_INFO, "Increase in PMTU to %s (%s) detected, restarting PMTU discovery", n->name, n->hostname); logger(DEBUG_TRAFFIC, LOG_INFO, "Increase in PMTU to %s (%s) detected, restarting PMTU discovery", n->name, n->hostname);
n->maxmtu = MTU; n->maxmtu = MTU;
n->mtuprobes = 10; n->mtuprobes = 30;
return; return;
} }
@ -181,7 +181,7 @@ static void udp_probe_h(node_t *n, vpn_packet_t *packet, length_t len) {
n->rtt = diff.tv_sec + diff.tv_usec * 1e-6; n->rtt = diff.tv_sec + diff.tv_usec * 1e-6;
n->probe_time = probe_timestamp; n->probe_time = probe_timestamp;
} else if(n->probe_counter == 3) { } else if(n->probe_counter == 3) {
/* TODO: this will never fire after initial MTU discovery. */ /* TODO: this will never fire - we're not sending batches of three anymore. */
struct timeval probe_timestamp_diff; struct timeval probe_timestamp_diff;
timersub(&probe_timestamp, &n->probe_time, &probe_timestamp_diff); timersub(&probe_timestamp, &n->probe_time, &probe_timestamp_diff);
n->bandwidth = 2.0 * probelen / (probe_timestamp_diff.tv_sec + probe_timestamp_diff.tv_usec * 1e-6); n->bandwidth = 2.0 * probelen / (probe_timestamp_diff.tv_sec + probe_timestamp_diff.tv_usec * 1e-6);
@ -903,8 +903,8 @@ static void try_mtu(node_t *n) {
return; return;
} }
/* mtuprobes == 0..29: initial discovery, send bursts with 1 second interval, mtuprobes++ /* mtuprobes == 0..89: initial discovery, send bursts with 1 second interval, mtuprobes++
mtuprobes == 30: fix MTU, and go to -1 mtuprobes == 90: fix MTU, and go to -1
mtuprobes == -1: send one >maxmtu probe every pingtimeout */ mtuprobes == -1: send one >maxmtu probe every pingtimeout */
struct timeval now; struct timeval now;
@ -912,7 +912,7 @@ static void try_mtu(node_t *n) {
struct timeval elapsed; struct timeval elapsed;
timersub(&now, &n->probe_sent_time, &elapsed); timersub(&now, &n->probe_sent_time, &elapsed);
if(n->mtuprobes >= 0) { if(n->mtuprobes >= 0) {
if(n->mtuprobes != 0 && elapsed.tv_sec < 1) if(n->mtuprobes != 0 && elapsed.tv_sec == 0 && elapsed.tv_usec < 333333)
return; return;
} else { } else {
if(elapsed.tv_sec < pingtimeout) if(elapsed.tv_sec < pingtimeout)
@ -928,15 +928,12 @@ static void try_mtu(node_t *n) {
if(n->maxmtu + 8 < MTU) if(n->maxmtu + 8 < MTU)
send_udp_probe_packet(n, n->maxmtu + 8); send_udp_probe_packet(n, n->maxmtu + 8);
} else { } else {
/* Probes are sent in batches of three, with random sizes between the /* Probes are sent with random sizes between the
lower and upper boundaries for the MTU thus far discovered. */ lower and upper boundaries for the MTU thus far discovered. */
for (int i = 0; i < 3; i++) {
int len = n->maxmtu; int len = n->maxmtu;
if(n->minmtu < n->maxmtu) if(n->minmtu < n->maxmtu)
len = n->minmtu + 1 + rand() % (n->maxmtu - n->minmtu); len = n->minmtu + 1 + rand() % (n->maxmtu - n->minmtu);
send_udp_probe_packet(n, MAX(len, 64)); send_udp_probe_packet(n, MAX(len, 64));
}
if(n->mtuprobes >= 0) if(n->mtuprobes >= 0)
n->mtuprobes++; n->mtuprobes++;