Skip to content

Commit e12d814

Browse files
bagderbch
authored andcommitted
socketpair: allow EWOULDBLOCK when reading the pair check bytes
Reported-by: Gunamoi Software Co-authored-by: Jay Satiro Fixes curl#10561 Closes curl#10562
1 parent 54bdeee commit e12d814

File tree

1 file changed

+56
-5
lines changed

1 file changed

+56
-5
lines changed

lib/socketpair.c

+56-5
Original file line numberDiff line numberDiff line change
@@ -85,9 +85,22 @@ int Curl_socketpair(int domain, int type, int protocol,
8585

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

88+
#if defined(WIN32) || defined(__CYGWIN__)
89+
/* don't set SO_REUSEADDR on Windows */
90+
(void)reuse;
91+
#ifdef SO_EXCLUSIVEADDRUSE
92+
{
93+
int exclusive = 1;
94+
if(setsockopt(listener, SOL_SOCKET, SO_EXCLUSIVEADDRUSE,
95+
(char *)&exclusive, (curl_socklen_t)sizeof(exclusive)) == -1)
96+
goto error;
97+
}
98+
#endif
99+
#else
88100
if(setsockopt(listener, SOL_SOCKET, SO_REUSEADDR,
89101
(char *)&reuse, (curl_socklen_t)sizeof(reuse)) == -1)
90102
goto error;
103+
#endif
91104
if(bind(listener, &a.addr, sizeof(a.inaddr)) == -1)
92105
goto error;
93106
if(getsockname(listener, &a.addr, &addrlen) == -1 ||
@@ -113,14 +126,52 @@ int Curl_socketpair(int domain, int type, int protocol,
113126
goto error;
114127
else {
115128
struct curltime check;
116-
struct curltime now = Curl_now();
129+
struct curltime start = Curl_now();
130+
char *p = (char *)✓
131+
size_t s = sizeof(check);
117132

118133
/* write data to the socket */
119-
swrite(socks[0], &now, sizeof(now));
134+
swrite(socks[0], &start, sizeof(start));
120135
/* 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;
136+
do {
137+
ssize_t nread;
138+
139+
pfd[0].fd = socks[1];
140+
pfd[0].events = POLLIN;
141+
pfd[0].revents = 0;
142+
(void)Curl_poll(pfd, 1, 1000); /* one second */
143+
144+
nread = sread(socks[1], p, s);
145+
if(nread == -1) {
146+
int sockerr = SOCKERRNO;
147+
/* Don't block forever */
148+
if(Curl_timediff(Curl_now(), start) > (60 * 1000))
149+
goto error;
150+
if(
151+
#ifdef WSAEWOULDBLOCK
152+
/* This is how Windows does it */
153+
(WSAEWOULDBLOCK == sockerr)
154+
#else
155+
/* errno may be EWOULDBLOCK or on some systems EAGAIN when it
156+
returned due to its inability to send off data without
157+
blocking. We therefore treat both error codes the same here */
158+
(EWOULDBLOCK == sockerr) || (EAGAIN == sockerr) ||
159+
(EINTR == sockerr) || (EINPROGRESS == sockerr)
160+
#endif
161+
) {
162+
continue;
163+
}
164+
goto error;
165+
}
166+
s -= nread;
167+
if(s) {
168+
p += nread;
169+
continue;
170+
}
171+
if(memcmp(&start, &check, sizeof(check)))
172+
goto error;
173+
break;
174+
} while(1);
124175
}
125176

126177
sclose(listener);

0 commit comments

Comments
 (0)