improved conjestion control

This commit is contained in:
notsecure 2014-05-21 20:42:26 -04:00
parent be4559de73
commit 1e4eb83880
2 changed files with 31 additions and 8 deletions

View File

@ -30,6 +30,8 @@
#include "net_crypto.h" #include "net_crypto.h"
#include "util.h" #include "util.h"
#include "math.h"
static uint8_t crypt_connection_id_not_valid(Net_Crypto *c, int crypt_connection_id) static uint8_t crypt_connection_id_not_valid(Net_Crypto *c, int crypt_connection_id)
{ {
return (uint32_t)crypt_connection_id >= c->crypto_connections_length; return (uint32_t)crypt_connection_id >= c->crypto_connections_length;
@ -2180,15 +2182,31 @@ static void send_crypto_packets(Net_Crypto *c)
} }
if ((PACKET_COUNTER_AVERAGE_INTERVAL + conn->packet_counter_set) < temp_time) { if ((PACKET_COUNTER_AVERAGE_INTERVAL + conn->packet_counter_set) < temp_time) {
conn->packet_recv_rate = (double)conn->packet_counter / ((double)(temp_time - conn->packet_counter_set) / 1000.0);
double dt = temp_time - conn->packet_counter_set;
conn->packet_recv_rate = (double)conn->packet_counter / (dt / 1000.0);
conn->packet_counter = 0; conn->packet_counter = 0;
conn->packet_counter_set = temp_time; conn->packet_counter_set = temp_time;
if ((double)num_packets_array(&conn->send_array) < 0.3 * (conn->packet_send_rate)) { //most important constant, higher makes it less agressive (minimize packet loss)
conn->packet_send_rate *= 1.2; //keep it >=0
} else if ((double)num_packets_array(&conn->send_array) > 0.4 * (conn->packet_send_rate)) { #define RATE_ADD_POWER 100.0
conn->packet_send_rate *= 0.8;
} double dropped = (conn->dropped + (double)conn->packets_resent / dt) / 2.0;
double realrate = (conn->packet_send_rate - dropped * 1000.0);
//newrate is reduced rate based on dropped packets
double newrate = (realrate + conn->packet_send_rate) / 2.0;
//rate_add is rate increase (needs to be improved)
double ratio = realrate / conn->packet_send_rate;
double rate_add = (newrate / 40.0 + sqrt(newrate)) * pow(ratio, RATE_ADD_POWER);;
conn->packet_send_rate = newrate + rate_add;
conn->dropped = dropped;
conn->packets_resent = 0;
if (conn->packet_send_rate < CRYPTO_PACKET_MIN_RATE || !conn->sending) if (conn->packet_send_rate < CRYPTO_PACKET_MIN_RATE || !conn->sending)
conn->packet_send_rate = CRYPTO_PACKET_MIN_RATE; conn->packet_send_rate = CRYPTO_PACKET_MIN_RATE;
@ -2202,7 +2220,7 @@ static void send_crypto_packets(Net_Crypto *c)
conn->last_packets_left_set = temp_time; conn->last_packets_left_set = temp_time;
conn->packets_left = conn->packet_send_rate; conn->packets_left = conn->packet_send_rate;
} else if (((1000.0 / conn->packet_send_rate) + conn->last_packets_left_set) < temp_time) { } else if (((1000.0 / conn->packet_send_rate) + conn->last_packets_left_set) < temp_time) {
uint32_t num_packets = conn->packet_send_rate * ((double)(temp_time - conn->last_packets_left_set) / 1000.0); uint32_t num_packets = conn->packet_send_rate * ((double)(temp_time - conn->last_packets_left_set) / 1000.0) + 0.5;
if (conn->packets_left > num_packets * 2 + CRYPTO_MIN_QUEUE_LENGTH) { if (conn->packets_left > num_packets * 2 + CRYPTO_MIN_QUEUE_LENGTH) {
conn->packets_left = num_packets * 2 + CRYPTO_MIN_QUEUE_LENGTH; conn->packets_left = num_packets * 2 + CRYPTO_MIN_QUEUE_LENGTH;
@ -2215,11 +2233,13 @@ static void send_crypto_packets(Net_Crypto *c)
int ret = send_requested_packets(c, i, conn->packets_left); int ret = send_requested_packets(c, i, conn->packets_left);
conn->packets_resent += ret;
if (ret != -1) { if (ret != -1) {
conn->packets_left -= ret; conn->packets_left -= ret;
} }
if (conn->sending != 0 && num_packets_array(&conn->send_array) < CRYPTO_MIN_QUEUE_LENGTH) { if (conn->sending != 0 && num_packets_array(&conn->send_array) < CRYPTO_MIN_QUEUE_LENGTH / 2) {
--conn->sending; --conn->sending;
} }
} }

View File

@ -137,6 +137,9 @@ typedef struct {
uint32_t packets_left; uint32_t packets_left;
uint64_t last_packets_left_set; uint64_t last_packets_left_set;
double dropped;
uint32_t packets_resent;
uint8_t sending; /* indicates if data is being sent or not. */ uint8_t sending; /* indicates if data is being sent or not. */
uint8_t killed; /* set to 1 to kill the connection. */ uint8_t killed; /* set to 1 to kill the connection. */