diff options
author | Steinar H. Gunderson <sesse@google.com> | 2007-09-29 18:18:47 +0000 |
---|---|---|
committer | Steinar H. Gunderson <sesse@google.com> | 2007-09-29 18:18:47 +0000 |
commit | 23f5d145ec69e2a8b62e75cc49b591fef04ca48d (patch) | |
tree | 3bbbeb0269bb896342bc24b839d7b5ac966bf6f7 /ares/ares_send.c | |
parent | b01ab65225dec292919f8123ef2557d0a94e4569 (diff) | |
download | curl-23f5d145ec69e2a8b62e75cc49b591fef04ca48d.tar.gz |
Previously, processing a large batch of timeouts was O(n^2) in the number of
outstanding queries, and processing a DNS response packet was O(n) in the
number of outstanding queries. To speed things up in Google, we added a few circular,
doubly-linked lists of queries that are hash-bucketed based on
the attributes we care about, so most important operations are now O(1).
It might be that the number of buckets are higher than most people would need,
but on a quick calculation it should only be 100kB or so even on a 64-bit
system, so I've let it stay as-is.
Diffstat (limited to 'ares/ares_send.c')
-rw-r--r-- | ares/ares_send.c | 17 |
1 files changed, 14 insertions, 3 deletions
diff --git a/ares/ares_send.c b/ares/ares_send.c index c8015803b..2007321f9 100644 --- a/ares/ares_send.c +++ b/ares/ares_send.c @@ -102,9 +102,20 @@ void ares_send(ares_channel channel, const unsigned char *qbuf, int qlen, query->error_status = ARES_ECONNREFUSED; query->timeouts = 0; - /* Chain the query into this channel's query list. */ - query->next = channel->queries; - channel->queries = query; + /* Initialize our list nodes. */ + ares__init_list_node(&(query->queries_by_qid), query); + ares__init_list_node(&(query->queries_by_timeout), query); + ares__init_list_node(&(query->queries_to_server), query); + ares__init_list_node(&(query->all_queries), query); + + /* Chain the query into the list of all queries. */ + ares__insert_in_list(&(query->all_queries), &(channel->all_queries)); + /* Keep track of queries bucketed by qid, so we can process DNS + * responses quickly. + */ + ares__insert_in_list( + &(query->queries_by_qid), + &(channel->queries_by_qid[query->qid % ARES_QID_TABLE_SIZE])); /* Perform the first query action. */ time(&now); |