aboutsummaryrefslogtreecommitdiff
path: root/src/core
diff options
context:
space:
mode:
authorDavid Goulet <dgoulet@torproject.org>2023-05-24 11:37:05 -0400
committerDavid Goulet <dgoulet@torproject.org>2023-05-24 11:37:05 -0400
commit9976da93679acc422c174f5e5b5bf890e387f1b1 (patch)
tree98410ddc3c152b3bb4f3c505374dd7fe16a99491 /src/core
parent0781c2968d26fba999a00c3989c10964f492f9dc (diff)
parent032d850a2c48a1d4c767accc50ad92907309e9fc (diff)
downloadtor-9976da93679acc422c174f5e5b5bf890e387f1b1.tar.gz
tor-9976da93679acc422c174f5e5b5bf890e387f1b1.zip
Merge branch 'tor-gitlab/mr/709'
Diffstat (limited to 'src/core')
-rw-r--r--src/core/or/conflux.c216
-rw-r--r--src/core/or/conflux_cell.c15
-rw-r--r--src/core/or/conflux_cell.h2
-rw-r--r--src/core/or/conflux_pool.c41
-rw-r--r--src/core/or/conflux_pool.h2
-rw-r--r--src/core/or/conflux_st.h2
6 files changed, 79 insertions, 199 deletions
diff --git a/src/core/or/conflux.c b/src/core/or/conflux.c
index e9a66b83e1..5f6af9268b 100644
--- a/src/core/or/conflux.c
+++ b/src/core/or/conflux.c
@@ -209,7 +209,7 @@ circuit_ready_to_send(const circuit_t *circ)
* cwnd, because inflight is decremented before this check */
// TODO-329-TUNING: This subtraction not be right.. It depends
// on call order wrt decisions and sendme arrival
- if (cc->inflight + cc->sendme_inc >= cc->cwnd) {
+ if (cc->inflight >= cc->cwnd) {
cc_sendable = false;
}
@@ -290,6 +290,21 @@ conflux_decide_circ_lowrtt(const conflux_t *cfx)
}
/**
+ * Returns the amount of room in a cwnd on a circuit.
+ */
+static inline uint64_t
+cwnd_available(const circuit_t *on_circ)
+{
+ const congestion_control_t *cc = circuit_ccontrol(on_circ);
+ tor_assert(cc);
+
+ if (cc->cwnd < cc->inflight)
+ return 0;
+
+ return cc->cwnd - cc->inflight;
+}
+
+/**
* Return the amount of congestion window we can send on
* on_circ during in_usec. However, if we're still in
* slow-start, send the whole window to establish the true
@@ -300,43 +315,31 @@ cwnd_sendable(const circuit_t *on_circ, uint64_t in_usec,
uint64_t our_usec)
{
const congestion_control_t *cc = circuit_ccontrol(on_circ);
-
tor_assert(cc);
-
- // TODO-329-TUNING: This function may want to consider inflight?
+ uint64_t cwnd_adjusted = cwnd_available(on_circ);
if (our_usec == 0 || in_usec == 0) {
log_fn(LOG_PROTOCOL_WARN, LD_CIRC,
"cwnd_sendable: Missing RTT data. in_usec: %" PRIu64
" our_usec: %" PRIu64, in_usec, our_usec);
- return cc->cwnd;
+ return cwnd_adjusted;
}
if (cc->in_slow_start) {
- return cc->cwnd;
+ return cwnd_adjusted;
} else {
- uint64_t sendable =
- conflux_params_get_send_pct()*cc->cwnd*in_usec/(100*our_usec);
+ /* For any given leg, it has min_rtt/2 time before the 'primary'
+ * leg's acks start arriving. So, the amount of data this
+ * 'secondary' leg can send while the min_rtt leg transmits these
+ * acks is:
+ * (cwnd_leg/(leg_rtt/2))*min_rtt/2 = cwnd_leg*min_rtt/leg_rtt.
+ */
+ uint64_t sendable = cwnd_adjusted*in_usec/our_usec;
return MIN(cc->cwnd, sendable);
}
}
/**
- * Returns the amount of room in a cwnd on a circuit.
- */
-static inline uint64_t
-cwnd_available(const circuit_t *on_circ)
-{
- const congestion_control_t *cc = circuit_ccontrol(on_circ);
- tor_assert(cc);
-
- if (cc->cwnd < cc->inflight)
- return 0;
-
- return cc->cwnd - cc->inflight;
-}
-
-/**
* Returns true if we can switch to a new circuit, false otherwise.
*
* This function assumes we're primarily switching between two circuits,
@@ -360,15 +363,14 @@ conflux_can_switch(const conflux_t *cfx)
* of the congestion window, then we can switch.
* We check the sendme_inc because there may be un-ackable
* data in inflight as well, and we can still switch then. */
+ // TODO-329-TUNING: Should we try to switch if the prev_leg is
+ // ready to send, instead of this?
if (ccontrol->inflight < ccontrol->sendme_inc ||
100*ccontrol->inflight <=
conflux_params_get_drain_pct()*ccontrol->cwnd) {
return true;
}
- // TODO-329-TUNING: Should we try to switch if the prev_leg is
- // ready to send?
-
return false;
}
@@ -407,14 +409,6 @@ conflux_decide_circ_cwndrtt(const conflux_t *cfx)
return leg->circ;
}
- /* For any given leg, it has min_rtt/2 time before the 'primary'
- * leg's acks start arriving. So, the amount of data this
- * 'secondary' leg can send while the min_rtt leg transmits these
- * acks is:
- * (cwnd_leg/(leg_rtt/2))*min_rtt/2 = cwnd_leg*min_rtt/leg_rtt.
- * So any leg with available room below that is no good.
- */
-
leg = NULL;
CONFLUX_FOR_EACH_LEG_BEGIN(cfx, l) {
@@ -425,8 +419,7 @@ conflux_decide_circ_cwndrtt(const conflux_t *cfx)
/* Pick a 'min_leg' with the lowest RTT that still has
* room in the congestion window. Note that this works for
* min_leg itself, up to inflight. */
- if (cwnd_sendable(l->circ, min_rtt, l->circ_rtts_usec) <=
- cwnd_available(l->circ)) {
+ if (cwnd_sendable(l->circ, min_rtt, l->circ_rtts_usec) > 0) {
leg = l;
}
} CONFLUX_FOR_EACH_LEG_END(l);
@@ -439,133 +432,6 @@ conflux_decide_circ_cwndrtt(const conflux_t *cfx)
}
/**
- * Favor the circuit with the highest send rate.
- *
- * Only spill over to other circuits if they are still in slow start.
- * In steady-state, we only use the max throughput circuit.
- */
-static const circuit_t *
-conflux_decide_circ_maxrate(const conflux_t *cfx)
-{
- uint64_t max_rate = 0;
- const conflux_leg_t *leg = NULL;
-
- /* Find the highest bandwidth leg */
- CONFLUX_FOR_EACH_LEG_BEGIN(cfx, l) {
- uint64_t rate;
- const congestion_control_t *cc = circuit_ccontrol(l->circ);
-
- rate = CELL_MAX_NETWORK_SIZE*USEC_PER_SEC *
- cc->cwnd / l->circ_rtts_usec;
- if (rate > max_rate) {
- max_rate = rate;
- leg = l;
- }
- } CONFLUX_FOR_EACH_LEG_END(l);
-
- /* If the package window is has room, use it */
- if (leg && circuit_ready_to_send(leg->circ)) {
- return leg->circ;
- }
-
- leg = NULL;
- max_rate = 0;
-
- /* Find the circuit with the max rate where in_slow_start == 1: */
- CONFLUX_FOR_EACH_LEG_BEGIN(cfx, l) {
- uint64_t rate;
- /* Ignore circuits with no room in the package window */
- if (!circuit_ready_to_send(l->circ)) {
- continue;
- }
-
- const congestion_control_t *cc = circuit_ccontrol(l->circ);
-
- rate = CELL_MAX_NETWORK_SIZE*USEC_PER_SEC *
- cc->cwnd / l->circ_rtts_usec;
-
- if (rate > max_rate && cc->in_slow_start) {
- max_rate = rate;
- leg = l;
- }
- } CONFLUX_FOR_EACH_LEG_END(l);
-
- /* If no sendable leg was found, don't send on any circuit. */
- if (!leg) {
- return NULL;
- }
- return leg->circ;
-}
-
-/**
- * Favor the circuit with the highest send rate that still has space
- * in the congestion window, but when it is full, pick the next
- * highest.
- */
-static const circuit_t *
-conflux_decide_circ_highrate(const conflux_t *cfx)
-{
- uint64_t max_rate = 0;
- uint64_t primary_leg_rtt = 0;
- const conflux_leg_t *leg = NULL;
-
- /* Find the highest bandwidth leg */
- CONFLUX_FOR_EACH_LEG_BEGIN(cfx, l) {
- uint64_t rate;
- const congestion_control_t *cc = circuit_ccontrol(l->circ);
-
- rate = CELL_MAX_NETWORK_SIZE*USEC_PER_SEC *
- cc->cwnd / l->circ_rtts_usec;
-
- if (rate > max_rate) {
- max_rate = rate;
- primary_leg_rtt = l->circ_rtts_usec;
- leg = l;
- }
- } CONFLUX_FOR_EACH_LEG_END(l);
-
- /* If the package window is has room, use it */
- if (leg && circuit_ready_to_send(leg->circ)) {
- return leg->circ;
- }
-
- /* Reset the max rate to find a new max */
- max_rate = 0;
- leg = NULL;
-
- /* For any given leg, it has primary_leg_rtt/2 time before the 'primary'
- * leg's acks start arriving. So, the amount of data a 'secondary'
- * leg can send while the primary leg transmits these acks is:
- * (cwnd_leg/(secondary_rtt/2))*primary_rtt/2
- * = cwnd_leg*primary_rtt/secondary_rtt.
- * So any leg with available room below that that is no good.
- */
- CONFLUX_FOR_EACH_LEG_BEGIN(cfx, l) {
- if (!circuit_ready_to_send(l->circ)) {
- continue;
- }
- const congestion_control_t *cc = circuit_ccontrol(l->circ);
-
- uint64_t rate = CELL_MAX_NETWORK_SIZE*USEC_PER_SEC *
- cc->cwnd / l->circ_rtts_usec;
-
- /* Pick the leg with the highest rate that still has room */
- if (rate > max_rate &&
- cwnd_sendable(l->circ, primary_leg_rtt, l->circ_rtts_usec) <=
- cwnd_available(l->circ)) {
- leg = l;
- max_rate = rate;
- }
- } CONFLUX_FOR_EACH_LEG_END(l);
-
- /* If no sendable leg was found, don't send on any circuit. */
- if (!leg) {
- return NULL;
- }
- return leg->circ;
-}
-
-/**
* This function is called when we want to send a relay cell on a
* conflux, as well as when we want to compute available space in
* to package from streams.
@@ -611,9 +477,12 @@ conflux_decide_circ_for_send(conflux_t *cfx,
tor_assert(cfx->curr_leg);
if (new_circ != cfx->curr_leg->circ) {
- cfx->cells_until_switch =
- cwnd_sendable(new_circ,cfx->curr_leg->circ_rtts_usec,
- new_leg->circ_rtts_usec);
+ // TODO-329-TUNING: This is one mechanism to rate limit switching,
+ // which should reduce the OOQ mem. However, we're not going to do that
+ // until we get some data on if the memory usage is high
+ cfx->cells_until_switch = 0;
+ //cwnd_sendable(new_circ,cfx->curr_leg->circ_rtts_usec,
+ // new_leg->circ_rtts_usec);
conflux_validate_stream_lists(cfx);
@@ -686,13 +555,10 @@ conflux_pick_first_leg(conflux_t *cfx)
tor_assert(min_leg);
}
- // TODO-329-TUNING: Does this create an edge condition by getting blocked,
- // is it possible that we get full before this point and block?
- // Esp if we switch to a new circuit that is not ready to
- // send because it has unacked inflight data.... This might cause
- // stalls?
- // That is the thinking with this -1 here, but maybe it is not needed.
- cfx->cells_until_switch = circuit_ccontrol(min_leg->circ)->cwnd - 1;
+ // TODO-329-TUNING: We may want to initialize this to a cwnd, to
+ // minimize early switching?
+ //cfx->cells_until_switch = circuit_ccontrol(min_leg->circ)->cwnd;
+ cfx->cells_until_switch = 0;
cfx->curr_leg = min_leg;
}
@@ -736,10 +602,6 @@ conflux_decide_next_circ(conflux_t *cfx)
return (circuit_t*)conflux_decide_circ_lowrtt(cfx);
case CONFLUX_ALG_CWNDRTT: // throughput (low oooq)
return (circuit_t*)conflux_decide_circ_cwndrtt(cfx);
- case CONFLUX_ALG_MAXRATE: // perf test (likely high ooq)
- return (circuit_t*)conflux_decide_circ_maxrate(cfx);
- case CONFLUX_ALG_HIGHRATE: // perf test (likely high ooq)
- return (circuit_t*)conflux_decide_circ_highrate(cfx);
default:
return NULL;
}
diff --git a/src/core/or/conflux_cell.c b/src/core/or/conflux_cell.c
index fa0bb1c23e..a59fa735f1 100644
--- a/src/core/or/conflux_cell.c
+++ b/src/core/or/conflux_cell.c
@@ -266,22 +266,13 @@ conflux_cell_parse_linked(const cell_t *cell, const uint16_t cell_len)
conflux_cell_link_t *
conflux_cell_new_link(const uint8_t *nonce, uint64_t last_seqno_sent,
- uint64_t last_seqno_recv, bool is_client)
+ uint64_t last_seqno_recv, uint8_t ux)
{
conflux_cell_link_t *link = tor_malloc_zero(sizeof(*link));
link->version = 0x01;
- if (is_client) {
- // TODO-329-TUNING: The default should probably be high-throughput,
- // but mobile clients may want to use low-memory.. We may also want
- // to move this choice upstairs, so that torrc can control it.
- link->desired_ux = CONFLUX_UX_HIGH_THROUGHPUT;
- } else {
- // TODO-329-TUNING: For exits, the default should be min-latency
- // but we need to fix the tests and evaluate this first.
- //link->desired_ux = CONFLUX_UX_MIN_LATENCY;
- link->desired_ux = CONFLUX_UX_HIGH_THROUGHPUT;
- }
+ link->desired_ux = ux;
+
link->last_seqno_sent = last_seqno_sent;
link->last_seqno_recv = last_seqno_recv;
memcpy(link->nonce, nonce, sizeof(link->nonce));
diff --git a/src/core/or/conflux_cell.h b/src/core/or/conflux_cell.h
index afaecae6f5..60fff42241 100644
--- a/src/core/or/conflux_cell.h
+++ b/src/core/or/conflux_cell.h
@@ -23,7 +23,7 @@ typedef struct conflux_cell_link_t {
conflux_cell_link_t *conflux_cell_new_link(const uint8_t *nonce,
uint64_t last_sent,
uint64_t last_recv,
- bool is_client);
+ uint8_t ux);
conflux_cell_link_t *conflux_cell_parse_link(const cell_t *cell,
const uint16_t cell_len);
diff --git a/src/core/or/conflux_pool.c b/src/core/or/conflux_pool.c
index ae14bd1b3c..c84613503f 100644
--- a/src/core/or/conflux_pool.c
+++ b/src/core/or/conflux_pool.c
@@ -36,6 +36,7 @@
#include "feature/nodelist/nodelist.h"
#include "feature/client/bridges.h"
+#include "app/config/config.h"
#include "lib/crypt_ops/crypto_rand.h"
#include "lib/crypt_ops/crypto_util.h"
@@ -130,6 +131,11 @@ get_linked_pool(bool is_client)
}
#endif
+/* For unit tests only: please treat these exactly as the defines in the
+ * code. */
+STATIC uint8_t DEFAULT_CLIENT_UX = CONFLUX_UX_HIGH_THROUGHPUT;
+STATIC uint8_t DEFAULT_EXIT_UX = CONFLUX_UX_MIN_LATENCY;
+
/** Helper: Format at 8 bytes the nonce for logging. */
static inline const char *
fmt_nonce(const uint8_t *nonce)
@@ -143,18 +149,19 @@ fmt_nonce(const uint8_t *nonce)
static uint8_t
conflux_choose_algorithm(uint8_t desired_ux)
{
- /* TODO-329-TUNING: Pick better algs here*/
switch (desired_ux) {
case CONFLUX_UX_NO_OPINION:
return CONFLUX_ALG_LOWRTT;
case CONFLUX_UX_MIN_LATENCY:
return CONFLUX_ALG_MINRTT;
- case CONFLUX_UX_LOW_MEM_LATENCY:
- return CONFLUX_ALG_MINRTT;
- case CONFLUX_UX_LOW_MEM_THROUGHPUT:
- return CONFLUX_ALG_CWNDRTT;
case CONFLUX_UX_HIGH_THROUGHPUT:
return CONFLUX_ALG_LOWRTT;
+ /* For now, we have no low mem algs, so use minRTT since it should
+ * switch less and thus use less mem */
+ /* TODO-329-TUNING: Pick better algs here*/
+ case CONFLUX_UX_LOW_MEM_THROUGHPUT:
+ case CONFLUX_UX_LOW_MEM_LATENCY:
+ return CONFLUX_ALG_MINRTT;
default:
/* Trunnel should protect us from this */
tor_assert_nonfatal_unreached();
@@ -1016,6 +1023,24 @@ get_exit_for_nonce(const uint8_t *nonce)
return exit;
}
+/**
+ * Return the currently configured client UX.
+ */
+static uint8_t
+get_client_ux(void)
+{
+#ifdef TOR_UNIT_TESTS
+ return DEFAULT_CLIENT_UX;
+#else
+ const or_options_t *opt = get_options();
+ tor_assert(opt);
+ (void)DEFAULT_CLIENT_UX;
+
+ /* Return the UX */
+ return opt->ConfluxClientUX;
+#endif
+}
+
/** Return true iff the given conflux object is allowed to launch a new leg. If
* the cfx object is NULL, then it is always allowed to launch a new leg. */
static bool
@@ -1109,7 +1134,7 @@ conflux_launch_leg(const uint8_t *nonce)
leg_t *leg = leg_new(TO_CIRCUIT(circ),
conflux_cell_new_link(nonce,
last_seq_sent, last_seq_recv,
- true));
+ get_client_ux()));
/* Increase the retry count for this conflux object as in this nonce. */
unlinked->cfx->num_leg_launch++;
@@ -1760,8 +1785,10 @@ conflux_process_link(circuit_t *circ, const cell_t *cell,
goto end;
}
+ /* Exits should always request min latency from clients */
conflux_cell_link_t *linked = conflux_cell_new_link(nonce, last_seq_sent,
- last_seq_recv, false);
+ last_seq_recv,
+ DEFAULT_EXIT_UX);
conflux_cell_send_linked(linked, TO_OR_CIRCUIT(circ));
tor_free(linked);
diff --git a/src/core/or/conflux_pool.h b/src/core/or/conflux_pool.h
index 547b4d3974..6bb858bb09 100644
--- a/src/core/or/conflux_pool.h
+++ b/src/core/or/conflux_pool.h
@@ -40,6 +40,8 @@ void conflux_process_linked_ack(circuit_t *circ);
bool launch_new_set(int num_legs);
digest256map_t *get_linked_pool(bool is_client);
digest256map_t *get_unlinked_pool(bool is_client);
+extern uint8_t DEFAULT_CLIENT_UX;
+extern uint8_t DEFAULT_EXIT_UX;
#endif /* defined(UNIT_TESTS) */
#endif /* TOR_CONFLUX_POOL_H */
diff --git a/src/core/or/conflux_st.h b/src/core/or/conflux_st.h
index 2112b0b7a1..8d85ad1fbe 100644
--- a/src/core/or/conflux_st.h
+++ b/src/core/or/conflux_st.h
@@ -20,8 +20,6 @@ typedef enum {
CONFLUX_ALG_MINRTT = 0,
CONFLUX_ALG_LOWRTT = 1,
CONFLUX_ALG_CWNDRTT = 2,
- CONFLUX_ALG_MAXRATE = 3,
- CONFLUX_ALG_HIGHRATE = 4
} conflux_alg_t;
/**