@@ -85,9 +85,22 @@ int Curl_socketpair(int domain, int type, int protocol,
85
85
86
86
socks [0 ] = socks [1 ] = CURL_SOCKET_BAD ;
87
87
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
88
100
if (setsockopt (listener , SOL_SOCKET , SO_REUSEADDR ,
89
101
(char * )& reuse , (curl_socklen_t )sizeof (reuse )) == -1 )
90
102
goto error ;
103
+ #endif
91
104
if (bind (listener , & a .addr , sizeof (a .inaddr )) == -1 )
92
105
goto error ;
93
106
if (getsockname (listener , & a .addr , & addrlen ) == -1 ||
@@ -113,14 +126,52 @@ int Curl_socketpair(int domain, int type, int protocol,
113
126
goto error ;
114
127
else {
115
128
struct curltime check ;
116
- struct curltime now = Curl_now ();
129
+ struct curltime start = Curl_now ();
130
+ char * p = (char * )& check ;
131
+ size_t s = sizeof (check );
117
132
118
133
/* write data to the socket */
119
- swrite (socks [0 ], & now , sizeof (now ));
134
+ swrite (socks [0 ], & start , sizeof (start ));
120
135
/* 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 );
124
175
}
125
176
126
177
sclose (listener );
0 commit comments