1
0
mirror of https://github.com/moparisthebest/curl synced 2025-01-12 06:28:04 -05:00

multi: fix memory leak when stopped during name resolve

When the application just started the transfer and then stops it while
the name resolve in the background thread hasn't completed, we need to
wait for the resolve to complete and then cleanup data accordingly.

Enabled test 1553 again and added test 1590 to also check when the host
name resolves successfully.

Detected by OSS-fuzz.
Closes #1968
This commit is contained in:
Daniel Stenberg 2018-06-14 16:28:59 +02:00
parent 676f4b742d
commit ac9a179fe9
No known key found for this signature in database
GPG Key ID: 5CC908FDB71E12C2
7 changed files with 97 additions and 24 deletions

View File

@ -5,7 +5,7 @@
* | (__| |_| | _ <| |___ * | (__| |_| | _ <| |___
* \___|\___/|_| \_\_____| * \___|\___/|_| \_\_____|
* *
* Copyright (C) 1998 - 2017, Daniel Stenberg, <daniel@haxx.se>, et al. * Copyright (C) 1998 - 2018, Daniel Stenberg, <daniel@haxx.se>, et al.
* *
* This software is licensed as described in the file COPYING, which * This software is licensed as described in the file COPYING, which
* you should have received as part of this distribution. The terms * you should have received as part of this distribution. The terms
@ -312,22 +312,25 @@ CURLcode Curl_resolver_is_resolved(struct connectdata *conn,
conn->async.os_specific; conn->async.os_specific;
CURLcode result = CURLE_OK; CURLcode result = CURLE_OK;
if(dns)
*dns = NULL; *dns = NULL;
waitperform(conn, 0); waitperform(conn, 0);
if(res && !res->num_pending) { if(res && !res->num_pending) {
if(dns) {
(void)Curl_addrinfo_callback(conn, res->last_status, res->temp_ai); (void)Curl_addrinfo_callback(conn, res->last_status, res->temp_ai);
/* temp_ai ownership is moved to the connection, so we need not free-up /* temp_ai ownership is moved to the connection, so we need not free-up
them */ them */
res->temp_ai = NULL; res->temp_ai = NULL;
}
if(!conn->async.dns) { if(!conn->async.dns) {
failf(data, "Could not resolve: %s (%s)", failf(data, "Could not resolve: %s (%s)",
conn->async.hostname, ares_strerror(conn->async.status)); conn->async.hostname, ares_strerror(conn->async.status));
result = conn->bits.proxy?CURLE_COULDNT_RESOLVE_PROXY: result = conn->bits.proxy?CURLE_COULDNT_RESOLVE_PROXY:
CURLE_COULDNT_RESOLVE_HOST; CURLE_COULDNT_RESOLVE_HOST;
} }
else else if(dns)
*dns = conn->async.dns; *dns = conn->async.dns;
destroy_async_data(&conn->async); destroy_async_data(&conn->async);
@ -390,7 +393,7 @@ CURLcode Curl_resolver_wait_resolv(struct connectdata *conn,
timeout_ms = 1000; timeout_ms = 1000;
waitperform(conn, timeout_ms); waitperform(conn, timeout_ms);
result = Curl_resolver_is_resolved(conn, &temp_entry); result = Curl_resolver_is_resolved(conn, entry?&temp_entry:NULL);
if(result || conn->async.done) if(result || conn->async.done)
break; break;

View File

@ -481,8 +481,10 @@ CURLcode Curl_resolver_wait_resolv(struct connectdata *conn,
DEBUGASSERT(conn && td); DEBUGASSERT(conn && td);
/* wait for the thread to resolve the name */ /* wait for the thread to resolve the name */
if(Curl_thread_join(&td->thread_hnd)) if(Curl_thread_join(&td->thread_hnd)) {
if(entry)
result = getaddrinfo_complete(conn); result = getaddrinfo_complete(conn);
}
else else
DEBUGASSERT(0); DEBUGASSERT(0);

View File

@ -69,7 +69,7 @@
#define GOOD_MULTI_HANDLE(x) \ #define GOOD_MULTI_HANDLE(x) \
((x) && (x)->type == CURL_MULTI_HANDLE) ((x) && (x)->type == CURL_MULTI_HANDLE)
static void singlesocket(struct Curl_multi *multi, static CURLMcode singlesocket(struct Curl_multi *multi,
struct Curl_easy *data); struct Curl_easy *data);
static int update_timer(struct Curl_multi *multi); static int update_timer(struct Curl_multi *multi);
@ -515,6 +515,11 @@ static CURLcode multi_done(struct connectdata **connp,
/* Stop if multi_done() has already been called */ /* Stop if multi_done() has already been called */
return CURLE_OK; return CURLE_OK;
if(data->mstate == CURLM_STATE_WAITRESOLVE) {
/* still waiting for the resolve to complete */
(void)Curl_resolver_wait_resolv(conn, NULL);
}
Curl_getoff_all_pipelines(data, conn); Curl_getoff_all_pipelines(data, conn);
/* Cleanup possible redirect junk */ /* Cleanup possible redirect junk */
@ -2304,7 +2309,7 @@ CURLMsg *curl_multi_info_read(struct Curl_multi *multi, int *msgs_in_queue)
* and if we have a different state in any of those sockets from last time we * and if we have a different state in any of those sockets from last time we
* call the callback accordingly. * call the callback accordingly.
*/ */
static void singlesocket(struct Curl_multi *multi, static CURLMcode singlesocket(struct Curl_multi *multi,
struct Curl_easy *data) struct Curl_easy *data)
{ {
curl_socket_t socks[MAX_SOCKSPEREASYHANDLE]; curl_socket_t socks[MAX_SOCKSPEREASYHANDLE];
@ -2352,7 +2357,7 @@ static void singlesocket(struct Curl_multi *multi,
entry = sh_addentry(&multi->sockhash, s, data); entry = sh_addentry(&multi->sockhash, s, data);
if(!entry) if(!entry)
/* fatal */ /* fatal */
return; return CURLM_OUT_OF_MEMORY;
} }
/* we know (entry != NULL) at this point, see the logic above */ /* we know (entry != NULL) at this point, see the logic above */
@ -2440,6 +2445,7 @@ static void singlesocket(struct Curl_multi *multi,
memcpy(data->sockets, socks, num*sizeof(curl_socket_t)); memcpy(data->sockets, socks, num*sizeof(curl_socket_t));
data->numsocks = num; data->numsocks = num;
return CURLM_OK;
} }
void Curl_updatesocket(struct Curl_easy *data) void Curl_updatesocket(struct Curl_easy *data)
@ -2553,8 +2559,8 @@ static CURLMcode multi_socket(struct Curl_multi *multi,
and callbacks */ and callbacks */
if(result != CURLM_BAD_HANDLE) { if(result != CURLM_BAD_HANDLE) {
data = multi->easyp; data = multi->easyp;
while(data) { while(data && !result) {
singlesocket(multi, data); result = singlesocket(multi, data);
data = data->next; data = data->next;
} }
} }
@ -2608,10 +2614,13 @@ static CURLMcode multi_socket(struct Curl_multi *multi,
/* clear the bitmask only if not locked */ /* clear the bitmask only if not locked */
data->easy_conn->cselect_bits = 0; data->easy_conn->cselect_bits = 0;
if(CURLM_OK >= result) if(CURLM_OK >= result) {
/* get the socket(s) and check if the state has been changed since /* get the socket(s) and check if the state has been changed since
last */ last */
singlesocket(multi, data); result = singlesocket(multi, data);
if(result)
return result;
}
/* Now we fall-through and do the timer-based stuff, since we don't want /* Now we fall-through and do the timer-based stuff, since we don't want
to force the user to have to deal with timeouts as long as at least to force the user to have to deal with timeouts as long as at least
@ -2645,10 +2654,13 @@ static CURLMcode multi_socket(struct Curl_multi *multi,
result = multi_runsingle(multi, now, data); result = multi_runsingle(multi, now, data);
sigpipe_restore(&pipe_st); sigpipe_restore(&pipe_st);
if(CURLM_OK >= result) if(CURLM_OK >= result) {
/* get the socket(s) and check if the state has been changed since /* get the socket(s) and check if the state has been changed since
last */ last */
singlesocket(multi, data); result = singlesocket(multi, data);
if(result)
return result;
}
} }
/* Check if there's one (more) expired timer to deal with! This function /* Check if there's one (more) expired timer to deal with! This function

View File

@ -18,5 +18,3 @@
1510 1510
# Pipelining test that is causing false positives a little too often # Pipelining test that is causing false positives a little too often
1903 1903
# causes memory leaks for now:
1553

View File

@ -176,6 +176,8 @@ test1525 test1526 test1527 test1528 test1529 test1530 test1531 test1532 \
test1533 test1534 test1535 test1536 test1537 test1538 \ test1533 test1534 test1535 test1536 test1537 test1538 \
test1540 \ test1540 \
test1550 test1551 test1552 test1553 test1554 test1555 test1556 \ test1550 test1551 test1552 test1553 test1554 test1555 test1556 \
\
test1590 \
test1600 test1601 test1602 test1603 test1604 test1605 test1606 test1607 \ test1600 test1601 test1602 test1603 test1604 test1605 test1606 test1607 \
test1608 test1609 \ test1608 test1609 \
\ \

View File

@ -38,6 +38,8 @@ IMAP cleanup before a connection was created
<tool> <tool>
lib1553 lib1553
</tool> </tool>
# this MUST use a host name that doesn't resolve
<command> <command>
imap://non-existing-host.haxx.se:%IMAPPORT/1553 imap://non-existing-host.haxx.se:%IMAPPORT/1553
</command> </command>

54
tests/data/test1590 Normal file
View File

@ -0,0 +1,54 @@
<testcase>
<info>
<keywords>
IMAP
Clear Text
FETCH
</keywords>
</info>
#
# Server-side
<reply>
<data>
From: me@somewhere
To: fake@nowhere
body
--
yours sincerely
</data>
<datacheck>
</datacheck>
<servercmd>
</servercmd>
</reply>
#
# Client-side
<client>
<server>
imap
</server>
<name>
IMAP cleanup before a connection was created
</name>
# tool is what to use instead of 'curl'
<tool>
lib1553
</tool>
# it is important this uses a host name that resolves successfully
<command>
imap://localhost:%IMAPPORT/1590
</command>
</client>
#
# Verify data after the test has been "shot"
<verify>
<protocol>
</protocol>
</verify>
</testcase>