diff options
author | Roger Dingledine <arma@torproject.org> | 2003-10-21 08:37:07 +0000 |
---|---|---|
committer | Roger Dingledine <arma@torproject.org> | 2003-10-21 08:37:07 +0000 |
commit | 4a66865d0be89445e70230548e7cf2a1cbd59db5 (patch) | |
tree | 48f8d1fbd755c03d97023f4a50d8095e919515cd /src | |
parent | f8a72b1c217c182f41a71122eb5e01bc5d0e7f77 (diff) | |
download | tor-4a66865d0be89445e70230548e7cf2a1cbd59db5.tar.gz tor-4a66865d0be89445e70230548e7cf2a1cbd59db5.zip |
send the end cell when we realize we're going to end,
not when we're closing the stream.
this lets us put a payload in the end cell if we want to,
to describe why we're closing the stream.
there are still some places where we don't send the end cell
immediately. i need to track them down. but it's a low priority,
since i've made it send the end cell when we close the stream if
we haven't already sent it.
svn:r640
Diffstat (limited to 'src')
-rw-r--r-- | src/or/circuit.c | 95 | ||||
-rw-r--r-- | src/or/connection.c | 7 | ||||
-rw-r--r-- | src/or/connection_edge.c | 72 | ||||
-rw-r--r-- | src/or/dns.c | 4 | ||||
-rw-r--r-- | src/or/or.h | 5 | ||||
-rw-r--r-- | src/or/test.c | 10 |
6 files changed, 125 insertions, 68 deletions
diff --git a/src/or/circuit.c b/src/or/circuit.c index be05caafd7..d3053eb273 100644 --- a/src/or/circuit.c +++ b/src/or/circuit.c @@ -512,52 +512,59 @@ void circuit_about_to_close_connection(connection_t *conn) { circuit_t *circ; connection_t *prevconn; - if(!connection_speaks_cells(conn)) { - /* it's an edge conn. need to remove it from the linked list of - * conn's for this circuit. Send an 'end' relay command. - * But don't kill the circuit. - */ - - circ = circuit_get_by_conn(conn); - if(!circ) + switch(conn->type) { + case CONN_TYPE_OR: + /* We must close all the circuits on it. */ + while((circ = circuit_get_by_conn(conn))) { + if(circ->n_conn == conn) /* it's closing in front of us */ + circ->n_conn = NULL; + if(circ->p_conn == conn) /* it's closing behind us */ + circ->p_conn = NULL; + circuit_close(circ); + } return; + case CONN_TYPE_AP: + case CONN_TYPE_EXIT: - if(conn == circ->p_streams) { - circ->p_streams = conn->next_stream; - goto send_end; - } - if(conn == circ->n_streams) { - circ->n_streams = conn->next_stream; - goto send_end; - } - for(prevconn = circ->p_streams; prevconn && prevconn->next_stream && prevconn->next_stream != conn; prevconn = prevconn->next_stream) ; - if(prevconn && prevconn->next_stream) { - prevconn->next_stream = conn->next_stream; - goto send_end; - } - for(prevconn = circ->n_streams; prevconn && prevconn->next_stream && prevconn->next_stream != conn; prevconn = prevconn->next_stream) ; - if(prevconn && prevconn->next_stream) { - prevconn->next_stream = conn->next_stream; - goto send_end; - } - log_fn(LOG_ERR,"edge conn not in circuit's list?"); - assert(0); /* should never get here */ -send_end: - connection_edge_send_command(conn, circ, RELAY_COMMAND_END, - NULL, 0, conn->cpath_layer); - return; - } + /* It's an edge conn. Need to remove it from the linked list of + * conn's for this circuit. Confirm that 'end' relay command has + * been sent. But don't kill the circuit. + */ - /* this connection speaks cells. We must close all the circuits on it. */ - while((circ = circuit_get_by_conn(conn))) { - if(circ->n_conn == conn) /* it's closing in front of us */ - circ->n_conn = NULL; - if(circ->p_conn == conn) /* it's closing behind us */ - circ->p_conn = NULL; - circuit_close(circ); - } + circ = circuit_get_by_conn(conn); + if(!circ) + return; + + if(!conn->has_sent_end) { + log_fn(LOG_INFO,"Edge connection hasn't sent end yet? Bug."); + connection_edge_send_command(conn, circ, RELAY_COMMAND_END, + NULL, 0, conn->cpath_layer); + } + + if(conn == circ->p_streams) { + circ->p_streams = conn->next_stream; + return; + } + if(conn == circ->n_streams) { + circ->n_streams = conn->next_stream; + return; + } + for(prevconn = circ->p_streams; prevconn && prevconn->next_stream && prevconn->next_stream != conn; prevconn = prevconn->next_stream) ; + if(prevconn && prevconn->next_stream) { + prevconn->next_stream = conn->next_stream; + return; + } + for(prevconn = circ->n_streams; prevconn && prevconn->next_stream && prevconn->next_stream != conn; prevconn = prevconn->next_stream) ; + if(prevconn && prevconn->next_stream) { + prevconn->next_stream = conn->next_stream; + return; + } + log_fn(LOG_ERR,"edge conn not in circuit's list?"); + assert(0); /* should never get here */ + } /* end switch */ } + /* FIXME this now leaves some out */ void circuit_dump_by_conn(connection_t *conn, int severity) { circuit_t *circ; @@ -903,7 +910,11 @@ int circuit_truncated(circuit_t *circ, crypt_path_t *layer) { for(stream = circ->p_streams; stream; stream=stream->next_stream) { if(stream->cpath_layer == victim) { log_fn(LOG_INFO, "Marking stream %d for close.", *(int*)stream->stream_id); -/*ENDCLOSE*/ stream->marked_for_close = 1; + /* no need to send 'end' relay cells, + * because the other side's already dead + */ + stream->marked_for_close = 1; + stream->has_sent_end = 1; } } diff --git a/src/or/connection.c b/src/or/connection.c index 2c8a0032a0..a7bd89a71a 100644 --- a/src/or/connection.c +++ b/src/or/connection.c @@ -353,9 +353,9 @@ int connection_read_to_buf(connection_t *conn) { } else { /* do a rudimentary round-robin so one connection can't hog a thickpipe */ if(connection_speaks_cells(conn)) { - at_most = 10*(CELL_NETWORK_SIZE); + at_most = 30*(CELL_NETWORK_SIZE); } else { - at_most = 10*(CELL_PAYLOAD_SIZE - RELAY_HEADER_SIZE); + at_most = 30*(CELL_PAYLOAD_SIZE - RELAY_HEADER_SIZE); } if(at_most > global_read_bucket) @@ -644,7 +644,8 @@ int connection_send_destroy(aci_t aci, connection_t *conn) { if(!connection_speaks_cells(conn)) { log_fn(LOG_INFO,"Aci %d: At an edge. Marking connection for close.", aci); -/*ENDCLOSE*/ conn->marked_for_close = 1; + connection_edge_end(conn, NULL, 0, conn->cpath_layer); + /* if they already sent a destroy, they know. XXX can just close? */ return 0; } diff --git a/src/or/connection_edge.c b/src/or/connection_edge.c index b79da2d088..394139932d 100644 --- a/src/or/connection_edge.c +++ b/src/or/connection_edge.c @@ -26,27 +26,34 @@ int connection_edge_process_inbuf(connection_t *conn) { /* eof reached; we're done reading, but we might want to write more. */ conn->done_receiving = 1; shutdown(conn->s, 0); /* XXX check return, refactor NM */ - if (conn->done_sending) -/*ENDCLOSE*/ conn->marked_for_close = 1; - - /* XXX Factor out common logic here and in circuit_about_to_close NM */ - connection_edge_send_command(conn, circuit_get_by_conn(conn), RELAY_COMMAND_END, - NULL, 0, conn->cpath_layer); + if (conn->done_sending) { + connection_edge_end(conn, NULL, 0, conn->cpath_layer); + } else { + connection_edge_send_command(conn, circuit_get_by_conn(conn), RELAY_COMMAND_END, + NULL, 0, conn->cpath_layer); + } return 0; #else /* eof reached, kill it. */ log_fn(LOG_INFO,"conn (fd %d) reached eof. Closing.", conn->s); -/*ENDCLOSE*/ return -1; + connection_edge_end(conn, NULL, 0, conn->cpath_layer); + return -1; #endif } switch(conn->state) { case AP_CONN_STATE_SOCKS_WAIT: -/*ENDCLOSE*/ return connection_ap_handshake_process_socks(conn); + if(connection_ap_handshake_process_socks(conn) < 0) { + connection_edge_end(conn, NULL, 0, conn->cpath_layer); + return -1; + } + return 0; case AP_CONN_STATE_OPEN: case EXIT_CONN_STATE_OPEN: - if(connection_edge_package_raw_inbuf(conn) < 0) -/*ENDCLOSE*/ return -1; + if(connection_edge_package_raw_inbuf(conn) < 0) { + connection_edge_end(conn, NULL, 0, conn->cpath_layer); + return -1; + } return 0; case EXIT_CONN_STATE_CONNECTING: log_fn(LOG_INFO,"text from server while in 'connecting' state at exit. Leaving it on buffer."); @@ -56,6 +63,25 @@ int connection_edge_process_inbuf(connection_t *conn) { return 0; } +void connection_edge_end(connection_t *conn, void *payload, int payload_len, + crypt_path_t *cpath_layer) { + circuit_t *circ = circuit_get_by_conn(conn); + + if(conn->has_sent_end) { + log_fn(LOG_WARN,"It appears I've already sent the end. Are you calling me twice?"); + return; + } + + if(circ) { + log_fn(LOG_DEBUG,"Marking conn (fd %d) and sending end.",conn->s); + connection_edge_send_command(conn, circ, RELAY_COMMAND_END, + payload, payload_len, cpath_layer); + } + + conn->marked_for_close = 1; + conn->has_sent_end = 1; +} + void connection_edge_send_command(connection_t *fromconn, circuit_t *circ, int relay_command, void *payload, int payload_len, crypt_path_t *cpath_layer) { cell_t cell; @@ -128,6 +154,7 @@ int connection_edge_process_relay_cell(cell_t *cell, circuit_t *circ, connection return 0; } else { log_fn(LOG_WARN,"Got an unexpected relay cell, not in 'open' state. Closing."); + connection_edge_end(conn, NULL, 0, conn->cpath_layer); return -1; } } @@ -148,12 +175,15 @@ int connection_edge_process_relay_cell(cell_t *cell, circuit_t *circ, connection if((edge_type == EDGE_AP && --layer_hint->deliver_window < 0) || (edge_type == EDGE_EXIT && --circ->deliver_window < 0)) { log_fn(LOG_WARN,"(relay data) circ deliver_window below 0. Killing."); + connection_edge_end(conn, NULL, 0, conn->cpath_layer); return -1; } log_fn(LOG_DEBUG,"circ deliver_window now %d.", edge_type == EDGE_AP ? layer_hint->deliver_window : circ->deliver_window); - if(circuit_consider_sending_sendme(circ, edge_type, layer_hint) < 0) + if(circuit_consider_sending_sendme(circ, edge_type, layer_hint) < 0) { + conn->has_sent_end = 1; /* we failed because conn is broken. can't send end. */ return -1; + } if(!conn) { log_fn(LOG_INFO,"relay cell dropped, unknown stream %d.",*(int*)conn->stream_id); @@ -181,10 +211,14 @@ int connection_edge_process_relay_cell(cell_t *cell, circuit_t *circ, connection #ifdef HALF_OPEN conn->done_sending = 1; shutdown(conn->s, 1); /* XXX check return; refactor NM */ - if (conn->done_receiving) -/*ENDCLOSE*/ conn->marked_for_close = 1; + if (conn->done_receiving) { + conn->marked_for_close = 1; + conn->has_sent_end = 1; /* no need to send end, we just got one! */ + } +#else + conn->marked_for_close = 1; + conn->has_sent_end = 1; /* no need to send end, we just got one! */ #endif -/*ENDCLOSE*/ conn->marked_for_close = 1; break; case RELAY_COMMAND_EXTEND: if(conn) { @@ -233,7 +267,8 @@ int connection_edge_process_relay_cell(cell_t *cell, circuit_t *circ, connection } log_fn(LOG_INFO,"Connected! Notifying application."); if(connection_ap_handshake_socks_reply(conn, NULL, 0, 1) < 0) { -/*ENDCLOSE*/ conn->marked_for_close = 1; + log_fn(LOG_INFO,"Writing to socks-speaking application failed. Closing."); + connection_edge_end(conn, NULL, 0, conn->cpath_layer); } break; case RELAY_COMMAND_SENDME: @@ -529,10 +564,14 @@ static int connection_ap_handshake_socks_reply(connection_t *conn, char *reply, return 0; /* if socks_version isn't 4 or 5, don't send anything */ } -/*ENDCLOSE*/ static int connection_exit_begin_conn(cell_t *cell, circuit_t *circ) { +static int connection_exit_begin_conn(cell_t *cell, circuit_t *circ) { connection_t *n_stream; char *colon; + /* XXX currently we don't send an end cell back if we drop the + * begin because it's malformed. + */ + if(!memchr(cell->payload+RELAY_HEADER_SIZE+STREAM_ID_SIZE,0, cell->length-RELAY_HEADER_SIZE-STREAM_ID_SIZE)) { log_fn(LOG_WARN,"relay begin cell has no \\0. Dropping."); @@ -578,6 +617,7 @@ static int connection_ap_handshake_socks_reply(connection_t *conn, char *reply, /* else fall through */ case -1: /* resolve failed */ log_fn(LOG_WARN,"Resolve or connect failed (%s).", n_stream->address); + connection_edge_end(n_stream, NULL, 0, NULL); connection_remove(n_stream); connection_free(n_stream); case 0: /* resolve added to pending list */ diff --git a/src/or/dns.c b/src/or/dns.c index 7c0566c80b..ac25a9dd5f 100644 --- a/src/or/dns.c +++ b/src/or/dns.c @@ -220,7 +220,7 @@ void dns_cancel_pending_resolve(char *question, connection_t *onlyconn) { /* mark all pending connections to fail */ while(resolve->pending_connections) { pend = resolve->pending_connections; -/*ENDCLOSE*/ pend->conn->marked_for_close = 1; + connection_edge_end(pend->conn, NULL, 0, NULL); resolve->pending_connections = pend->next; free(pend); } @@ -273,7 +273,7 @@ static void dns_found_answer(char *question, uint32_t answer) { pend = resolve->pending_connections; pend->conn->addr = resolve->answer; if(resolve->state == CACHE_STATE_FAILED || connection_exit_connect(pend->conn) < 0) { -/*ENDCLOSE*/ pend->conn->marked_for_close = 1; + connection_edge_end(pend->conn, NULL, 0, NULL); } resolve->pending_connections = pend->next; free(pend); diff --git a/src/or/or.h b/src/or/or.h index 639a68ff3e..45631e2720 100644 --- a/src/or/or.h +++ b/src/or/or.h @@ -310,6 +310,8 @@ struct connection_t { int done_sending; /* for half-open connections; not used currently */ int done_receiving; + char has_sent_end; /* for debugging: set once we've set the stream end, + and check in circuit_about_to_close_connection() */ }; typedef struct connection_t connection_t; @@ -581,6 +583,9 @@ void assert_connection_ok(connection_t *conn, time_t now); /********************************* connection_edge.c ***************************/ int connection_edge_process_inbuf(connection_t *conn); +void connection_edge_end(connection_t *conn, void *payload, int payload_len, + crypt_path_t *cpath_layer); + void connection_edge_send_command(connection_t *fromconn, circuit_t *circ, int relay_command, void *payload, int payload_len, crypt_path_t *cpath_layer); diff --git a/src/or/test.c b/src/or/test.c index 269022b5b3..7d692c1f35 100644 --- a/src/or/test.c +++ b/src/or/test.c @@ -46,7 +46,7 @@ setup_directory() { void test_buffers() { -#define MAX_BUF_SIZE 640*1024 +#define MAX_BUF_SIZE 1024*1024 char str[256]; char str2[256]; @@ -61,7 +61,7 @@ test_buffers() { if (!(buf = buf_new())) test_fail(); - test_eq(buf_capacity(buf), 2*1024); + test_eq(buf_capacity(buf), 512*1024); test_eq(buf_datalen(buf), 0); /**** @@ -77,7 +77,7 @@ test_buffers() { s = open("/tmp/tor_test/data", O_RDONLY, 0); eof = 0; i = read_to_buf(s, 10, buf, &eof); - test_eq(buf_capacity(buf), 2*1024); + test_eq(buf_capacity(buf), 512*1024); test_eq(buf_datalen(buf), 10); test_eq(eof, 0); test_eq(i, 10); @@ -85,7 +85,7 @@ test_buffers() { /* Test reading 0 bytes. */ i = read_to_buf(s, 0, buf, &eof); - test_eq(buf_capacity(buf), MAX_BUF_SIZE); + test_eq(buf_capacity(buf), 512*1024); test_eq(buf_datalen(buf), 10); test_eq(eof, 0); test_eq(i, 0); @@ -103,7 +103,7 @@ test_buffers() { /* Now test when buffer is filled with more data to read. */ buf2 = buf_new_with_capacity(32); i = read_to_buf(s, 128, buf2, &eof); - test_eq(buf_capacity(buf2), 32); + test_eq(buf_capacity(buf2), 128); test_eq(buf_datalen(buf2), 32); test_eq(eof, 0); test_eq(i, 32); |