summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNick Mathewson <nickm@torproject.org>2012-10-11 00:35:58 -0400
committerNick Mathewson <nickm@torproject.org>2012-10-11 00:35:58 -0400
commita45760b53b6aef609e3d48f9e0abb5e1550e44fb (patch)
tree59be7c1b419c09c27948e4da1511a30d34f0e95a
parent7ea904cbc0acbe1575ff68700572da76e4e4b10d (diff)
downloadtor-a45760b53b6aef609e3d48f9e0abb5e1550e44fb.tar.gz
tor-a45760b53b6aef609e3d48f9e0abb5e1550e44fb.zip
Make very sure to handle cells in-order on channels.
Fix on code for 6465, not yet in any release.
-rw-r--r--src/or/channel.c23
1 files changed, 13 insertions, 10 deletions
diff --git a/src/or/channel.c b/src/or/channel.c
index b52db405e1..6527288c4a 100644
--- a/src/or/channel.c
+++ b/src/or/channel.c
@@ -2427,6 +2427,8 @@ channel_listener_queue_incoming(channel_listener_t *listener,
void
channel_process_cells(channel_t *chan)
{
+ smartlist_t *queue;
+ int putback = 0;
tor_assert(chan);
tor_assert(chan->state == CHANNEL_STATE_CLOSING ||
chan->state == CHANNEL_STATE_MAINT ||
@@ -2442,17 +2444,20 @@ channel_process_cells(channel_t *chan)
/* Nothing we can do if we have no cells */
if (!(chan->incoming_queue)) return;
+ queue = chan->incoming_queue;
+ chan->incoming_queue = NULL;
/*
* Process cells until we're done or find one we have no current handler
* for.
*/
- SMARTLIST_FOREACH_BEGIN(chan->incoming_queue,
- cell_queue_entry_t *, q) {
+ SMARTLIST_FOREACH_BEGIN(queue, cell_queue_entry_t *, q) {
tor_assert(q);
tor_assert(q->type == CELL_QUEUE_FIXED ||
q->type == CELL_QUEUE_VAR);
- if (q->type == CELL_QUEUE_FIXED &&
+ if (putback) {
+ smartlist_add(chan->incoming_queue, q);
+ } else if (q->type == CELL_QUEUE_FIXED &&
chan->cell_handler) {
/* Handle a fixed-length cell */
tor_assert(q->u.fixed.cell);
@@ -2462,7 +2467,6 @@ channel_process_cells(channel_t *chan)
q->u.fixed.cell, chan,
U64_PRINTF_ARG(chan->global_identifier));
chan->cell_handler(chan, q->u.fixed.cell);
- SMARTLIST_DEL_CURRENT(chan->incoming_queue, q);
tor_free(q);
} else if (q->type == CELL_QUEUE_VAR &&
chan->var_cell_handler) {
@@ -2474,19 +2478,18 @@ channel_process_cells(channel_t *chan)
q->u.var.var_cell, chan,
U64_PRINTF_ARG(chan->global_identifier));
chan->var_cell_handler(chan, q->u.var.var_cell);
- SMARTLIST_DEL_CURRENT(chan->incoming_queue, q);
tor_free(q);
} else {
/* Can't handle this one */
- break;
+ if (!chan->incoming_queue)
+ chan->incoming_queue = smartlist_new();
+ smartlist_add(chan->incoming_queue, q);
+ putback = 1;
}
} SMARTLIST_FOREACH_END(q);
/* If the list is empty, free it */
- if (smartlist_len(chan->incoming_queue) == 0 ) {
- smartlist_free(chan->incoming_queue);
- chan->incoming_queue = NULL;
- }
+ smartlist_free(queue);
}
/**