@@ -381,6 +381,11 @@ struct Curl_multi *Curl_multi_handle(int hashsize, /* socket hash */
381
381
multi -> max_concurrent_streams = 100 ;
382
382
multi -> ipv6_works = Curl_ipv6works (NULL );
383
383
384
+ #ifdef USE_WINSOCK
385
+ multi -> wsa_event = WSACreateEvent ();
386
+ if (multi -> wsa_event == WSA_INVALID_EVENT )
387
+ goto error ;
388
+ #else
384
389
#ifdef ENABLE_WAKEUP
385
390
if (Curl_socketpair (AF_UNIX , SOCK_STREAM , 0 , multi -> wakeup_pair ) < 0 ) {
386
391
multi -> wakeup_pair [0 ] = CURL_SOCKET_BAD ;
@@ -393,6 +398,7 @@ struct Curl_multi *Curl_multi_handle(int hashsize, /* socket hash */
393
398
multi -> wakeup_pair [0 ] = CURL_SOCKET_BAD ;
394
399
multi -> wakeup_pair [1 ] = CURL_SOCKET_BAD ;
395
400
}
401
+ #endif
396
402
#endif
397
403
398
404
return multi ;
@@ -1080,9 +1086,15 @@ static CURLMcode multi_wait(struct Curl_multi *multi,
1080
1086
unsigned int curlfds ;
1081
1087
long timeout_internal ;
1082
1088
int retcode = 0 ;
1089
+ #ifndef USE_WINSOCK
1083
1090
struct pollfd a_few_on_stack [NUM_POLLS_ON_STACK ];
1084
1091
struct pollfd * ufds = & a_few_on_stack [0 ];
1085
1092
bool ufds_malloc = FALSE;
1093
+ #else
1094
+ struct pollfd pre_poll ;
1095
+ WSANETWORKEVENTS wsa_events ;
1096
+ DEBUGASSERT (multi -> wsa_event != WSA_INVALID_EVENT );
1097
+ #endif
1086
1098
1087
1099
if (!GOOD_MULTI_HANDLE (multi ))
1088
1100
return CURLM_BAD_HANDLE ;
@@ -1128,11 +1140,16 @@ static CURLMcode multi_wait(struct Curl_multi *multi,
1128
1140
nfds += extra_nfds ; /* add the externally provided ones */
1129
1141
1130
1142
#ifdef ENABLE_WAKEUP
1143
+ #ifdef USE_WINSOCK
1144
+ if (use_wakeup ) {
1145
+ #else
1131
1146
if (use_wakeup && multi -> wakeup_pair [0 ] != CURL_SOCKET_BAD ) {
1147
+ #endif
1132
1148
++ nfds ;
1133
1149
}
1134
1150
#endif
1135
1151
1152
+ #ifndef USE_WINSOCK
1136
1153
if (nfds > NUM_POLLS_ON_STACK ) {
1137
1154
/* 'nfds' is a 32 bit value and 'struct pollfd' is typically 8 bytes
1138
1155
big, so at 2^29 sockets this value might wrap. When a process gets
@@ -1143,7 +1160,9 @@ static CURLMcode multi_wait(struct Curl_multi *multi,
1143
1160
return CURLM_OUT_OF_MEMORY ;
1144
1161
ufds_malloc = TRUE;
1145
1162
}
1163
+
1146
1164
nfds = 0 ;
1165
+ #endif
1147
1166
1148
1167
/* only do the second loop if we found descriptors in the first stage run
1149
1168
above */
@@ -1156,22 +1175,40 @@ static CURLMcode multi_wait(struct Curl_multi *multi,
1156
1175
1157
1176
for (i = 0 ; i < MAX_SOCKSPEREASYHANDLE ; i ++ ) {
1158
1177
curl_socket_t s = CURL_SOCKET_BAD ;
1159
-
1178
+ #ifdef USE_WINSOCK
1179
+ long mask = 0 ;
1180
+ #endif
1160
1181
if (bitmap & GETSOCK_READSOCK (i )) {
1182
+ #ifdef USE_WINSOCK
1183
+ if (timeout_ms && SOCKET_READABLE (sockbunch [i ], 0 ) > 0 )
1184
+ timeout_ms = 0 ;
1185
+ mask |= FD_READ |FD_ACCEPT |FD_CLOSE ;
1186
+ #else
1161
1187
ufds [nfds ].fd = sockbunch [i ];
1162
1188
ufds [nfds ].events = POLLIN ;
1163
1189
++ nfds ;
1190
+ #endif
1164
1191
s = sockbunch [i ];
1165
1192
}
1166
1193
if (bitmap & GETSOCK_WRITESOCK (i )) {
1194
+ #ifdef USE_WINSOCK
1195
+ if (timeout_ms && SOCKET_WRITABLE (sockbunch [i ], 0 ) > 0 )
1196
+ timeout_ms = 0 ;
1197
+ mask |= FD_WRITE |FD_CONNECT |FD_CLOSE ;
1198
+ #else
1167
1199
ufds [nfds ].fd = sockbunch [i ];
1168
1200
ufds [nfds ].events = POLLOUT ;
1169
1201
++ nfds ;
1202
+ #endif
1170
1203
s = sockbunch [i ];
1171
1204
}
1172
1205
if (s == CURL_SOCKET_BAD ) {
1173
1206
break ;
1174
1207
}
1208
+ #ifdef USE_WINSOCK
1209
+ if (WSAEventSelect (s , multi -> wsa_event , mask ) != 0 )
1210
+ return CURLM_INTERNAL_ERROR ;
1211
+ #endif
1175
1212
}
1176
1213
1177
1214
data = data -> next ; /* check next handle */
@@ -1180,6 +1217,37 @@ static CURLMcode multi_wait(struct Curl_multi *multi,
1180
1217
1181
1218
/* Add external file descriptions from poll-like struct curl_waitfd */
1182
1219
for (i = 0 ; i < extra_nfds ; i ++ ) {
1220
+ #ifdef USE_WINSOCK
1221
+ long mask = 0 ;
1222
+ extra_fds [i ].revents = 0 ;
1223
+ pre_poll .fd = extra_fds [i ].fd ;
1224
+ pre_poll .events = 0 ;
1225
+ pre_poll .revents = 0 ;
1226
+ if (extra_fds [i ].events & CURL_WAIT_POLLIN ) {
1227
+ mask |= FD_READ |FD_ACCEPT |FD_CLOSE ;
1228
+ pre_poll .events |= POLLIN ;
1229
+ }
1230
+ if (extra_fds [i ].events & CURL_WAIT_POLLPRI ) {
1231
+ mask |= FD_OOB ;
1232
+ pre_poll .events |= POLLPRI ;
1233
+ }
1234
+ if (extra_fds [i ].events & CURL_WAIT_POLLOUT ) {
1235
+ mask |= FD_WRITE |FD_CONNECT |FD_CLOSE ;
1236
+ pre_poll .events |= POLLOUT ;
1237
+ }
1238
+ if (Curl_poll (& pre_poll , 1 , 0 ) > 0 ) {
1239
+ if (pre_poll .revents & POLLIN )
1240
+ extra_fds [i ].revents |= CURL_WAIT_POLLIN ;
1241
+ if (pre_poll .revents & POLLPRI )
1242
+ extra_fds [i ].revents |= CURL_WAIT_POLLPRI ;
1243
+ if (pre_poll .revents & POLLOUT )
1244
+ extra_fds [i ].revents |= CURL_WAIT_POLLOUT ;
1245
+ if (extra_fds [i ].revents )
1246
+ timeout_ms = 0 ;
1247
+ }
1248
+ if (WSAEventSelect (extra_fds [i ].fd , multi -> wsa_event , mask ) != 0 )
1249
+ return CURLM_INTERNAL_ERROR ;
1250
+ #else
1183
1251
ufds [nfds ].fd = extra_fds [i ].fd ;
1184
1252
ufds [nfds ].events = 0 ;
1185
1253
if (extra_fds [i ].events & CURL_WAIT_POLLIN )
@@ -1189,26 +1257,57 @@ static CURLMcode multi_wait(struct Curl_multi *multi,
1189
1257
if (extra_fds [i ].events & CURL_WAIT_POLLOUT )
1190
1258
ufds [nfds ].events |= POLLOUT ;
1191
1259
++ nfds ;
1260
+ #endif
1192
1261
}
1193
1262
1194
1263
#ifdef ENABLE_WAKEUP
1264
+ #ifndef USE_WINSOCK
1195
1265
if (use_wakeup && multi -> wakeup_pair [0 ] != CURL_SOCKET_BAD ) {
1196
1266
ufds [nfds ].fd = multi -> wakeup_pair [0 ];
1197
1267
ufds [nfds ].events = POLLIN ;
1198
1268
++ nfds ;
1199
1269
}
1270
+ #endif
1200
1271
#endif
1201
1272
1202
1273
if (nfds ) {
1203
1274
/* wait... */
1275
+ #ifdef USE_WINSOCK
1276
+ WSAWaitForMultipleEvents (1 , & multi -> wsa_event , FALSE, timeout_ms , FALSE);
1277
+ #else
1204
1278
int pollrc = Curl_poll (ufds , nfds , timeout_ms );
1279
+ #endif
1280
+
1281
+ #ifdef USE_WINSOCK
1282
+ /* With Winsock, we have to run this unconditionally to call
1283
+ WSAEventSelect(fd, event, 0) on all the sockets */
1284
+ {
1285
+ retcode = 0 ;
1286
+ #else
1205
1287
if (pollrc > 0 ) {
1206
1288
retcode = pollrc ;
1289
+ #endif
1207
1290
/* copy revents results from the poll to the curl_multi_wait poll
1208
1291
struct, the bit values of the actual underlying poll() implementation
1209
1292
may not be the same as the ones in the public libcurl API! */
1210
1293
for (i = 0 ; i < extra_nfds ; i ++ ) {
1211
1294
unsigned short mask = 0 ;
1295
+ #ifdef USE_WINSOCK
1296
+ wsa_events .lNetworkEvents = 0 ;
1297
+ mask = extra_fds [i ].revents ;
1298
+ if (WSAEnumNetworkEvents (extra_fds [i ].fd , multi -> wsa_event ,
1299
+ & wsa_events ) == 0 ) {
1300
+ if (wsa_events .lNetworkEvents & (FD_READ |FD_ACCEPT |FD_CLOSE ))
1301
+ mask |= CURL_WAIT_POLLIN ;
1302
+ if (wsa_events .lNetworkEvents & (FD_WRITE |FD_CONNECT |FD_CLOSE ))
1303
+ mask |= CURL_WAIT_POLLOUT ;
1304
+ if (wsa_events .lNetworkEvents & FD_OOB )
1305
+ mask |= CURL_WAIT_POLLPRI ;
1306
+ if (ret && wsa_events .lNetworkEvents != 0 )
1307
+ retcode ++ ;
1308
+ }
1309
+ WSAEventSelect (extra_fds [i ].fd , multi -> wsa_event , 0 );
1310
+ #else
1212
1311
unsigned r = ufds [curlfds + i ].revents ;
1213
1312
1214
1313
if (r & POLLIN )
@@ -1217,9 +1316,46 @@ static CURLMcode multi_wait(struct Curl_multi *multi,
1217
1316
mask |= CURL_WAIT_POLLOUT ;
1218
1317
if (r & POLLPRI )
1219
1318
mask |= CURL_WAIT_POLLPRI ;
1319
+ #endif
1220
1320
extra_fds [i ].revents = mask ;
1221
1321
}
1222
1322
1323
+ #ifdef USE_WINSOCK
1324
+ /* Count up all our own sockets that had activity,
1325
+ and remove them from the event. */
1326
+ if (curlfds ) {
1327
+ data = multi -> easyp ;
1328
+ while (data ) {
1329
+ bitmap = multi_getsock (data , sockbunch );
1330
+
1331
+ for (i = 0 ; i < MAX_SOCKSPEREASYHANDLE ; i ++ ) {
1332
+ if (bitmap & (GETSOCK_READSOCK (i ) | GETSOCK_WRITESOCK (i ))) {
1333
+ wsa_events .lNetworkEvents = 0 ;
1334
+ if (WSAEnumNetworkEvents (sockbunch [i ], multi -> wsa_event ,
1335
+ & wsa_events ) == 0 ) {
1336
+ if (ret && wsa_events .lNetworkEvents != 0 )
1337
+ retcode ++ ;
1338
+ }
1339
+ if (ret && !timeout_ms && wsa_events .lNetworkEvents == 0 ) {
1340
+ if ((bitmap & GETSOCK_READSOCK (i )) &&
1341
+ SOCKET_READABLE (sockbunch [i ], 0 ) > 0 )
1342
+ retcode ++ ;
1343
+ else if ((bitmap & GETSOCK_WRITESOCK (i )) &&
1344
+ SOCKET_WRITABLE (sockbunch [i ], 0 ) > 0 )
1345
+ retcode ++ ;
1346
+ }
1347
+ WSAEventSelect (sockbunch [i ], multi -> wsa_event , 0 );
1348
+ }
1349
+ else
1350
+ break ;
1351
+ }
1352
+
1353
+ data = data -> next ;
1354
+ }
1355
+ }
1356
+
1357
+ WSAResetEvent (multi -> wsa_event );
1358
+ #else
1223
1359
#ifdef ENABLE_WAKEUP
1224
1360
if (use_wakeup && multi -> wakeup_pair [0 ] != CURL_SOCKET_BAD ) {
1225
1361
if (ufds [curlfds + extra_nfds ].revents & POLLIN ) {
@@ -1232,23 +1368,24 @@ static CURLMcode multi_wait(struct Curl_multi *multi,
1232
1368
when there is no more data, breaking the loop. */
1233
1369
nread = sread (multi -> wakeup_pair [0 ], buf , sizeof (buf ));
1234
1370
if (nread <= 0 ) {
1235
- #ifndef USE_WINSOCK
1236
1371
if (nread < 0 && EINTR == SOCKERRNO )
1237
1372
continue ;
1238
- #endif
1239
1373
break ;
1240
1374
}
1241
1375
}
1242
1376
/* do not count the wakeup socket into the returned value */
1243
1377
retcode -- ;
1244
1378
}
1245
1379
}
1380
+ #endif
1246
1381
#endif
1247
1382
}
1248
1383
}
1249
1384
1385
+ #ifndef USE_WINSOCK
1250
1386
if (ufds_malloc )
1251
1387
free (ufds );
1388
+ #endif
1252
1389
if (ret )
1253
1390
* ret = retcode ;
1254
1391
if (!extrawait || nfds )
@@ -1303,6 +1440,10 @@ CURLMcode curl_multi_wakeup(struct Curl_multi *multi)
1303
1440
return CURLM_BAD_HANDLE ;
1304
1441
1305
1442
#ifdef ENABLE_WAKEUP
1443
+ #ifdef USE_WINSOCK
1444
+ if (WSASetEvent (multi -> wsa_event ))
1445
+ return CURLM_OK ;
1446
+ #else
1306
1447
/* the wakeup_pair variable is only written during init and cleanup,
1307
1448
making it safe to access from another thread after the init part
1308
1449
and before cleanup */
@@ -1335,6 +1476,7 @@ CURLMcode curl_multi_wakeup(struct Curl_multi *multi)
1335
1476
return CURLM_OK ;
1336
1477
}
1337
1478
}
1479
+ #endif
1338
1480
#endif
1339
1481
return CURLM_WAKEUP_FAILURE ;
1340
1482
}
@@ -2478,9 +2620,13 @@ CURLMcode curl_multi_cleanup(struct Curl_multi *multi)
2478
2620
Curl_hash_destroy (& multi -> hostcache );
2479
2621
Curl_psl_destroy (& multi -> psl );
2480
2622
2623
+ #ifdef USE_WINSOCK
2624
+ WSACloseEvent (multi -> wsa_event );
2625
+ #else
2481
2626
#ifdef ENABLE_WAKEUP
2482
2627
sclose (multi -> wakeup_pair [0 ]);
2483
2628
sclose (multi -> wakeup_pair [1 ]);
2629
+ #endif
2484
2630
#endif
2485
2631
free (multi );
2486
2632
0 commit comments