diff options
-rw-r--r-- | src/or/connection_edge.c | 4 | ||||
-rw-r--r-- | src/or/dns.c | 60 | ||||
-rw-r--r-- | src/or/or.h | 2 |
3 files changed, 44 insertions, 22 deletions
diff --git a/src/or/connection_edge.c b/src/or/connection_edge.c index 6a65984774..dfe5d57036 100644 --- a/src/or/connection_edge.c +++ b/src/or/connection_edge.c @@ -118,6 +118,10 @@ int connection_edge_process_relay_cell(cell_t *cell, circuit_t *circ, connection if(conn->type == CONN_TYPE_EXIT && relay_command == RELAY_COMMAND_END) { log_fn(LOG_INFO,"Exit got end before we're connected. Marking for close."); conn->marked_for_close = 1; + if(conn->state == EXIT_CONN_STATE_RESOLVING) { + log_fn(LOG_INFO,"...and informing resolver we don't want the answer anymore."); + dns_cancel_pending_resolve(conn->address, conn); + } } else { log_fn(LOG_DEBUG,"Got an unexpected relay cell, not in 'open' state. Dropping."); } diff --git a/src/or/dns.c b/src/or/dns.c index 16c55ee6f6..080f041eca 100644 --- a/src/or/dns.c +++ b/src/or/dns.c @@ -20,7 +20,6 @@ int num_workers=0; int num_workers_busy=0; static int dns_assign_to_worker(connection_t *exitconn); -static void dns_cancel_pending_resolve(char *question); static void dns_found_answer(char *question, uint32_t answer); static void dnsworker_main(int fd); static int dns_spawn_worker(void); @@ -154,7 +153,7 @@ static int dns_assign_to_worker(connection_t *exitconn) { if(!dnsconn) { log(LOG_INFO,"dns_assign_to_worker(): no idle dns workers. Failing."); - dns_cancel_pending_resolve(exitconn->address); + dns_cancel_pending_resolve(exitconn->address, NULL); return -1; } @@ -168,7 +167,7 @@ static int dns_assign_to_worker(connection_t *exitconn) { connection_write_to_buf(dnsconn->address, len, dnsconn) < 0) { log(LOG_NOTICE,"dns_assign_to_worker(): Write failed. Closing worker and failing resolve."); dnsconn->marked_for_close = 1; - dns_cancel_pending_resolve(exitconn->address); + dns_cancel_pending_resolve(exitconn->address, NULL); return -1; } @@ -176,8 +175,12 @@ static int dns_assign_to_worker(connection_t *exitconn) { return 0; } -static void dns_cancel_pending_resolve(char *question) { - struct pending_connection_t *pend; +/* if onlyconn is NULL, cancel the whole thing. if onlyconn is defined, + * then remove onlyconn from the pending list, and if the pending list + * is now empty, cancel the whole thing. + */ +void dns_cancel_pending_resolve(char *question, connection_t *onlyconn) { + struct pending_connection_t *pend, *victim; struct cached_resolve search; struct cached_resolve *resolve, *tmp; @@ -185,18 +188,39 @@ static void dns_cancel_pending_resolve(char *question) { resolve = SPLAY_FIND(cache_tree, &cache_root, &search); if(!resolve) { - log_fn(LOG_ERR,"Answer to unasked question '%s'? Dropping.", question); + log_fn(LOG_INFO,"Answer to unasked question '%s'? Dropping.", question); return; } assert(resolve->state == CACHE_STATE_PENDING); + assert(resolve->pending_connections); - /* mark all pending connections to fail */ - while(resolve->pending_connections) { + if(onlyconn) { pend = resolve->pending_connections; - pend->conn->marked_for_close = 1; - resolve->pending_connections = pend->next; - free(pend); + if(pend->conn == onlyconn) { + resolve->pending_connections = pend->next; + free(pend); + if(resolve->pending_connections) /* more pending, don't cancel it */ + return; + } else { + for( ; pend->next; pend = pend->next) { + if(pend->next->conn == onlyconn) { + victim = pend->next; + pend->next = victim->next; + free(victim); + return; /* more are pending */ + } + } + assert(0); /* not reachable unless onlyconn not in pending list */ + } + } else { + /* mark all pending connections to fail */ + while(resolve->pending_connections) { + pend = resolve->pending_connections; + pend->conn->marked_for_close = 1; + resolve->pending_connections = pend->next; + free(pend); + } } /* remove resolve from the linked list */ @@ -229,19 +253,11 @@ static void dns_found_answer(char *question, uint32_t answer) { resolve = SPLAY_FIND(cache_tree, &cache_root, &search); if(!resolve) { - log_fn(LOG_ERR,"Answer to unasked question '%s'? Dropping.", question); + log_fn(LOG_INFO,"Answer to unasked question '%s'? Dropping.", question); return; } assert(resolve->state == CACHE_STATE_PENDING); - /* XXX this is a bug which hasn't been found yet. Probably something - * about slaves answering questions when they're not supposed to, and - * reusing the old question. - */ - if(resolve->state != CACHE_STATE_PENDING) { - log(LOG_ERR,"dns_found_answer(): BUG: resolve '%s' in state %d (not pending). Dropping.",question, resolve->state); - return; - } resolve->answer = ntohl(answer); if(resolve->answer) @@ -276,7 +292,7 @@ int connection_dns_process_inbuf(connection_t *conn) { if(conn->inbuf_reached_eof) { log(LOG_ERR,"connection_dnsworker_process_inbuf(): Read eof. Worker dying."); if(conn->state == DNSWORKER_STATE_BUSY) - dns_cancel_pending_resolve(conn->address); + dns_cancel_pending_resolve(conn->address, NULL); return -1; } @@ -398,7 +414,7 @@ static void spawn_enough_workers(void) { assert(dnsconn); /* tell the exit connection that it's failed */ - dns_cancel_pending_resolve(dnsconn->address); + dns_cancel_pending_resolve(dnsconn->address, NULL); dnsconn->marked_for_close = 1; num_workers_busy--; diff --git a/src/or/or.h b/src/or/or.h index 36d3004b6b..b71723ee60 100644 --- a/src/or/or.h +++ b/src/or/or.h @@ -395,6 +395,7 @@ typedef struct { int KeepalivePeriod; int MaxOnionsPending; int NewCircuitPeriod; + int TotalBandwidth; int Role; int loglevel; } or_options_t; @@ -638,6 +639,7 @@ int connection_dir_handle_listener_read(connection_t *conn); void dns_init(void); int connection_dns_finished_flushing(connection_t *conn); int connection_dns_process_inbuf(connection_t *conn); +void dns_cancel_pending_resolve(char *question, connection_t *onlyconn); int dns_resolve(connection_t *exitconn); /********************************* main.c ***************************/ |