Skip to content

Commit 7202b81

Browse files
Stefan Zagergitster
authored andcommitted
Fix potential hang in https handshake
It has been observed that curl_multi_timeout may return a very long timeout value (e.g., 294 seconds and some usec) just before curl_multi_fdset returns no file descriptors for reading. The upshot is that select() will hang for a long time -- long enough for an https handshake to be dropped. The observed behavior is that the git command will hang at the terminal and never transfer any data. This patch is a workaround for a probable bug in libcurl. The bug only seems to manifest around a very specific set of circumstances: - curl version (from curl/curlver.h): #define LIBCURL_VERSION_NUM 0x071307 - git-remote-https running on an ubuntu-lucid VM. - Connecting through squid proxy running on another VM. Interestingly, the problem doesn't manifest if a host connects through squid proxy running on localhost; only if the proxy is on a separate VM (not sure if the squid host needs to be on a separate physical machine). That would seem to suggest that this issue is timing-sensitive. This patch is more or less in line with a recommendation in the curl docs about how to behave when curl_multi_fdset doesn't return and file descriptors: http://curl.haxx.se/libcurl/c/curl_multi_fdset.html Signed-off-by: Stefan Zager <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 7e20105 commit 7202b81

File tree

1 file changed

+12
-0
lines changed

1 file changed

+12
-0
lines changed

http.c

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -631,6 +631,18 @@ void run_active_slot(struct active_request_slot *slot)
631631
FD_ZERO(&excfds);
632632
curl_multi_fdset(curlm, &readfds, &writefds, &excfds, &max_fd);
633633

634+
/*
635+
* It can happen that curl_multi_timeout returns a pathologically
636+
* long timeout when curl_multi_fdset returns no file descriptors
637+
* to read. See commit message for more details.
638+
*/
639+
if (max_fd < 0 &&
640+
(select_timeout.tv_sec > 0 ||
641+
select_timeout.tv_usec > 50000)) {
642+
select_timeout.tv_sec = 0;
643+
select_timeout.tv_usec = 50000;
644+
}
645+
634646
select(max_fd+1, &readfds, &writefds, &excfds, &select_timeout);
635647
}
636648
}

0 commit comments

Comments
 (0)