diff options
author | Roger Dingledine <arma@torproject.org> | 2004-10-30 05:04:52 +0000 |
---|---|---|
committer | Roger Dingledine <arma@torproject.org> | 2004-10-30 05:04:52 +0000 |
commit | 51e16233cbecf44332fb85b4db994d95c2e3bc9d (patch) | |
tree | c84da7a836cfcebc7421922b7e87bfd94e2013d0 /src | |
parent | 5c68db8b3dc74f05ab2bb6e397647f5d4f1dad0e (diff) | |
download | tor-51e16233cbecf44332fb85b4db994d95c2e3bc9d.tar.gz tor-51e16233cbecf44332fb85b4db994d95c2e3bc9d.zip |
Fix paul gardner's assert bug. Turns out when circuit_launch_by_nickname()
failed at the first hop, it would try to relaunch another circ right
then, even though the first circuit hadn't been populated yet with its
pending_final_cpath.
svn:r2624
Diffstat (limited to 'src')
-rw-r--r-- | src/or/circuitbuild.c | 4 | ||||
-rw-r--r-- | src/or/rendservice.c | 42 |
2 files changed, 27 insertions, 19 deletions
diff --git a/src/or/circuitbuild.c b/src/or/circuitbuild.c index 4c7536da53..45f675e770 100644 --- a/src/or/circuitbuild.c +++ b/src/or/circuitbuild.c @@ -1167,7 +1167,7 @@ onion_extend_cpath(crypt_path_t **head_ptr, cpath_build_state_t } log_fn(LOG_DEBUG,"Chose router %s for hop %d (exit is %s)", - choice->nickname, cur_len, state->chosen_exit_name); + choice->nickname, cur_len+1, state->chosen_exit_name); hop = tor_malloc_zero(sizeof(crypt_path_t)); @@ -1184,7 +1184,7 @@ onion_extend_cpath(crypt_path_t **head_ptr, cpath_build_state_t hop->deliver_window = CIRCWINDOW_START; log_fn(LOG_DEBUG, "Extended circuit path with %s for hop %d", - choice->nickname, cur_len); + choice->nickname, cur_len+1); *router_out = choice; return 0; diff --git a/src/or/rendservice.c b/src/or/rendservice.c index c5bdd53438..e9b1d3ed7c 100644 --- a/src/or/rendservice.c +++ b/src/or/rendservice.c @@ -30,6 +30,9 @@ typedef struct rend_service_port_config_t { /** Don't try to build more than this many circuits before giving up * for a while.*/ #define MAX_INTRO_CIRCS_PER_PERIOD 10 +/** How many times will a hidden service operator attempt to connect to + * a requested rendezvous point before giving up? */ +#define MAX_REND_FAILURES 3 /** Represents a single hidden service running at this OP. */ typedef struct rend_service_t { @@ -341,7 +344,7 @@ rend_service_introduce(circuit_t *circuit, const char *request, size_t request_l char buf[RELAY_PAYLOAD_SIZE]; char keys[DIGEST_LEN+CPATH_KEY_MATERIAL_LEN]; /* Holds KH, Df, Db, Kf, Kb */ rend_service_t *service; - int r; + int r, i; size_t len, keylen; crypto_dh_env_t *dh = NULL; circuit_t *launched = NULL; @@ -444,16 +447,19 @@ rend_service_introduce(circuit_t *circuit, const char *request, size_t request_l /* Launch a circuit to alice's chosen rendezvous point. */ - launched = circuit_launch_by_nickname(CIRCUIT_PURPOSE_S_CONNECT_REND, rp_nickname); - log_fn(LOG_INFO, - "Accepted intro; launching circuit to '%s' (cookie %s) for service %s", - rp_nickname, hexcookie, serviceid); - if (!launched) { - log_fn(LOG_WARN, - "Can't launch circuit to rendezvous point '%s' for service %s", + for(i=0;i<MAX_REND_FAILURES;i++) { + launched = circuit_launch_by_nickname(CIRCUIT_PURPOSE_S_CONNECT_REND, rp_nickname); + if (launched) + break; + } + if(!launched) { /* give up */ + log_fn(LOG_WARN,"Giving up launching first hop of circuit to rendezvous point '%s' for service %s", rp_nickname, serviceid); goto err; } + log_fn(LOG_INFO, + "Accepted intro; launching circuit to '%s' (cookie %s) for service %s", + rp_nickname, hexcookie, serviceid); tor_assert(launched->build_state); /* Fill in the circuit's state. */ memcpy(launched->rend_pk_digest, circuit->rend_pk_digest, @@ -477,10 +483,6 @@ rend_service_introduce(circuit_t *circuit, const char *request, size_t request_l return -1; } -/** How many times will a hidden service operator attempt to connect to - * a requested rendezvous point before giving up? */ -#define MAX_REND_FAILURES 3 - /** Called when we fail building a rendezvous circuit at some point other * than the last hop: launches a new circuit to the same rendezvous point. */ @@ -499,20 +501,26 @@ rend_service_relaunch_rendezvous(circuit_t *oldcirc) return; } + oldstate = oldcirc->build_state; + tor_assert(oldstate); + + if(oldstate->pending_final_cpath == NULL) { + log_fn(LOG_INFO,"Skipping relaunch of circ that failed on its first hop. Initiator will retry."); + return; + } + log_fn(LOG_INFO,"Reattempting rendezvous circuit to %s", - oldcirc->build_state->chosen_exit_name); + oldstate->chosen_exit_name); newcirc = circuit_launch_by_nickname(CIRCUIT_PURPOSE_S_CONNECT_REND, - oldcirc->build_state->chosen_exit_name); + oldstate->chosen_exit_name); if (!newcirc) { log_fn(LOG_WARN,"Couldn't relaunch rendezvous circuit to %s", - oldcirc->build_state->chosen_exit_name); + oldstate->chosen_exit_name); return; } - oldstate = oldcirc->build_state; newstate = newcirc->build_state; tor_assert(newstate); - tor_assert(oldstate); newstate->failure_count = oldstate->failure_count+1; newstate->pending_final_cpath = oldstate->pending_final_cpath; oldstate->pending_final_cpath = NULL; |