Skip to content

Commit 88c67b4

Browse files
committed
DEBUG windows socketpair using repeat.bat
1. open sysinternals' dbgview 2. edit repeat.bat var CURL to path of a repo non-debug build curl.exe 3. run repeat.bat. should repro in less than 5 min, may take 20 min, stops after 30-60 min. ~~~ curl: (6) getaddrinfo() thread failed to start 0.000000 Reproduced in 738 tries. Ran from 15:58:20.66 to 16:00:18.40. ~~~ Ref: curl#10561 (comment)
1 parent 41dfb7f commit 88c67b4

File tree

4 files changed

+85
-12
lines changed

4 files changed

+85
-12
lines changed

lib/asyn-thread.c

+7
Original file line numberDiff line numberDiff line change
@@ -230,13 +230,15 @@ int init_thread_sync_data(struct thread_data *td,
230230
if(!tsd->mtx)
231231
goto err_exit;
232232

233+
OutputDebugStringA("marker1");
233234
Curl_mutex_init(tsd->mtx);
234235

235236
#ifndef CURL_DISABLE_SOCKETPAIR
236237
/* create socket pair, avoid AF_LOCAL since it doesn't build on Solaris */
237238
if(Curl_socketpair(AF_UNIX, SOCK_STREAM, 0, &tsd->sock_pair[0]) < 0) {
238239
tsd->sock_pair[0] = CURL_SOCKET_BAD;
239240
tsd->sock_pair[1] = CURL_SOCKET_BAD;
241+
OutputDebugStringA("marker2");
240242
goto err_exit;
241243
}
242244
#endif
@@ -245,6 +247,7 @@ int init_thread_sync_data(struct thread_data *td,
245247
/* Copying hostname string because original can be destroyed by parent
246248
* thread during gethostbyname execution.
247249
*/
250+
OutputDebugStringA("marker3");
248251
tsd->hostname = strdup(hostname);
249252
if(!tsd->hostname)
250253
goto err_exit;
@@ -258,6 +261,7 @@ int init_thread_sync_data(struct thread_data *td,
258261
tsd->sock_pair[0] = CURL_SOCKET_BAD;
259262
}
260263
#endif
264+
OutputDebugStringA("marker4");
261265
destroy_thread_sync_data(tsd);
262266
return 0;
263267
}
@@ -440,6 +444,8 @@ static bool init_resolve_thread(struct Curl_easy *data,
440444
asp->dns = NULL;
441445
td->thread_hnd = curl_thread_t_null;
442446

447+
OutputDebugStringA(aprintf("::init_resolve_thread"));
448+
443449
if(!init_thread_sync_data(td, hostname, port, hints)) {
444450
asp->tdata = NULL;
445451
free(td);
@@ -456,6 +462,7 @@ static bool init_resolve_thread(struct Curl_easy *data,
456462

457463
#ifdef HAVE_GETADDRINFO
458464
td->thread_hnd = Curl_thread_create(getaddrinfo_thread, &td->tsd);
465+
OutputDebugStringA(aprintf("td->thread_hnd: %p", td->thread_hnd));
459466
#else
460467
td->thread_hnd = Curl_thread_create(gethostbyname_thread, &td->tsd);
461468
#endif

lib/curl_threads.c

+4
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,8 @@
3333
#elif defined(USE_THREADS_WIN32)
3434
# include <process.h>
3535
#endif
36+
#include "curl_printf.h"
37+
3638

3739
#include "curl_threads.h"
3840
#include "curl_memory.h"
@@ -117,6 +119,7 @@ curl_thread_t Curl_thread_create(unsigned int (CURL_STDCALL *func) (void *),
117119
thread_handle = CreateThread(NULL, 0, func, arg, 0, NULL);
118120
#else
119121
thread_handle = _beginthreadex(NULL, 0, func, arg, 0, NULL);
122+
OutputDebugStringA(aprintf("thread_handle: %p", thread_handle));
120123
#endif
121124
t = (curl_thread_t)thread_handle;
122125
if((t == 0) || (t == LongToHandle(-1L))) {
@@ -126,6 +129,7 @@ curl_thread_t Curl_thread_create(unsigned int (CURL_STDCALL *func) (void *),
126129
gle == ERROR_NOT_ENOUGH_MEMORY) ?
127130
EACCES : EINVAL);
128131
#endif
132+
OutputDebugStringA(aprintf("err t: %p", t));
129133
return curl_thread_t_null;
130134
}
131135
return t;

lib/socketpair.c

+39-12
Original file line numberDiff line numberDiff line change
@@ -85,42 +85,69 @@ int Curl_socketpair(int domain, int type, int protocol,
8585

8686
socks[0] = socks[1] = CURL_SOCKET_BAD;
8787

88+
#define xstr(s) str(s)
89+
#define str(s) #s
90+
#define GOTO_ERROR \
91+
do { \
92+
OutputDebugStringA("error - line " ## xstr(__LINE__)); \
93+
goto error; \
94+
} while(0)
95+
8896
if(setsockopt(listener, SOL_SOCKET, SO_REUSEADDR,
8997
(char *)&reuse, (curl_socklen_t)sizeof(reuse)) == -1)
90-
goto error;
98+
GOTO_ERROR;
9199
if(bind(listener, &a.addr, sizeof(a.inaddr)) == -1)
92-
goto error;
100+
GOTO_ERROR;
93101
if(getsockname(listener, &a.addr, &addrlen) == -1 ||
94102
addrlen < (int)sizeof(a.inaddr))
95-
goto error;
103+
GOTO_ERROR;
96104
if(listen(listener, 1) == -1)
97-
goto error;
105+
GOTO_ERROR;
98106
socks[0] = socket(AF_INET, SOCK_STREAM, 0);
99107
if(socks[0] == CURL_SOCKET_BAD)
100-
goto error;
108+
GOTO_ERROR;
101109
if(connect(socks[0], &a.addr, sizeof(a.inaddr)) == -1)
102-
goto error;
110+
GOTO_ERROR;
103111

104112
/* use non-blocking accept to make sure we don't block forever */
105113
if(curlx_nonblock(listener, TRUE) < 0)
106-
goto error;
114+
GOTO_ERROR;
107115
pfd[0].fd = listener;
108116
pfd[0].events = POLLIN;
109117
pfd[0].revents = 0;
110118
(void)Curl_poll(pfd, 1, 1000); /* one second */
111119
socks[1] = accept(listener, NULL, NULL);
112120
if(socks[1] == CURL_SOCKET_BAD)
113-
goto error;
121+
GOTO_ERROR;
114122
else {
115123
struct curltime check;
116124
struct curltime now = Curl_now();
125+
ssize_t out, in;
117126

118127
/* write data to the socket */
119-
swrite(socks[0], &now, sizeof(now));
128+
WSASetLastError(0);
129+
out = swrite(socks[0], &now, sizeof(now));
130+
if(out != sizeof(now)) {
131+
if(out == SOCKET_ERROR)
132+
OutputDebugStringA(aprintf("send gle: %d", WSAGetLastError()));
133+
else
134+
OutputDebugStringA(aprintf("out: %zd", out));
135+
GOTO_ERROR;
136+
}
137+
120138
/* verify that we read the correct data */
121-
if((sizeof(now) != sread(socks[1], &check, sizeof(check)) ||
122-
memcmp(&now, &check, sizeof(check))))
123-
goto error;
139+
WSASetLastError(0);
140+
in = sread(socks[1], &check, sizeof(check));
141+
if(sizeof(now) != in) {
142+
if(in == SOCKET_ERROR)
143+
OutputDebugStringA(aprintf("recv gle: %d", WSAGetLastError()));
144+
else
145+
OutputDebugStringA(aprintf("in: %zd", in));
146+
GOTO_ERROR;
147+
}
148+
149+
if(memcmp(&now, &check, sizeof(check)))
150+
GOTO_ERROR;
124151
}
125152

126153
sclose(listener);

repeat.bat

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
@echo off
2+
setlocal ENABLEEXTENSIONS ENABLEDELAYEDEXPANSION
3+
4+
REM set this as the path to your curl
5+
set "CURL=X:\j\curl\curl\build\Win32\VC10\DLL Release - DLL Windows SSPI - DLL WinIDN\curl.exe"
6+
7+
set START_TIME=%TIME%
8+
9+
REM 20000 is approx 30-60 min of repeated transfers to http://curl.se
10+
set MAX=20000
11+
12+
echo Use Sysinternals dbgview to collect debug output while this runs.
13+
timeout 5 /nobreak
14+
15+
REM note the host here is purposely http://curl.se so the command does a real
16+
REM resolve but then completes faster, since http://curl.se just serves a
17+
REM redirect that is not followed.
18+
19+
FOR /L %%G IN (1,1,%MAX%) DO (
20+
"%CURL%" -sS -f http://curl.se -o NUL --write-out "%%{time_total} "
21+
if !ERRORLEVEL! NEQ 0 (
22+
echo.
23+
if !ERRORLEVEL! EQU 6 (
24+
echo Reproduced in %%G tries.
25+
) else (
26+
echo Unexpected error code !ERRORLEVEL!, likely not a repro.
27+
)
28+
goto bye
29+
)
30+
)
31+
32+
echo Failed to reproduce the bug with %MAX% tries.
33+
34+
:bye
35+
echo Ran from %START_TIME% to !TIME!.

0 commit comments

Comments
 (0)