• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /***************************************************************************
2  *                                  _   _ ____  _
3  *  Project                     ___| | | |  _ \| |
4  *                             / __| | | | |_) | |
5  *                            | (__| |_| |  _ <| |___
6  *                             \___|\___/|_| \_\_____|
7  *
8  * Copyright (C) 1998 - 2016, Daniel Stenberg, <daniel@haxx.se>, et al.
9  *
10  * This software is licensed as described in the file COPYING, which
11  * you should have received as part of this distribution. The terms
12  * are also available at https://curl.haxx.se/docs/copyright.html.
13  *
14  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15  * copies of the Software, and permit persons to whom the Software is
16  * furnished to do so, under the terms of the COPYING file.
17  *
18  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19  * KIND, either express or implied.
20  *
21  ***************************************************************************/
22 
23 #include "curl_setup.h"
24 
25 #include <curl/curl.h>
26 
27 #include "urldata.h"
28 #include "transfer.h"
29 #include "url.h"
30 #include "connect.h"
31 #include "progress.h"
32 #include "easyif.h"
33 #include "share.h"
34 #include "multiif.h"
35 #include "sendf.h"
36 #include "timeval.h"
37 #include "http.h"
38 #include "select.h"
39 #include "warnless.h"
40 #include "speedcheck.h"
41 #include "conncache.h"
42 #include "multihandle.h"
43 #include "pipeline.h"
44 #include "sigpipe.h"
45 /* The last 3 #include files should be in this order */
46 #include "curl_printf.h"
47 #include "curl_memory.h"
48 #include "memdebug.h"
49 
50 /*
51   CURL_SOCKET_HASH_TABLE_SIZE should be a prime number. Increasing it from 97
52   to 911 takes on a 32-bit machine 4 x 804 = 3211 more bytes.  Still, every
53   CURL handle takes 45-50 K memory, therefore this 3K are not significant.
54 */
55 #ifndef CURL_SOCKET_HASH_TABLE_SIZE
56 #define CURL_SOCKET_HASH_TABLE_SIZE 911
57 #endif
58 
59 #define CURL_CONNECTION_HASH_SIZE 97
60 
61 #define CURL_MULTI_HANDLE 0x000bab1e
62 
63 #define GOOD_MULTI_HANDLE(x) \
64   ((x) && (x)->type == CURL_MULTI_HANDLE)
65 
66 static void singlesocket(struct Curl_multi *multi,
67                          struct Curl_easy *data);
68 static int update_timer(struct Curl_multi *multi);
69 
70 static CURLMcode add_next_timeout(struct timeval now,
71                                   struct Curl_multi *multi,
72                                   struct Curl_easy *d);
73 static CURLMcode multi_timeout(struct Curl_multi *multi,
74                                long *timeout_ms);
75 
76 #ifdef DEBUGBUILD
77 static const char * const statename[]={
78   "INIT",
79   "CONNECT_PEND",
80   "CONNECT",
81   "WAITRESOLVE",
82   "WAITCONNECT",
83   "WAITPROXYCONNECT",
84   "SENDPROTOCONNECT",
85   "PROTOCONNECT",
86   "WAITDO",
87   "DO",
88   "DOING",
89   "DO_MORE",
90   "DO_DONE",
91   "WAITPERFORM",
92   "PERFORM",
93   "TOOFAST",
94   "DONE",
95   "COMPLETED",
96   "MSGSENT",
97 };
98 #endif
99 
100 static void multi_freetimeout(void *a, void *b);
101 
102 /* function pointer called once when switching TO a state */
103 typedef void (*init_multistate_func)(struct Curl_easy *data);
104 
105 /* always use this function to change state, to make debugging easier */
mstate(struct Curl_easy * data,CURLMstate state,int lineno)106 static void mstate(struct Curl_easy *data, CURLMstate state
107 #ifdef DEBUGBUILD
108                    , int lineno
109 #endif
110 )
111 {
112   CURLMstate oldstate = data->mstate;
113   static const init_multistate_func finit[CURLM_STATE_LAST] = {
114     NULL,
115     NULL,
116     Curl_init_CONNECT, /* CONNECT */
117     /* the rest is NULL too */
118   };
119 
120 #if defined(DEBUGBUILD) && defined(CURL_DISABLE_VERBOSE_STRINGS)
121   (void) lineno;
122 #endif
123 
124   if(oldstate == state)
125     /* don't bother when the new state is the same as the old state */
126     return;
127 
128   data->mstate = state;
129 
130 #if defined(DEBUGBUILD) && !defined(CURL_DISABLE_VERBOSE_STRINGS)
131   if(data->mstate >= CURLM_STATE_CONNECT_PEND &&
132      data->mstate < CURLM_STATE_COMPLETED) {
133     long connection_id = -5000;
134 
135     if(data->easy_conn)
136       connection_id = data->easy_conn->connection_id;
137 
138     infof(data,
139           "STATE: %s => %s handle %p; line %d (connection #%ld)\n",
140           statename[oldstate], statename[data->mstate],
141           (void *)data, lineno, connection_id);
142   }
143 #endif
144 
145   if(state == CURLM_STATE_COMPLETED)
146     /* changing to COMPLETED means there's one less easy handle 'alive' */
147     data->multi->num_alive--;
148 
149   /* if this state has an init-function, run it */
150   if(finit[state])
151     finit[state](data);
152 }
153 
154 #ifndef DEBUGBUILD
155 #define multistate(x,y) mstate(x,y)
156 #else
157 #define multistate(x,y) mstate(x,y, __LINE__)
158 #endif
159 
160 /*
161  * We add one of these structs to the sockhash for a particular socket
162  */
163 
164 struct Curl_sh_entry {
165   struct Curl_easy *easy;
166   int action;  /* what action READ/WRITE this socket waits for */
167   curl_socket_t socket; /* mainly to ease debugging */
168   void *socketp; /* settable by users with curl_multi_assign() */
169 };
170 /* bits for 'action' having no bits means this socket is not expecting any
171    action */
172 #define SH_READ  1
173 #define SH_WRITE 2
174 
175 /* look up a given socket in the socket hash, skip invalid sockets */
sh_getentry(struct curl_hash * sh,curl_socket_t s)176 static struct Curl_sh_entry *sh_getentry(struct curl_hash *sh,
177                                          curl_socket_t s)
178 {
179   if(s != CURL_SOCKET_BAD)
180     /* only look for proper sockets */
181     return Curl_hash_pick(sh, (char *)&s, sizeof(curl_socket_t));
182   return NULL;
183 }
184 
185 /* make sure this socket is present in the hash for this handle */
sh_addentry(struct curl_hash * sh,curl_socket_t s,struct Curl_easy * data)186 static struct Curl_sh_entry *sh_addentry(struct curl_hash *sh,
187                                          curl_socket_t s,
188                                          struct Curl_easy *data)
189 {
190   struct Curl_sh_entry *there = sh_getentry(sh, s);
191   struct Curl_sh_entry *check;
192 
193   if(there)
194     /* it is present, return fine */
195     return there;
196 
197   /* not present, add it */
198   check = calloc(1, sizeof(struct Curl_sh_entry));
199   if(!check)
200     return NULL; /* major failure */
201 
202   check->easy = data;
203   check->socket = s;
204 
205   /* make/add new hash entry */
206   if(!Curl_hash_add(sh, (char *)&s, sizeof(curl_socket_t), check)) {
207     free(check);
208     return NULL; /* major failure */
209   }
210 
211   return check; /* things are good in sockhash land */
212 }
213 
214 
215 /* delete the given socket + handle from the hash */
sh_delentry(struct curl_hash * sh,curl_socket_t s)216 static void sh_delentry(struct curl_hash *sh, curl_socket_t s)
217 {
218   /* We remove the hash entry. This will end up in a call to
219      sh_freeentry(). */
220   Curl_hash_delete(sh, (char *)&s, sizeof(curl_socket_t));
221 }
222 
223 /*
224  * free a sockhash entry
225  */
sh_freeentry(void * freethis)226 static void sh_freeentry(void *freethis)
227 {
228   struct Curl_sh_entry *p = (struct Curl_sh_entry *) freethis;
229 
230   free(p);
231 }
232 
fd_key_compare(void * k1,size_t k1_len,void * k2,size_t k2_len)233 static size_t fd_key_compare(void *k1, size_t k1_len, void *k2, size_t k2_len)
234 {
235   (void) k1_len; (void) k2_len;
236 
237   return (*((curl_socket_t *) k1)) == (*((curl_socket_t *) k2));
238 }
239 
hash_fd(void * key,size_t key_length,size_t slots_num)240 static size_t hash_fd(void *key, size_t key_length, size_t slots_num)
241 {
242   curl_socket_t fd = *((curl_socket_t *) key);
243   (void) key_length;
244 
245   return (fd % slots_num);
246 }
247 
248 /*
249  * sh_init() creates a new socket hash and returns the handle for it.
250  *
251  * Quote from README.multi_socket:
252  *
253  * "Some tests at 7000 and 9000 connections showed that the socket hash lookup
254  * is somewhat of a bottle neck. Its current implementation may be a bit too
255  * limiting. It simply has a fixed-size array, and on each entry in the array
256  * it has a linked list with entries. So the hash only checks which list to
257  * scan through. The code I had used so for used a list with merely 7 slots
258  * (as that is what the DNS hash uses) but with 7000 connections that would
259  * make an average of 1000 nodes in each list to run through. I upped that to
260  * 97 slots (I believe a prime is suitable) and noticed a significant speed
261  * increase.  I need to reconsider the hash implementation or use a rather
262  * large default value like this. At 9000 connections I was still below 10us
263  * per call."
264  *
265  */
sh_init(struct curl_hash * hash,int hashsize)266 static int sh_init(struct curl_hash *hash, int hashsize)
267 {
268   return Curl_hash_init(hash, hashsize, hash_fd, fd_key_compare,
269                         sh_freeentry);
270 }
271 
272 /*
273  * multi_addmsg()
274  *
275  * Called when a transfer is completed. Adds the given msg pointer to
276  * the list kept in the multi handle.
277  */
multi_addmsg(struct Curl_multi * multi,struct Curl_message * msg)278 static CURLMcode multi_addmsg(struct Curl_multi *multi,
279                               struct Curl_message *msg)
280 {
281   if(!Curl_llist_insert_next(multi->msglist, multi->msglist->tail, msg))
282     return CURLM_OUT_OF_MEMORY;
283 
284   return CURLM_OK;
285 }
286 
287 /*
288  * multi_freeamsg()
289  *
290  * Callback used by the llist system when a single list entry is destroyed.
291  */
multi_freeamsg(void * a,void * b)292 static void multi_freeamsg(void *a, void *b)
293 {
294   (void)a;
295   (void)b;
296 }
297 
Curl_multi_handle(int hashsize,int chashsize)298 struct Curl_multi *Curl_multi_handle(int hashsize, /* socket hash */
299                                      int chashsize) /* connection hash */
300 {
301   struct Curl_multi *multi = calloc(1, sizeof(struct Curl_multi));
302 
303   if(!multi)
304     return NULL;
305 
306   multi->type = CURL_MULTI_HANDLE;
307 
308   if(Curl_mk_dnscache(&multi->hostcache))
309     goto error;
310 
311   if(sh_init(&multi->sockhash, hashsize))
312     goto error;
313 
314   if(Curl_conncache_init(&multi->conn_cache, chashsize))
315     goto error;
316 
317   multi->msglist = Curl_llist_alloc(multi_freeamsg);
318   if(!multi->msglist)
319     goto error;
320 
321   multi->pending = Curl_llist_alloc(multi_freeamsg);
322   if(!multi->pending)
323     goto error;
324 
325   /* allocate a new easy handle to use when closing cached connections */
326   multi->closure_handle = curl_easy_init();
327   if(!multi->closure_handle)
328     goto error;
329 
330   multi->closure_handle->multi = multi;
331   multi->closure_handle->state.conn_cache = &multi->conn_cache;
332 
333   multi->max_pipeline_length = 5;
334 
335   /* -1 means it not set by user, use the default value */
336   multi->maxconnects = -1;
337   return multi;
338 
339   error:
340 
341   Curl_hash_destroy(&multi->sockhash);
342   Curl_hash_destroy(&multi->hostcache);
343   Curl_conncache_destroy(&multi->conn_cache);
344   Curl_close(multi->closure_handle);
345   multi->closure_handle = NULL;
346   Curl_llist_destroy(multi->msglist, NULL);
347   Curl_llist_destroy(multi->pending, NULL);
348 
349   free(multi);
350   return NULL;
351 }
352 
curl_multi_init(void)353 struct Curl_multi *curl_multi_init(void)
354 {
355   return Curl_multi_handle(CURL_SOCKET_HASH_TABLE_SIZE,
356                            CURL_CONNECTION_HASH_SIZE);
357 }
358 
curl_multi_add_handle(struct Curl_multi * multi,struct Curl_easy * data)359 CURLMcode curl_multi_add_handle(struct Curl_multi *multi,
360                                 struct Curl_easy *data)
361 {
362   struct curl_llist *timeoutlist;
363 
364   /* First, make some basic checks that the CURLM handle is a good handle */
365   if(!GOOD_MULTI_HANDLE(multi))
366     return CURLM_BAD_HANDLE;
367 
368   /* Verify that we got a somewhat good easy handle too */
369   if(!GOOD_EASY_HANDLE(data))
370     return CURLM_BAD_EASY_HANDLE;
371 
372   /* Prevent users from adding same easy handle more than once and prevent
373      adding to more than one multi stack */
374   if(data->multi)
375     return CURLM_ADDED_ALREADY;
376 
377   /* Allocate and initialize timeout list for easy handle */
378   timeoutlist = Curl_llist_alloc(multi_freetimeout);
379   if(!timeoutlist)
380     return CURLM_OUT_OF_MEMORY;
381 
382   /*
383    * No failure allowed in this function beyond this point. And no
384    * modification of easy nor multi handle allowed before this except for
385    * potential multi's connection cache growing which won't be undone in this
386    * function no matter what.
387    */
388 
389   /* Make easy handle use timeout list initialized above */
390   data->state.timeoutlist = timeoutlist;
391   timeoutlist = NULL;
392 
393   /* set the easy handle */
394   multistate(data, CURLM_STATE_INIT);
395 
396   if((data->set.global_dns_cache) &&
397      (data->dns.hostcachetype != HCACHE_GLOBAL)) {
398     /* global dns cache was requested but still isn't */
399     struct curl_hash *global = Curl_global_host_cache_init();
400     if(global) {
401       /* only do this if the global cache init works */
402       data->dns.hostcache = global;
403       data->dns.hostcachetype = HCACHE_GLOBAL;
404     }
405   }
406   /* for multi interface connections, we share DNS cache automatically if the
407      easy handle's one is currently not set. */
408   else if(!data->dns.hostcache ||
409      (data->dns.hostcachetype == HCACHE_NONE)) {
410     data->dns.hostcache = &multi->hostcache;
411     data->dns.hostcachetype = HCACHE_MULTI;
412   }
413 
414   /* Point to the multi's connection cache */
415   data->state.conn_cache = &multi->conn_cache;
416 
417   /* This adds the new entry at the 'end' of the doubly-linked circular
418      list of Curl_easy structs to try and maintain a FIFO queue so
419      the pipelined requests are in order. */
420 
421   /* We add this new entry last in the list. */
422 
423   data->next = NULL; /* end of the line */
424   if(multi->easyp) {
425     struct Curl_easy *last = multi->easylp;
426     last->next = data;
427     data->prev = last;
428     multi->easylp = data; /* the new last node */
429   }
430   else {
431     /* first node, make prev NULL! */
432     data->prev = NULL;
433     multi->easylp = multi->easyp = data; /* both first and last */
434   }
435 
436   /* make the Curl_easy refer back to this multi handle */
437   data->multi = multi;
438 
439   /* Set the timeout for this handle to expire really soon so that it will
440      be taken care of even when this handle is added in the midst of operation
441      when only the curl_multi_socket() API is used. During that flow, only
442      sockets that time-out or have actions will be dealt with. Since this
443      handle has no action yet, we make sure it times out to get things to
444      happen. */
445   Curl_expire(data, 1);
446 
447   /* increase the node-counter */
448   multi->num_easy++;
449 
450   /* increase the alive-counter */
451   multi->num_alive++;
452 
453   /* A somewhat crude work-around for a little glitch in update_timer() that
454      happens if the lastcall time is set to the same time when the handle is
455      removed as when the next handle is added, as then the check in
456      update_timer() that prevents calling the application multiple times with
457      the same timer infor will not trigger and then the new handle's timeout
458      will not be notified to the app.
459 
460      The work-around is thus simply to clear the 'lastcall' variable to force
461      update_timer() to always trigger a callback to the app when a new easy
462      handle is added */
463   memset(&multi->timer_lastcall, 0, sizeof(multi->timer_lastcall));
464 
465   update_timer(multi);
466   return CURLM_OK;
467 }
468 
469 #if 0
470 /* Debug-function, used like this:
471  *
472  * Curl_hash_print(multi->sockhash, debug_print_sock_hash);
473  *
474  * Enable the hash print function first by editing hash.c
475  */
476 static void debug_print_sock_hash(void *p)
477 {
478   struct Curl_sh_entry *sh = (struct Curl_sh_entry *)p;
479 
480   fprintf(stderr, " [easy %p/magic %x/socket %d]",
481           (void *)sh->data, sh->data->magic, (int)sh->socket);
482 }
483 #endif
484 
485 /* Mark the connection as 'idle', or close it if the cache is full.
486    Returns TRUE if the connection is kept, or FALSE if it was closed. */
487 static bool
ConnectionDone(struct Curl_easy * data,struct connectdata * conn)488 ConnectionDone(struct Curl_easy *data, struct connectdata *conn)
489 {
490   /* data->multi->maxconnects can be negative, deal with it. */
491   size_t maxconnects =
492     (data->multi->maxconnects < 0) ? data->multi->num_easy * 4:
493     data->multi->maxconnects;
494   struct connectdata *conn_candidate = NULL;
495 
496   /* Mark the current connection as 'unused' */
497   conn->inuse = FALSE;
498 
499   if(maxconnects > 0 &&
500      data->state.conn_cache->num_connections > maxconnects) {
501     infof(data, "Connection cache is full, closing the oldest one.\n");
502 
503     conn_candidate = Curl_oldest_idle_connection(data);
504 
505     if(conn_candidate) {
506       /* Set the connection's owner correctly */
507       conn_candidate->data = data;
508 
509       /* the winner gets the honour of being disconnected */
510       (void)Curl_disconnect(conn_candidate, /* dead_connection */ FALSE);
511     }
512   }
513 
514   return (conn_candidate == conn) ? FALSE : TRUE;
515 }
516 
multi_done(struct connectdata ** connp,CURLcode status,bool premature)517 static CURLcode multi_done(struct connectdata **connp,
518                           CURLcode status,  /* an error if this is called
519                                                after an error was detected */
520                           bool premature)
521 {
522   CURLcode result;
523   struct connectdata *conn;
524   struct Curl_easy *data;
525 
526   DEBUGASSERT(*connp);
527 
528   conn = *connp;
529   data = conn->data;
530 
531   DEBUGF(infof(data, "multi_done\n"));
532 
533   if(data->state.done)
534     /* Stop if multi_done() has already been called */
535     return CURLE_OK;
536 
537   Curl_getoff_all_pipelines(data, conn);
538 
539   /* Cleanup possible redirect junk */
540   free(data->req.newurl);
541   data->req.newurl = NULL;
542   free(data->req.location);
543   data->req.location = NULL;
544 
545   switch(status) {
546   case CURLE_ABORTED_BY_CALLBACK:
547   case CURLE_READ_ERROR:
548   case CURLE_WRITE_ERROR:
549     /* When we're aborted due to a callback return code it basically have to
550        be counted as premature as there is trouble ahead if we don't. We have
551        many callbacks and protocols work differently, we could potentially do
552        this more fine-grained in the future. */
553     premature = TRUE;
554   default:
555     break;
556   }
557 
558   /* this calls the protocol-specific function pointer previously set */
559   if(conn->handler->done)
560     result = conn->handler->done(conn, status, premature);
561   else
562     result = status;
563 
564   if(CURLE_ABORTED_BY_CALLBACK != result) {
565     /* avoid this if we already aborted by callback to avoid this calling
566        another callback */
567     CURLcode rc = Curl_pgrsDone(conn);
568     if(!result && rc)
569       result = CURLE_ABORTED_BY_CALLBACK;
570   }
571 
572   if((!premature &&
573       conn->send_pipe->size + conn->recv_pipe->size != 0 &&
574       !data->set.reuse_forbid &&
575       !conn->bits.close)) {
576     /* Stop if pipeline is not empty and we do not have to close
577        connection. */
578     DEBUGF(infof(data, "Connection still in use, no more multi_done now!\n"));
579     return CURLE_OK;
580   }
581 
582   data->state.done = TRUE; /* called just now! */
583   Curl_resolver_cancel(conn);
584 
585   if(conn->dns_entry) {
586     Curl_resolv_unlock(data, conn->dns_entry); /* done with this */
587     conn->dns_entry = NULL;
588   }
589 
590   /* if the transfer was completed in a paused state there can be buffered
591      data left to write and then kill */
592   free(data->state.tempwrite);
593   data->state.tempwrite = NULL;
594 
595   /* if data->set.reuse_forbid is TRUE, it means the libcurl client has
596      forced us to close this connection. This is ignored for requests taking
597      place in a NTLM authentication handshake
598 
599      if conn->bits.close is TRUE, it means that the connection should be
600      closed in spite of all our efforts to be nice, due to protocol
601      restrictions in our or the server's end
602 
603      if premature is TRUE, it means this connection was said to be DONE before
604      the entire request operation is complete and thus we can't know in what
605      state it is for re-using, so we're forced to close it. In a perfect world
606      we can add code that keep track of if we really must close it here or not,
607      but currently we have no such detail knowledge.
608   */
609 
610   if((data->set.reuse_forbid
611 #if defined(USE_NTLM)
612       && !(conn->ntlm.state == NTLMSTATE_TYPE2 ||
613            conn->proxyntlm.state == NTLMSTATE_TYPE2)
614 #endif
615      ) || conn->bits.close || premature) {
616     CURLcode res2 = Curl_disconnect(conn, premature); /* close connection */
617 
618     /* If we had an error already, make sure we return that one. But
619        if we got a new error, return that. */
620     if(!result && res2)
621       result = res2;
622   }
623   else {
624     /* the connection is no longer in use */
625     if(ConnectionDone(data, conn)) {
626       /* remember the most recently used connection */
627       data->state.lastconnect = conn;
628 
629       infof(data, "Connection #%ld to host %s left intact\n",
630             conn->connection_id,
631             conn->bits.httpproxy?conn->proxy.dispname:conn->host.dispname);
632     }
633     else
634       data->state.lastconnect = NULL;
635   }
636 
637   *connp = NULL; /* to make the caller of this function better detect that
638                     this was either closed or handed over to the connection
639                     cache here, and therefore cannot be used from this point on
640                  */
641   Curl_free_request_state(data);
642 
643   return result;
644 }
645 
curl_multi_remove_handle(struct Curl_multi * multi,struct Curl_easy * data)646 CURLMcode curl_multi_remove_handle(struct Curl_multi *multi,
647                                    struct Curl_easy *data)
648 {
649   struct Curl_easy *easy = data;
650   bool premature;
651   bool easy_owns_conn;
652   struct curl_llist_element *e;
653 
654   /* First, make some basic checks that the CURLM handle is a good handle */
655   if(!GOOD_MULTI_HANDLE(multi))
656     return CURLM_BAD_HANDLE;
657 
658   /* Verify that we got a somewhat good easy handle too */
659   if(!GOOD_EASY_HANDLE(data))
660     return CURLM_BAD_EASY_HANDLE;
661 
662   /* Prevent users from trying to remove same easy handle more than once */
663   if(!data->multi)
664     return CURLM_OK; /* it is already removed so let's say it is fine! */
665 
666   premature = (data->mstate < CURLM_STATE_COMPLETED) ? TRUE : FALSE;
667   easy_owns_conn = (data->easy_conn && (data->easy_conn->data == easy)) ?
668     TRUE : FALSE;
669 
670   /* If the 'state' is not INIT or COMPLETED, we might need to do something
671      nice to put the easy_handle in a good known state when this returns. */
672   if(premature) {
673     /* this handle is "alive" so we need to count down the total number of
674        alive connections when this is removed */
675     multi->num_alive--;
676 
677     /* When this handle gets removed, other handles may be able to get the
678        connection */
679     Curl_multi_process_pending_handles(multi);
680   }
681 
682   if(data->easy_conn &&
683      data->mstate > CURLM_STATE_DO &&
684      data->mstate < CURLM_STATE_COMPLETED) {
685     /* If the handle is in a pipeline and has started sending off its
686        request but not received its response yet, we need to close
687        connection. */
688     connclose(data->easy_conn, "Removed with partial response");
689     /* Set connection owner so that the DONE function closes it.  We can
690        safely do this here since connection is killed. */
691     data->easy_conn->data = easy;
692     easy_owns_conn = TRUE;
693   }
694 
695   /* The timer must be shut down before data->multi is set to NULL,
696      else the timenode will remain in the splay tree after
697      curl_easy_cleanup is called. */
698   Curl_expire(data, 0);
699 
700   if(data->dns.hostcachetype == HCACHE_MULTI) {
701     /* stop using the multi handle's DNS cache */
702     data->dns.hostcache = NULL;
703     data->dns.hostcachetype = HCACHE_NONE;
704   }
705 
706   if(data->easy_conn) {
707 
708     /* we must call multi_done() here (if we still own the connection) so that
709        we don't leave a half-baked one around */
710     if(easy_owns_conn) {
711 
712       /* multi_done() clears the conn->data field to lose the association
713          between the easy handle and the connection
714 
715          Note that this ignores the return code simply because there's
716          nothing really useful to do with it anyway! */
717       (void)multi_done(&data->easy_conn, data->result, premature);
718     }
719     else
720       /* Clear connection pipelines, if multi_done above was not called */
721       Curl_getoff_all_pipelines(data, data->easy_conn);
722   }
723 
724   Curl_wildcard_dtor(&data->wildcard);
725 
726   /* destroy the timeout list that is held in the easy handle, do this *after*
727      multi_done() as that may actually call Curl_expire that uses this */
728   if(data->state.timeoutlist) {
729     Curl_llist_destroy(data->state.timeoutlist, NULL);
730     data->state.timeoutlist = NULL;
731   }
732 
733   /* as this was using a shared connection cache we clear the pointer to that
734      since we're not part of that multi handle anymore */
735   data->state.conn_cache = NULL;
736 
737   /* change state without using multistate(), only to make singlesocket() do
738      what we want */
739   data->mstate = CURLM_STATE_COMPLETED;
740   singlesocket(multi, easy); /* to let the application know what sockets that
741                                 vanish with this handle */
742 
743   /* Remove the association between the connection and the handle */
744   if(data->easy_conn) {
745     data->easy_conn->data = NULL;
746     data->easy_conn = NULL;
747   }
748 
749   data->multi = NULL; /* clear the association to this multi handle */
750 
751   /* make sure there's no pending message in the queue sent from this easy
752      handle */
753 
754   for(e = multi->msglist->head; e; e = e->next) {
755     struct Curl_message *msg = e->ptr;
756 
757     if(msg->extmsg.easy_handle == easy) {
758       Curl_llist_remove(multi->msglist, e, NULL);
759       /* there can only be one from this specific handle */
760       break;
761     }
762   }
763 
764   /* make the previous node point to our next */
765   if(data->prev)
766     data->prev->next = data->next;
767   else
768     multi->easyp = data->next; /* point to first node */
769 
770   /* make our next point to our previous node */
771   if(data->next)
772     data->next->prev = data->prev;
773   else
774     multi->easylp = data->prev; /* point to last node */
775 
776   /* NOTE NOTE NOTE
777      We do not touch the easy handle here! */
778   multi->num_easy--; /* one less to care about now */
779 
780   update_timer(multi);
781   return CURLM_OK;
782 }
783 
784 /* Return TRUE if the application asked for a certain set of pipelining */
Curl_pipeline_wanted(const struct Curl_multi * multi,int bits)785 bool Curl_pipeline_wanted(const struct Curl_multi *multi, int bits)
786 {
787   return (multi && (multi->pipelining & bits)) ? TRUE : FALSE;
788 }
789 
Curl_multi_handlePipeBreak(struct Curl_easy * data)790 void Curl_multi_handlePipeBreak(struct Curl_easy *data)
791 {
792   data->easy_conn = NULL;
793 }
794 
waitconnect_getsock(struct connectdata * conn,curl_socket_t * sock,int numsocks)795 static int waitconnect_getsock(struct connectdata *conn,
796                                curl_socket_t *sock,
797                                int numsocks)
798 {
799   int i;
800   int s=0;
801   int rc=0;
802 
803   if(!numsocks)
804     return GETSOCK_BLANK;
805 
806   for(i=0; i<2; i++) {
807     if(conn->tempsock[i] != CURL_SOCKET_BAD) {
808       sock[s] = conn->tempsock[i];
809       rc |= GETSOCK_WRITESOCK(s++);
810     }
811   }
812 
813   return rc;
814 }
815 
waitproxyconnect_getsock(struct connectdata * conn,curl_socket_t * sock,int numsocks)816 static int waitproxyconnect_getsock(struct connectdata *conn,
817                                     curl_socket_t *sock,
818                                     int numsocks)
819 {
820   if(!numsocks)
821     return GETSOCK_BLANK;
822 
823   sock[0] = conn->sock[FIRSTSOCKET];
824 
825   /* when we've sent a CONNECT to a proxy, we should rather wait for the
826      socket to become readable to be able to get the response headers */
827   if(conn->tunnel_state[FIRSTSOCKET] == TUNNEL_CONNECT)
828     return GETSOCK_READSOCK(0);
829 
830   return GETSOCK_WRITESOCK(0);
831 }
832 
domore_getsock(struct connectdata * conn,curl_socket_t * socks,int numsocks)833 static int domore_getsock(struct connectdata *conn,
834                           curl_socket_t *socks,
835                           int numsocks)
836 {
837   if(conn && conn->handler->domore_getsock)
838     return conn->handler->domore_getsock(conn, socks, numsocks);
839   return GETSOCK_BLANK;
840 }
841 
842 /* returns bitmapped flags for this handle and its sockets */
multi_getsock(struct Curl_easy * data,curl_socket_t * socks,int numsocks)843 static int multi_getsock(struct Curl_easy *data,
844                          curl_socket_t *socks, /* points to numsocks number
845                                                   of sockets */
846                          int numsocks)
847 {
848   /* If the pipe broke, or if there's no connection left for this easy handle,
849      then we MUST bail out now with no bitmask set. The no connection case can
850      happen when this is called from curl_multi_remove_handle() =>
851      singlesocket() => multi_getsock().
852   */
853   if(data->state.pipe_broke || !data->easy_conn)
854     return 0;
855 
856   if(data->mstate > CURLM_STATE_CONNECT &&
857      data->mstate < CURLM_STATE_COMPLETED) {
858     /* Set up ownership correctly */
859     data->easy_conn->data = data;
860   }
861 
862   switch(data->mstate) {
863   default:
864 #if 0 /* switch back on these cases to get the compiler to check for all enums
865          to be present */
866   case CURLM_STATE_TOOFAST:  /* returns 0, so will not select. */
867   case CURLM_STATE_COMPLETED:
868   case CURLM_STATE_MSGSENT:
869   case CURLM_STATE_INIT:
870   case CURLM_STATE_CONNECT:
871   case CURLM_STATE_WAITDO:
872   case CURLM_STATE_DONE:
873   case CURLM_STATE_LAST:
874     /* this will get called with CURLM_STATE_COMPLETED when a handle is
875        removed */
876 #endif
877     return 0;
878 
879   case CURLM_STATE_WAITRESOLVE:
880     return Curl_resolver_getsock(data->easy_conn, socks, numsocks);
881 
882   case CURLM_STATE_PROTOCONNECT:
883   case CURLM_STATE_SENDPROTOCONNECT:
884     return Curl_protocol_getsock(data->easy_conn, socks, numsocks);
885 
886   case CURLM_STATE_DO:
887   case CURLM_STATE_DOING:
888     return Curl_doing_getsock(data->easy_conn, socks, numsocks);
889 
890   case CURLM_STATE_WAITPROXYCONNECT:
891     return waitproxyconnect_getsock(data->easy_conn, socks, numsocks);
892 
893   case CURLM_STATE_WAITCONNECT:
894     return waitconnect_getsock(data->easy_conn, socks, numsocks);
895 
896   case CURLM_STATE_DO_MORE:
897     return domore_getsock(data->easy_conn, socks, numsocks);
898 
899   case CURLM_STATE_DO_DONE: /* since is set after DO is completed, we switch
900                                to waiting for the same as the *PERFORM
901                                states */
902   case CURLM_STATE_PERFORM:
903   case CURLM_STATE_WAITPERFORM:
904     return Curl_single_getsock(data->easy_conn, socks, numsocks);
905   }
906 
907 }
908 
curl_multi_fdset(struct Curl_multi * multi,fd_set * read_fd_set,fd_set * write_fd_set,fd_set * exc_fd_set,int * max_fd)909 CURLMcode curl_multi_fdset(struct Curl_multi *multi,
910                            fd_set *read_fd_set, fd_set *write_fd_set,
911                            fd_set *exc_fd_set, int *max_fd)
912 {
913   /* Scan through all the easy handles to get the file descriptors set.
914      Some easy handles may not have connected to the remote host yet,
915      and then we must make sure that is done. */
916   struct Curl_easy *data;
917   int this_max_fd=-1;
918   curl_socket_t sockbunch[MAX_SOCKSPEREASYHANDLE];
919   int bitmap;
920   int i;
921   (void)exc_fd_set; /* not used */
922 
923   if(!GOOD_MULTI_HANDLE(multi))
924     return CURLM_BAD_HANDLE;
925 
926   data=multi->easyp;
927   while(data) {
928     bitmap = multi_getsock(data, sockbunch, MAX_SOCKSPEREASYHANDLE);
929 
930     for(i=0; i< MAX_SOCKSPEREASYHANDLE; i++) {
931       curl_socket_t s = CURL_SOCKET_BAD;
932 
933       if((bitmap & GETSOCK_READSOCK(i)) && VALID_SOCK((sockbunch[i]))) {
934         FD_SET(sockbunch[i], read_fd_set);
935         s = sockbunch[i];
936       }
937       if((bitmap & GETSOCK_WRITESOCK(i)) && VALID_SOCK((sockbunch[i]))) {
938         FD_SET(sockbunch[i], write_fd_set);
939         s = sockbunch[i];
940       }
941       if(s == CURL_SOCKET_BAD)
942         /* this socket is unused, break out of loop */
943         break;
944       else {
945         if((int)s > this_max_fd)
946           this_max_fd = (int)s;
947       }
948     }
949 
950     data = data->next; /* check next handle */
951   }
952 
953   *max_fd = this_max_fd;
954 
955   return CURLM_OK;
956 }
957 
curl_multi_wait(struct Curl_multi * multi,struct curl_waitfd extra_fds[],unsigned int extra_nfds,int timeout_ms,int * ret)958 CURLMcode curl_multi_wait(struct Curl_multi *multi,
959                           struct curl_waitfd extra_fds[],
960                           unsigned int extra_nfds,
961                           int timeout_ms,
962                           int *ret)
963 {
964   struct Curl_easy *data;
965   curl_socket_t sockbunch[MAX_SOCKSPEREASYHANDLE];
966   int bitmap;
967   unsigned int i;
968   unsigned int nfds = 0;
969   unsigned int curlfds;
970   struct pollfd *ufds = NULL;
971   long timeout_internal;
972   int retcode = 0;
973 
974   if(!GOOD_MULTI_HANDLE(multi))
975     return CURLM_BAD_HANDLE;
976 
977   /* If the internally desired timeout is actually shorter than requested from
978      the outside, then use the shorter time! But only if the internal timer
979      is actually larger than -1! */
980   (void)multi_timeout(multi, &timeout_internal);
981   if((timeout_internal >= 0) && (timeout_internal < (long)timeout_ms))
982     timeout_ms = (int)timeout_internal;
983 
984   /* Count up how many fds we have from the multi handle */
985   data=multi->easyp;
986   while(data) {
987     bitmap = multi_getsock(data, sockbunch, MAX_SOCKSPEREASYHANDLE);
988 
989     for(i=0; i< MAX_SOCKSPEREASYHANDLE; i++) {
990       curl_socket_t s = CURL_SOCKET_BAD;
991 
992       if(bitmap & GETSOCK_READSOCK(i)) {
993         ++nfds;
994         s = sockbunch[i];
995       }
996       if(bitmap & GETSOCK_WRITESOCK(i)) {
997         ++nfds;
998         s = sockbunch[i];
999       }
1000       if(s == CURL_SOCKET_BAD) {
1001         break;
1002       }
1003     }
1004 
1005     data = data->next; /* check next handle */
1006   }
1007 
1008   curlfds = nfds; /* number of internal file descriptors */
1009   nfds += extra_nfds; /* add the externally provided ones */
1010 
1011   if(nfds || extra_nfds) {
1012     ufds = malloc(nfds * sizeof(struct pollfd));
1013     if(!ufds)
1014       return CURLM_OUT_OF_MEMORY;
1015   }
1016   nfds = 0;
1017 
1018   /* only do the second loop if we found descriptors in the first stage run
1019      above */
1020 
1021   if(curlfds) {
1022     /* Add the curl handles to our pollfds first */
1023     data=multi->easyp;
1024     while(data) {
1025       bitmap = multi_getsock(data, sockbunch, MAX_SOCKSPEREASYHANDLE);
1026 
1027       for(i=0; i< MAX_SOCKSPEREASYHANDLE; i++) {
1028         curl_socket_t s = CURL_SOCKET_BAD;
1029 
1030         if(bitmap & GETSOCK_READSOCK(i)) {
1031           ufds[nfds].fd = sockbunch[i];
1032           ufds[nfds].events = POLLIN;
1033           ++nfds;
1034           s = sockbunch[i];
1035         }
1036         if(bitmap & GETSOCK_WRITESOCK(i)) {
1037           ufds[nfds].fd = sockbunch[i];
1038           ufds[nfds].events = POLLOUT;
1039           ++nfds;
1040           s = sockbunch[i];
1041         }
1042         if(s == CURL_SOCKET_BAD) {
1043           break;
1044         }
1045       }
1046 
1047       data = data->next; /* check next handle */
1048     }
1049   }
1050 
1051   /* Add external file descriptions from poll-like struct curl_waitfd */
1052   for(i = 0; i < extra_nfds; i++) {
1053     ufds[nfds].fd = extra_fds[i].fd;
1054     ufds[nfds].events = 0;
1055     if(extra_fds[i].events & CURL_WAIT_POLLIN)
1056       ufds[nfds].events |= POLLIN;
1057     if(extra_fds[i].events & CURL_WAIT_POLLPRI)
1058       ufds[nfds].events |= POLLPRI;
1059     if(extra_fds[i].events & CURL_WAIT_POLLOUT)
1060       ufds[nfds].events |= POLLOUT;
1061     ++nfds;
1062   }
1063 
1064   if(nfds) {
1065     int pollrc;
1066     /* wait... */
1067     pollrc = Curl_poll(ufds, nfds, timeout_ms);
1068     DEBUGF(infof(data, "Curl_poll(%d ds, %d ms) == %d\n",
1069                  nfds, timeout_ms, pollrc));
1070 
1071     if(pollrc > 0) {
1072       retcode = pollrc;
1073       /* copy revents results from the poll to the curl_multi_wait poll
1074          struct, the bit values of the actual underlying poll() implementation
1075          may not be the same as the ones in the public libcurl API! */
1076       for(i = 0; i < extra_nfds; i++) {
1077         unsigned short mask = 0;
1078         unsigned r = ufds[curlfds + i].revents;
1079 
1080         if(r & POLLIN)
1081           mask |= CURL_WAIT_POLLIN;
1082         if(r & POLLOUT)
1083           mask |= CURL_WAIT_POLLOUT;
1084         if(r & POLLPRI)
1085           mask |= CURL_WAIT_POLLPRI;
1086 
1087         extra_fds[i].revents = mask;
1088       }
1089     }
1090   }
1091 
1092   free(ufds);
1093   if(ret)
1094     *ret = retcode;
1095   return CURLM_OK;
1096 }
1097 
1098 /*
1099  * Curl_multi_connchanged() is called to tell that there is a connection in
1100  * this multi handle that has changed state (pipelining become possible, the
1101  * number of allowed streams changed or similar), and a subsequent use of this
1102  * multi handle should move CONNECT_PEND handles back to CONNECT to have them
1103  * retry.
1104  */
Curl_multi_connchanged(struct Curl_multi * multi)1105 void Curl_multi_connchanged(struct Curl_multi *multi)
1106 {
1107   multi->recheckstate = TRUE;
1108 }
1109 
1110 /*
1111  * multi_ischanged() is called
1112  *
1113  * Returns TRUE/FALSE whether the state is changed to trigger a CONNECT_PEND
1114  * => CONNECT action.
1115  *
1116  * Set 'clear' to TRUE to have it also clear the state variable.
1117  */
multi_ischanged(struct Curl_multi * multi,bool clear)1118 static bool multi_ischanged(struct Curl_multi *multi, bool clear)
1119 {
1120   bool retval = multi->recheckstate;
1121   if(clear)
1122     multi->recheckstate = FALSE;
1123   return retval;
1124 }
1125 
Curl_multi_add_perform(struct Curl_multi * multi,struct Curl_easy * data,struct connectdata * conn)1126 CURLMcode Curl_multi_add_perform(struct Curl_multi *multi,
1127                                  struct Curl_easy *data,
1128                                  struct connectdata *conn)
1129 {
1130   CURLMcode rc;
1131 
1132   rc = curl_multi_add_handle(multi, data);
1133   if(!rc) {
1134     struct SingleRequest *k = &data->req;
1135 
1136     /* pass in NULL for 'conn' here since we don't want to init the
1137        connection, only this transfer */
1138     Curl_init_do(data, NULL);
1139 
1140     /* take this handle to the perform state right away */
1141     multistate(data, CURLM_STATE_PERFORM);
1142     data->easy_conn = conn;
1143     k->keepon |= KEEP_RECV; /* setup to receive! */
1144   }
1145   return rc;
1146 }
1147 
multi_reconnect_request(struct connectdata ** connp)1148 static CURLcode multi_reconnect_request(struct connectdata **connp)
1149 {
1150   CURLcode result = CURLE_OK;
1151   struct connectdata *conn = *connp;
1152   struct Curl_easy *data = conn->data;
1153 
1154   /* This was a re-use of a connection and we got a write error in the
1155    * DO-phase. Then we DISCONNECT this connection and have another attempt to
1156    * CONNECT and then DO again! The retry cannot possibly find another
1157    * connection to re-use, since we only keep one possible connection for
1158    * each.  */
1159 
1160   infof(data, "Re-used connection seems dead, get a new one\n");
1161 
1162   connclose(conn, "Reconnect dead connection"); /* enforce close */
1163   result = multi_done(&conn, result, FALSE); /* we are so done with this */
1164 
1165   /* conn may no longer be a good pointer, clear it to avoid mistakes by
1166      parent functions */
1167   *connp = NULL;
1168 
1169   /*
1170    * We need to check for CURLE_SEND_ERROR here as well. This could happen
1171    * when the request failed on a FTP connection and thus multi_done() itself
1172    * tried to use the connection (again).
1173    */
1174   if(!result || (CURLE_SEND_ERROR == result)) {
1175     bool async;
1176     bool protocol_done = TRUE;
1177 
1178     /* Now, redo the connect and get a new connection */
1179     result = Curl_connect(data, connp, &async, &protocol_done);
1180     if(!result) {
1181       /* We have connected or sent away a name resolve query fine */
1182 
1183       conn = *connp; /* setup conn to again point to something nice */
1184       if(async) {
1185         /* Now, if async is TRUE here, we need to wait for the name
1186            to resolve */
1187         result = Curl_resolver_wait_resolv(conn, NULL);
1188         if(result)
1189           return result;
1190 
1191         /* Resolved, continue with the connection */
1192         result = Curl_async_resolved(conn, &protocol_done);
1193         if(result)
1194           return result;
1195       }
1196     }
1197   }
1198 
1199   return result;
1200 }
1201 
1202 /*
1203  * do_complete is called when the DO actions are complete.
1204  *
1205  * We init chunking and trailer bits to their default values here immediately
1206  * before receiving any header data for the current request in the pipeline.
1207  */
do_complete(struct connectdata * conn)1208 static void do_complete(struct connectdata *conn)
1209 {
1210   conn->data->req.chunk=FALSE;
1211   conn->data->req.maxfd = (conn->sockfd>conn->writesockfd?
1212                            conn->sockfd:conn->writesockfd)+1;
1213   Curl_pgrsTime(conn->data, TIMER_PRETRANSFER);
1214 }
1215 
multi_do(struct connectdata ** connp,bool * done)1216 static CURLcode multi_do(struct connectdata **connp, bool *done)
1217 {
1218   CURLcode result=CURLE_OK;
1219   struct connectdata *conn = *connp;
1220   struct Curl_easy *data = conn->data;
1221 
1222   if(conn->handler->do_it) {
1223     /* generic protocol-specific function pointer set in curl_connect() */
1224     result = conn->handler->do_it(conn, done);
1225 
1226     /* This was formerly done in transfer.c, but we better do it here */
1227     if((CURLE_SEND_ERROR == result) && conn->bits.reuse) {
1228       /*
1229        * If the connection is using an easy handle, call reconnect
1230        * to re-establish the connection.  Otherwise, let the multi logic
1231        * figure out how to re-establish the connection.
1232        */
1233       if(!data->multi) {
1234         result = multi_reconnect_request(connp);
1235 
1236         if(!result) {
1237           /* ... finally back to actually retry the DO phase */
1238           conn = *connp; /* re-assign conn since multi_reconnect_request
1239                             creates a new connection */
1240           result = conn->handler->do_it(conn, done);
1241         }
1242       }
1243       else
1244         return result;
1245     }
1246 
1247     if(!result && *done)
1248       /* do_complete must be called after the protocol-specific DO function */
1249       do_complete(conn);
1250   }
1251   return result;
1252 }
1253 
1254 /*
1255  * multi_do_more() is called during the DO_MORE multi state. It is basically a
1256  * second stage DO state which (wrongly) was introduced to support FTP's
1257  * second connection.
1258  *
1259  * TODO: A future libcurl should be able to work away this state.
1260  *
1261  * 'complete' can return 0 for incomplete, 1 for done and -1 for go back to
1262  * DOING state there's more work to do!
1263  */
1264 
multi_do_more(struct connectdata * conn,int * complete)1265 static CURLcode multi_do_more(struct connectdata *conn, int *complete)
1266 {
1267   CURLcode result=CURLE_OK;
1268 
1269   *complete = 0;
1270 
1271   if(conn->handler->do_more)
1272     result = conn->handler->do_more(conn, complete);
1273 
1274   if(!result && (*complete == 1))
1275     /* do_complete must be called after the protocol-specific DO function */
1276     do_complete(conn);
1277 
1278   return result;
1279 }
1280 
multi_runsingle(struct Curl_multi * multi,struct timeval now,struct Curl_easy * data)1281 static CURLMcode multi_runsingle(struct Curl_multi *multi,
1282                                  struct timeval now,
1283                                  struct Curl_easy *data)
1284 {
1285   struct Curl_message *msg = NULL;
1286   bool connected;
1287   bool async;
1288   bool protocol_connect = FALSE;
1289   bool dophase_done = FALSE;
1290   bool done = FALSE;
1291   CURLMcode rc;
1292   CURLcode result = CURLE_OK;
1293   struct SingleRequest *k;
1294   long timeout_ms;
1295   int control;
1296 
1297   if(!GOOD_EASY_HANDLE(data))
1298     return CURLM_BAD_EASY_HANDLE;
1299 
1300   do {
1301     bool disconnect_conn = FALSE;
1302     rc = CURLM_OK;
1303 
1304     /* Handle the case when the pipe breaks, i.e., the connection
1305        we're using gets cleaned up and we're left with nothing. */
1306     if(data->state.pipe_broke) {
1307       infof(data, "Pipe broke: handle %p, url = %s\n",
1308             (void *)data, data->state.path);
1309 
1310       if(data->mstate < CURLM_STATE_COMPLETED) {
1311         /* Head back to the CONNECT state */
1312         multistate(data, CURLM_STATE_CONNECT);
1313         rc = CURLM_CALL_MULTI_PERFORM;
1314         result = CURLE_OK;
1315       }
1316 
1317       data->state.pipe_broke = FALSE;
1318       data->easy_conn = NULL;
1319       continue;
1320     }
1321 
1322     if(!data->easy_conn &&
1323        data->mstate > CURLM_STATE_CONNECT &&
1324        data->mstate < CURLM_STATE_DONE) {
1325       /* In all these states, the code will blindly access 'data->easy_conn'
1326          so this is precaution that it isn't NULL. And it silences static
1327          analyzers. */
1328       failf(data, "In state %d with no easy_conn, bail out!\n", data->mstate);
1329       return CURLM_INTERNAL_ERROR;
1330     }
1331 
1332     if(multi_ischanged(multi, TRUE)) {
1333       DEBUGF(infof(data, "multi changed, check CONNECT_PEND queue!\n"));
1334       Curl_multi_process_pending_handles(multi);
1335     }
1336 
1337     if(data->easy_conn && data->mstate > CURLM_STATE_CONNECT &&
1338        data->mstate < CURLM_STATE_COMPLETED)
1339       /* Make sure we set the connection's current owner */
1340       data->easy_conn->data = data;
1341 
1342     if(data->easy_conn &&
1343        (data->mstate >= CURLM_STATE_CONNECT) &&
1344        (data->mstate < CURLM_STATE_COMPLETED)) {
1345       /* we need to wait for the connect state as only then is the start time
1346          stored, but we must not check already completed handles */
1347 
1348       timeout_ms = Curl_timeleft(data, &now,
1349                                  (data->mstate <= CURLM_STATE_WAITDO)?
1350                                  TRUE:FALSE);
1351 
1352       if(timeout_ms < 0) {
1353         /* Handle timed out */
1354         if(data->mstate == CURLM_STATE_WAITRESOLVE)
1355           failf(data, "Resolving timed out after %ld milliseconds",
1356                 Curl_tvdiff(now, data->progress.t_startsingle));
1357         else if(data->mstate == CURLM_STATE_WAITCONNECT)
1358           failf(data, "Connection timed out after %ld milliseconds",
1359                 Curl_tvdiff(now, data->progress.t_startsingle));
1360         else {
1361           k = &data->req;
1362           if(k->size != -1) {
1363             failf(data, "Operation timed out after %ld milliseconds with %"
1364                   CURL_FORMAT_CURL_OFF_T " out of %"
1365                   CURL_FORMAT_CURL_OFF_T " bytes received",
1366                   Curl_tvdiff(now, data->progress.t_startsingle),
1367                   k->bytecount, k->size);
1368           }
1369           else {
1370             failf(data, "Operation timed out after %ld milliseconds with %"
1371                   CURL_FORMAT_CURL_OFF_T " bytes received",
1372                   Curl_tvdiff(now, data->progress.t_startsingle),
1373                   k->bytecount);
1374           }
1375         }
1376 
1377         /* Force connection closed if the connection has indeed been used */
1378         if(data->mstate > CURLM_STATE_DO) {
1379           connclose(data->easy_conn, "Disconnected with pending data");
1380           disconnect_conn = TRUE;
1381         }
1382         result = CURLE_OPERATION_TIMEDOUT;
1383         (void)multi_done(&data->easy_conn, result, TRUE);
1384         /* Skip the statemachine and go directly to error handling section. */
1385         goto statemachine_end;
1386       }
1387     }
1388 
1389     switch(data->mstate) {
1390     case CURLM_STATE_INIT:
1391       /* init this transfer. */
1392       result=Curl_pretransfer(data);
1393 
1394       if(!result) {
1395         /* after init, go CONNECT */
1396         multistate(data, CURLM_STATE_CONNECT);
1397         Curl_pgrsTime(data, TIMER_STARTOP);
1398         rc = CURLM_CALL_MULTI_PERFORM;
1399       }
1400       break;
1401 
1402     case CURLM_STATE_CONNECT_PEND:
1403       /* We will stay here until there is a connection available. Then
1404          we try again in the CURLM_STATE_CONNECT state. */
1405       break;
1406 
1407     case CURLM_STATE_CONNECT:
1408       /* Connect. We want to get a connection identifier filled in. */
1409       Curl_pgrsTime(data, TIMER_STARTSINGLE);
1410       result = Curl_connect(data, &data->easy_conn,
1411                             &async, &protocol_connect);
1412       if(CURLE_NO_CONNECTION_AVAILABLE == result) {
1413         /* There was no connection available. We will go to the pending
1414            state and wait for an available connection. */
1415         multistate(data, CURLM_STATE_CONNECT_PEND);
1416 
1417         /* add this handle to the list of connect-pending handles */
1418         if(!Curl_llist_insert_next(multi->pending, multi->pending->tail, data))
1419           result = CURLE_OUT_OF_MEMORY;
1420         else
1421           result = CURLE_OK;
1422         break;
1423       }
1424 
1425       if(!result) {
1426         /* Add this handle to the send or pend pipeline */
1427         result = Curl_add_handle_to_pipeline(data, data->easy_conn);
1428         if(result)
1429           disconnect_conn = TRUE;
1430         else {
1431           if(async)
1432             /* We're now waiting for an asynchronous name lookup */
1433             multistate(data, CURLM_STATE_WAITRESOLVE);
1434           else {
1435             /* after the connect has been sent off, go WAITCONNECT unless the
1436                protocol connect is already done and we can go directly to
1437                WAITDO or DO! */
1438             rc = CURLM_CALL_MULTI_PERFORM;
1439 
1440             if(protocol_connect)
1441               multistate(data, Curl_pipeline_wanted(multi, CURLPIPE_HTTP1)?
1442                          CURLM_STATE_WAITDO:CURLM_STATE_DO);
1443             else {
1444 #ifndef CURL_DISABLE_HTTP
1445               if(data->easy_conn->tunnel_state[FIRSTSOCKET] == TUNNEL_CONNECT)
1446                 multistate(data, CURLM_STATE_WAITPROXYCONNECT);
1447               else
1448 #endif
1449                 multistate(data, CURLM_STATE_WAITCONNECT);
1450             }
1451           }
1452         }
1453       }
1454       break;
1455 
1456     case CURLM_STATE_WAITRESOLVE:
1457       /* awaiting an asynch name resolve to complete */
1458     {
1459       struct Curl_dns_entry *dns = NULL;
1460       struct connectdata *conn = data->easy_conn;
1461       const char *hostname;
1462 
1463       if(conn->bits.proxy)
1464         hostname = conn->proxy.name;
1465       else if(conn->bits.conn_to_host)
1466         hostname = conn->conn_to_host.name;
1467       else
1468         hostname = conn->host.name;
1469 
1470       /* check if we have the name resolved by now */
1471       dns = Curl_fetch_addr(conn, hostname, (int)conn->port);
1472 
1473       if(dns) {
1474 #ifdef CURLRES_ASYNCH
1475         conn->async.dns = dns;
1476         conn->async.done = TRUE;
1477 #endif
1478         result = CURLE_OK;
1479         infof(data, "Hostname '%s' was found in DNS cache\n", hostname);
1480       }
1481 
1482       if(!dns)
1483         result = Curl_resolver_is_resolved(data->easy_conn, &dns);
1484 
1485       /* Update sockets here, because the socket(s) may have been
1486          closed and the application thus needs to be told, even if it
1487          is likely that the same socket(s) will again be used further
1488          down.  If the name has not yet been resolved, it is likely
1489          that new sockets have been opened in an attempt to contact
1490          another resolver. */
1491       singlesocket(multi, data);
1492 
1493       if(dns) {
1494         /* Perform the next step in the connection phase, and then move on
1495            to the WAITCONNECT state */
1496         result = Curl_async_resolved(data->easy_conn, &protocol_connect);
1497 
1498         if(result)
1499           /* if Curl_async_resolved() returns failure, the connection struct
1500              is already freed and gone */
1501           data->easy_conn = NULL;           /* no more connection */
1502         else {
1503           /* call again please so that we get the next socket setup */
1504           rc = CURLM_CALL_MULTI_PERFORM;
1505           if(protocol_connect)
1506             multistate(data, Curl_pipeline_wanted(multi, CURLPIPE_HTTP1)?
1507                        CURLM_STATE_WAITDO:CURLM_STATE_DO);
1508           else {
1509 #ifndef CURL_DISABLE_HTTP
1510             if(data->easy_conn->tunnel_state[FIRSTSOCKET] == TUNNEL_CONNECT)
1511               multistate(data, CURLM_STATE_WAITPROXYCONNECT);
1512             else
1513 #endif
1514               multistate(data, CURLM_STATE_WAITCONNECT);
1515           }
1516         }
1517       }
1518 
1519       if(result) {
1520         /* failure detected */
1521         disconnect_conn = TRUE;
1522         break;
1523       }
1524     }
1525     break;
1526 
1527 #ifndef CURL_DISABLE_HTTP
1528     case CURLM_STATE_WAITPROXYCONNECT:
1529       /* this is HTTP-specific, but sending CONNECT to a proxy is HTTP... */
1530       result = Curl_http_connect(data->easy_conn, &protocol_connect);
1531 
1532       if(data->easy_conn->bits.proxy_connect_closed) {
1533         rc = CURLM_CALL_MULTI_PERFORM;
1534         /* connect back to proxy again */
1535         result = CURLE_OK;
1536         multi_done(&data->easy_conn, CURLE_OK, FALSE);
1537         multistate(data, CURLM_STATE_CONNECT);
1538       }
1539       else if(!result) {
1540         if(data->easy_conn->tunnel_state[FIRSTSOCKET] == TUNNEL_COMPLETE) {
1541           rc = CURLM_CALL_MULTI_PERFORM;
1542           /* initiate protocol connect phase */
1543           multistate(data, CURLM_STATE_SENDPROTOCONNECT);
1544         }
1545       }
1546       break;
1547 #endif
1548 
1549     case CURLM_STATE_WAITCONNECT:
1550       /* awaiting a completion of an asynch TCP connect */
1551       result = Curl_is_connected(data->easy_conn, FIRSTSOCKET, &connected);
1552       if(connected && !result) {
1553         rc = CURLM_CALL_MULTI_PERFORM;
1554         multistate(data, data->easy_conn->bits.tunnel_proxy?
1555                    CURLM_STATE_WAITPROXYCONNECT:
1556                    CURLM_STATE_SENDPROTOCONNECT);
1557       }
1558       else if(result) {
1559         /* failure detected */
1560         /* Just break, the cleaning up is handled all in one place */
1561         disconnect_conn = TRUE;
1562         break;
1563       }
1564       break;
1565 
1566     case CURLM_STATE_SENDPROTOCONNECT:
1567       result = Curl_protocol_connect(data->easy_conn, &protocol_connect);
1568       if(!protocol_connect)
1569         /* switch to waiting state */
1570         multistate(data, CURLM_STATE_PROTOCONNECT);
1571       else if(!result) {
1572         /* protocol connect has completed, go WAITDO or DO */
1573         multistate(data, Curl_pipeline_wanted(multi, CURLPIPE_HTTP1)?
1574                    CURLM_STATE_WAITDO:CURLM_STATE_DO);
1575         rc = CURLM_CALL_MULTI_PERFORM;
1576       }
1577       else if(result) {
1578         /* failure detected */
1579         Curl_posttransfer(data);
1580         multi_done(&data->easy_conn, result, TRUE);
1581         disconnect_conn = TRUE;
1582       }
1583       break;
1584 
1585     case CURLM_STATE_PROTOCONNECT:
1586       /* protocol-specific connect phase */
1587       result = Curl_protocol_connecting(data->easy_conn, &protocol_connect);
1588       if(!result && protocol_connect) {
1589         /* after the connect has completed, go WAITDO or DO */
1590         multistate(data, Curl_pipeline_wanted(multi, CURLPIPE_HTTP1)?
1591                    CURLM_STATE_WAITDO:CURLM_STATE_DO);
1592         rc = CURLM_CALL_MULTI_PERFORM;
1593       }
1594       else if(result) {
1595         /* failure detected */
1596         Curl_posttransfer(data);
1597         multi_done(&data->easy_conn, result, TRUE);
1598         disconnect_conn = TRUE;
1599       }
1600       break;
1601 
1602     case CURLM_STATE_WAITDO:
1603       /* Wait for our turn to DO when we're pipelining requests */
1604       if(Curl_pipeline_checkget_write(data, data->easy_conn)) {
1605         /* Grabbed the channel */
1606         multistate(data, CURLM_STATE_DO);
1607         rc = CURLM_CALL_MULTI_PERFORM;
1608       }
1609       break;
1610 
1611     case CURLM_STATE_DO:
1612       if(data->set.connect_only) {
1613         /* keep connection open for application to use the socket */
1614         connkeep(data->easy_conn, "CONNECT_ONLY");
1615         multistate(data, CURLM_STATE_DONE);
1616         result = CURLE_OK;
1617         rc = CURLM_CALL_MULTI_PERFORM;
1618       }
1619       else {
1620         /* Perform the protocol's DO action */
1621         result = multi_do(&data->easy_conn, &dophase_done);
1622 
1623         /* When multi_do() returns failure, data->easy_conn might be NULL! */
1624 
1625         if(!result) {
1626           if(!dophase_done) {
1627             /* some steps needed for wildcard matching */
1628             if(data->set.wildcardmatch) {
1629               struct WildcardData *wc = &data->wildcard;
1630               if(wc->state == CURLWC_DONE || wc->state == CURLWC_SKIP) {
1631                 /* skip some states if it is important */
1632                 multi_done(&data->easy_conn, CURLE_OK, FALSE);
1633                 multistate(data, CURLM_STATE_DONE);
1634                 rc = CURLM_CALL_MULTI_PERFORM;
1635                 break;
1636               }
1637             }
1638             /* DO was not completed in one function call, we must continue
1639                DOING... */
1640             multistate(data, CURLM_STATE_DOING);
1641             rc = CURLM_OK;
1642           }
1643 
1644           /* after DO, go DO_DONE... or DO_MORE */
1645           else if(data->easy_conn->bits.do_more) {
1646             /* we're supposed to do more, but we need to sit down, relax
1647                and wait a little while first */
1648             multistate(data, CURLM_STATE_DO_MORE);
1649             rc = CURLM_OK;
1650           }
1651           else {
1652             /* we're done with the DO, now DO_DONE */
1653             multistate(data, CURLM_STATE_DO_DONE);
1654             rc = CURLM_CALL_MULTI_PERFORM;
1655           }
1656         }
1657         else if((CURLE_SEND_ERROR == result) &&
1658                 data->easy_conn->bits.reuse) {
1659           /*
1660            * In this situation, a connection that we were trying to use
1661            * may have unexpectedly died.  If possible, send the connection
1662            * back to the CONNECT phase so we can try again.
1663            */
1664           char *newurl = NULL;
1665           followtype follow=FOLLOW_NONE;
1666           CURLcode drc;
1667           bool retry = FALSE;
1668 
1669           drc = Curl_retry_request(data->easy_conn, &newurl);
1670           if(drc) {
1671             /* a failure here pretty much implies an out of memory */
1672             result = drc;
1673             disconnect_conn = TRUE;
1674           }
1675           else
1676             retry = (newurl)?TRUE:FALSE;
1677 
1678           Curl_posttransfer(data);
1679           drc = multi_done(&data->easy_conn, result, FALSE);
1680 
1681           /* When set to retry the connection, we must to go back to
1682            * the CONNECT state */
1683           if(retry) {
1684             if(!drc || (drc == CURLE_SEND_ERROR)) {
1685               follow = FOLLOW_RETRY;
1686               drc = Curl_follow(data, newurl, follow);
1687               if(!drc) {
1688                 multistate(data, CURLM_STATE_CONNECT);
1689                 rc = CURLM_CALL_MULTI_PERFORM;
1690                 result = CURLE_OK;
1691               }
1692               else {
1693                 /* Follow failed */
1694                 result = drc;
1695                 free(newurl);
1696               }
1697             }
1698             else {
1699               /* done didn't return OK or SEND_ERROR */
1700               result = drc;
1701               free(newurl);
1702             }
1703           }
1704           else {
1705             /* Have error handler disconnect conn if we can't retry */
1706             disconnect_conn = TRUE;
1707             free(newurl);
1708           }
1709         }
1710         else {
1711           /* failure detected */
1712           Curl_posttransfer(data);
1713           if(data->easy_conn)
1714             multi_done(&data->easy_conn, result, FALSE);
1715           disconnect_conn = TRUE;
1716         }
1717       }
1718       break;
1719 
1720     case CURLM_STATE_DOING:
1721       /* we continue DOING until the DO phase is complete */
1722       result = Curl_protocol_doing(data->easy_conn,
1723                                    &dophase_done);
1724       if(!result) {
1725         if(dophase_done) {
1726           /* after DO, go DO_DONE or DO_MORE */
1727           multistate(data, data->easy_conn->bits.do_more?
1728                      CURLM_STATE_DO_MORE:
1729                      CURLM_STATE_DO_DONE);
1730           rc = CURLM_CALL_MULTI_PERFORM;
1731         } /* dophase_done */
1732       }
1733       else {
1734         /* failure detected */
1735         Curl_posttransfer(data);
1736         multi_done(&data->easy_conn, result, FALSE);
1737         disconnect_conn = TRUE;
1738       }
1739       break;
1740 
1741     case CURLM_STATE_DO_MORE:
1742       /*
1743        * When we are connected, DO MORE and then go DO_DONE
1744        */
1745       result = multi_do_more(data->easy_conn, &control);
1746 
1747       /* No need to remove this handle from the send pipeline here since that
1748          is done in multi_done() */
1749       if(!result) {
1750         if(control) {
1751           /* if positive, advance to DO_DONE
1752              if negative, go back to DOING */
1753           multistate(data, control==1?
1754                      CURLM_STATE_DO_DONE:
1755                      CURLM_STATE_DOING);
1756           rc = CURLM_CALL_MULTI_PERFORM;
1757         }
1758         else
1759           /* stay in DO_MORE */
1760           rc = CURLM_OK;
1761       }
1762       else {
1763         /* failure detected */
1764         Curl_posttransfer(data);
1765         multi_done(&data->easy_conn, result, FALSE);
1766         disconnect_conn = TRUE;
1767       }
1768       break;
1769 
1770     case CURLM_STATE_DO_DONE:
1771       /* Move ourselves from the send to recv pipeline */
1772       Curl_move_handle_from_send_to_recv_pipe(data, data->easy_conn);
1773       /* Check if we can move pending requests to send pipe */
1774       Curl_multi_process_pending_handles(multi);
1775 
1776       /* Only perform the transfer if there's a good socket to work with.
1777          Having both BAD is a signal to skip immediately to DONE */
1778       if((data->easy_conn->sockfd != CURL_SOCKET_BAD) ||
1779          (data->easy_conn->writesockfd != CURL_SOCKET_BAD))
1780         multistate(data, CURLM_STATE_WAITPERFORM);
1781       else
1782         multistate(data, CURLM_STATE_DONE);
1783       rc = CURLM_CALL_MULTI_PERFORM;
1784       break;
1785 
1786     case CURLM_STATE_WAITPERFORM:
1787       /* Wait for our turn to PERFORM */
1788       if(Curl_pipeline_checkget_read(data, data->easy_conn)) {
1789         /* Grabbed the channel */
1790         multistate(data, CURLM_STATE_PERFORM);
1791         rc = CURLM_CALL_MULTI_PERFORM;
1792       }
1793       break;
1794 
1795     case CURLM_STATE_TOOFAST: /* limit-rate exceeded in either direction */
1796       /* if both rates are within spec, resume transfer */
1797       if(Curl_pgrsUpdate(data->easy_conn))
1798         result = CURLE_ABORTED_BY_CALLBACK;
1799       else
1800         result = Curl_speedcheck(data, now);
1801 
1802       if(( (data->set.max_send_speed == 0) ||
1803            (data->progress.ulspeed < data->set.max_send_speed))  &&
1804          ( (data->set.max_recv_speed == 0) ||
1805            (data->progress.dlspeed < data->set.max_recv_speed)))
1806         multistate(data, CURLM_STATE_PERFORM);
1807       break;
1808 
1809     case CURLM_STATE_PERFORM:
1810     {
1811       char *newurl = NULL;
1812       bool retry = FALSE;
1813 
1814       /* check if over send speed */
1815       if((data->set.max_send_speed > 0) &&
1816          (data->progress.ulspeed > data->set.max_send_speed)) {
1817         int buffersize;
1818 
1819         multistate(data, CURLM_STATE_TOOFAST);
1820 
1821         /* calculate upload rate-limitation timeout. */
1822         buffersize = (int)(data->set.buffer_size ?
1823                            data->set.buffer_size : BUFSIZE);
1824         timeout_ms = Curl_sleep_time(data->set.max_send_speed,
1825                                      data->progress.ulspeed, buffersize);
1826         Curl_expire_latest(data, timeout_ms);
1827         break;
1828       }
1829 
1830       /* check if over recv speed */
1831       if((data->set.max_recv_speed > 0) &&
1832          (data->progress.dlspeed > data->set.max_recv_speed)) {
1833         int buffersize;
1834 
1835         multistate(data, CURLM_STATE_TOOFAST);
1836 
1837         /* Calculate download rate-limitation timeout. */
1838         buffersize = (int)(data->set.buffer_size ?
1839                            data->set.buffer_size : BUFSIZE);
1840         timeout_ms = Curl_sleep_time(data->set.max_recv_speed,
1841                                      data->progress.dlspeed, buffersize);
1842         Curl_expire_latest(data, timeout_ms);
1843         break;
1844       }
1845 
1846       /* read/write data if it is ready to do so */
1847       result = Curl_readwrite(data->easy_conn, data, &done);
1848 
1849       k = &data->req;
1850 
1851       if(!(k->keepon & KEEP_RECV))
1852         /* We're done receiving */
1853         Curl_pipeline_leave_read(data->easy_conn);
1854 
1855       if(!(k->keepon & KEEP_SEND))
1856         /* We're done sending */
1857         Curl_pipeline_leave_write(data->easy_conn);
1858 
1859       if(done || (result == CURLE_RECV_ERROR)) {
1860         /* If CURLE_RECV_ERROR happens early enough, we assume it was a race
1861          * condition and the server closed the re-used connection exactly when
1862          * we wanted to use it, so figure out if that is indeed the case.
1863          */
1864         CURLcode ret = Curl_retry_request(data->easy_conn, &newurl);
1865         if(!ret)
1866           retry = (newurl)?TRUE:FALSE;
1867 
1868         if(retry) {
1869           /* if we are to retry, set the result to OK and consider the
1870              request as done */
1871           result = CURLE_OK;
1872           done = TRUE;
1873         }
1874       }
1875 
1876       if(result) {
1877         /*
1878          * The transfer phase returned error, we mark the connection to get
1879          * closed to prevent being re-used. This is because we can't possibly
1880          * know if the connection is in a good shape or not now.  Unless it is
1881          * a protocol which uses two "channels" like FTP, as then the error
1882          * happened in the data connection.
1883          */
1884 
1885         if(!(data->easy_conn->handler->flags & PROTOPT_DUAL) &&
1886            result != CURLE_HTTP2_STREAM)
1887           connclose(data->easy_conn, "Transfer returned error");
1888 
1889         Curl_posttransfer(data);
1890         multi_done(&data->easy_conn, result, FALSE);
1891       }
1892       else if(done) {
1893         followtype follow=FOLLOW_NONE;
1894 
1895         /* call this even if the readwrite function returned error */
1896         Curl_posttransfer(data);
1897 
1898         /* we're no longer receiving */
1899         Curl_removeHandleFromPipeline(data, data->easy_conn->recv_pipe);
1900 
1901         /* expire the new receiving pipeline head */
1902         if(data->easy_conn->recv_pipe->head)
1903           Curl_expire_latest(data->easy_conn->recv_pipe->head->ptr, 1);
1904 
1905         /* Check if we can move pending requests to send pipe */
1906         Curl_multi_process_pending_handles(multi);
1907 
1908         /* When we follow redirects or is set to retry the connection, we must
1909            to go back to the CONNECT state */
1910         if(data->req.newurl || retry) {
1911           if(!retry) {
1912             /* if the URL is a follow-location and not just a retried request
1913                then figure out the URL here */
1914             free(newurl);
1915             newurl = data->req.newurl;
1916             data->req.newurl = NULL;
1917             follow = FOLLOW_REDIR;
1918           }
1919           else
1920             follow = FOLLOW_RETRY;
1921           result = multi_done(&data->easy_conn, CURLE_OK, FALSE);
1922           if(!result) {
1923             result = Curl_follow(data, newurl, follow);
1924             if(!result) {
1925               multistate(data, CURLM_STATE_CONNECT);
1926               rc = CURLM_CALL_MULTI_PERFORM;
1927               newurl = NULL; /* handed over the memory ownership to
1928                                 Curl_follow(), make sure we don't free() it
1929                                 here */
1930             }
1931           }
1932         }
1933         else {
1934           /* after the transfer is done, go DONE */
1935 
1936           /* but first check to see if we got a location info even though we're
1937              not following redirects */
1938           if(data->req.location) {
1939             free(newurl);
1940             newurl = data->req.location;
1941             data->req.location = NULL;
1942             result = Curl_follow(data, newurl, FOLLOW_FAKE);
1943             if(!result)
1944               newurl = NULL; /* allocation was handed over Curl_follow() */
1945             else
1946               disconnect_conn = TRUE;
1947           }
1948 
1949           multistate(data, CURLM_STATE_DONE);
1950           rc = CURLM_CALL_MULTI_PERFORM;
1951         }
1952       }
1953 
1954       free(newurl);
1955       break;
1956     }
1957 
1958     case CURLM_STATE_DONE:
1959       /* this state is highly transient, so run another loop after this */
1960       rc = CURLM_CALL_MULTI_PERFORM;
1961 
1962       if(data->easy_conn) {
1963         CURLcode res;
1964 
1965         /* Remove ourselves from the receive pipeline, if we are there. */
1966         Curl_removeHandleFromPipeline(data, data->easy_conn->recv_pipe);
1967         /* Check if we can move pending requests to send pipe */
1968         Curl_multi_process_pending_handles(multi);
1969 
1970         /* post-transfer command */
1971         res = multi_done(&data->easy_conn, result, FALSE);
1972 
1973         /* allow a previously set error code take precedence */
1974         if(!result)
1975           result = res;
1976 
1977         /*
1978          * If there are other handles on the pipeline, multi_done won't set
1979          * easy_conn to NULL.  In such a case, curl_multi_remove_handle() can
1980          * access free'd data, if the connection is free'd and the handle
1981          * removed before we perform the processing in CURLM_STATE_COMPLETED
1982          */
1983         if(data->easy_conn)
1984           data->easy_conn = NULL;
1985       }
1986 
1987       if(data->set.wildcardmatch) {
1988         if(data->wildcard.state != CURLWC_DONE) {
1989           /* if a wildcard is set and we are not ending -> lets start again
1990              with CURLM_STATE_INIT */
1991           multistate(data, CURLM_STATE_INIT);
1992           break;
1993         }
1994       }
1995 
1996       /* after we have DONE what we're supposed to do, go COMPLETED, and
1997          it doesn't matter what the multi_done() returned! */
1998       multistate(data, CURLM_STATE_COMPLETED);
1999       break;
2000 
2001     case CURLM_STATE_COMPLETED:
2002       /* this is a completed transfer, it is likely to still be connected */
2003 
2004       /* This node should be delinked from the list now and we should post
2005          an information message that we are complete. */
2006 
2007       /* Important: reset the conn pointer so that we don't point to memory
2008          that could be freed anytime */
2009       data->easy_conn = NULL;
2010 
2011       Curl_expire(data, 0); /* stop all timers */
2012       break;
2013 
2014     case CURLM_STATE_MSGSENT:
2015       data->result = result;
2016       return CURLM_OK; /* do nothing */
2017 
2018     default:
2019       return CURLM_INTERNAL_ERROR;
2020     }
2021     statemachine_end:
2022 
2023     if(data->mstate < CURLM_STATE_COMPLETED) {
2024       if(result) {
2025         /*
2026          * If an error was returned, and we aren't in completed state now,
2027          * then we go to completed and consider this transfer aborted.
2028          */
2029 
2030         /* NOTE: no attempt to disconnect connections must be made
2031            in the case blocks above - cleanup happens only here */
2032 
2033         data->state.pipe_broke = FALSE;
2034 
2035         /* Check if we can move pending requests to send pipe */
2036         Curl_multi_process_pending_handles(multi);
2037 
2038         if(data->easy_conn) {
2039           /* if this has a connection, unsubscribe from the pipelines */
2040           Curl_pipeline_leave_write(data->easy_conn);
2041           Curl_pipeline_leave_read(data->easy_conn);
2042           Curl_removeHandleFromPipeline(data, data->easy_conn->send_pipe);
2043           Curl_removeHandleFromPipeline(data, data->easy_conn->recv_pipe);
2044 
2045           if(disconnect_conn) {
2046             /* Don't attempt to send data over a connection that timed out */
2047             bool dead_connection = result == CURLE_OPERATION_TIMEDOUT;
2048             /* disconnect properly */
2049             Curl_disconnect(data->easy_conn, dead_connection);
2050 
2051             /* This is where we make sure that the easy_conn pointer is reset.
2052                We don't have to do this in every case block above where a
2053                failure is detected */
2054             data->easy_conn = NULL;
2055           }
2056         }
2057         else if(data->mstate == CURLM_STATE_CONNECT) {
2058           /* Curl_connect() failed */
2059           (void)Curl_posttransfer(data);
2060         }
2061 
2062         multistate(data, CURLM_STATE_COMPLETED);
2063       }
2064       /* if there's still a connection to use, call the progress function */
2065       else if(data->easy_conn && Curl_pgrsUpdate(data->easy_conn)) {
2066         /* aborted due to progress callback return code must close the
2067            connection */
2068         result = CURLE_ABORTED_BY_CALLBACK;
2069         connclose(data->easy_conn, "Aborted by callback");
2070 
2071         /* if not yet in DONE state, go there, otherwise COMPLETED */
2072         multistate(data, (data->mstate < CURLM_STATE_DONE)?
2073                    CURLM_STATE_DONE: CURLM_STATE_COMPLETED);
2074         rc = CURLM_CALL_MULTI_PERFORM;
2075       }
2076     }
2077 
2078     if(CURLM_STATE_COMPLETED == data->mstate) {
2079       /* now fill in the Curl_message with this info */
2080       msg = &data->msg;
2081 
2082       msg->extmsg.msg = CURLMSG_DONE;
2083       msg->extmsg.easy_handle = data;
2084       msg->extmsg.data.result = result;
2085 
2086       rc = multi_addmsg(multi, msg);
2087 
2088       multistate(data, CURLM_STATE_MSGSENT);
2089     }
2090   } while((rc == CURLM_CALL_MULTI_PERFORM) || multi_ischanged(multi, FALSE));
2091 
2092   data->result = result;
2093 
2094 
2095   return rc;
2096 }
2097 
2098 
curl_multi_perform(struct Curl_multi * multi,int * running_handles)2099 CURLMcode curl_multi_perform(struct Curl_multi *multi, int *running_handles)
2100 {
2101   struct Curl_easy *data;
2102   CURLMcode returncode=CURLM_OK;
2103   struct Curl_tree *t;
2104   struct timeval now = Curl_tvnow();
2105 
2106   if(!GOOD_MULTI_HANDLE(multi))
2107     return CURLM_BAD_HANDLE;
2108 
2109   data=multi->easyp;
2110   while(data) {
2111     CURLMcode result;
2112     SIGPIPE_VARIABLE(pipe_st);
2113 
2114     sigpipe_ignore(data, &pipe_st);
2115     result = multi_runsingle(multi, now, data);
2116     sigpipe_restore(&pipe_st);
2117 
2118     if(result)
2119       returncode = result;
2120 
2121     data = data->next; /* operate on next handle */
2122   }
2123 
2124   /*
2125    * Simply remove all expired timers from the splay since handles are dealt
2126    * with unconditionally by this function and curl_multi_timeout() requires
2127    * that already passed/handled expire times are removed from the splay.
2128    *
2129    * It is important that the 'now' value is set at the entry of this function
2130    * and not for the current time as it may have ticked a little while since
2131    * then and then we risk this loop to remove timers that actually have not
2132    * been handled!
2133    */
2134   do {
2135     multi->timetree = Curl_splaygetbest(now, multi->timetree, &t);
2136     if(t)
2137       /* the removed may have another timeout in queue */
2138       (void)add_next_timeout(now, multi, t->payload);
2139 
2140   } while(t);
2141 
2142   *running_handles = multi->num_alive;
2143 
2144   if(CURLM_OK >= returncode)
2145     update_timer(multi);
2146 
2147   return returncode;
2148 }
2149 
close_all_connections(struct Curl_multi * multi)2150 static void close_all_connections(struct Curl_multi *multi)
2151 {
2152   struct connectdata *conn;
2153 
2154   conn = Curl_conncache_find_first_connection(&multi->conn_cache);
2155   while(conn) {
2156     SIGPIPE_VARIABLE(pipe_st);
2157     conn->data = multi->closure_handle;
2158 
2159     sigpipe_ignore(conn->data, &pipe_st);
2160     conn->data->easy_conn = NULL; /* clear the easy handle's connection
2161                                      pointer */
2162     /* This will remove the connection from the cache */
2163     (void)Curl_disconnect(conn, FALSE);
2164     sigpipe_restore(&pipe_st);
2165 
2166     conn = Curl_conncache_find_first_connection(&multi->conn_cache);
2167   }
2168 }
2169 
curl_multi_cleanup(struct Curl_multi * multi)2170 CURLMcode curl_multi_cleanup(struct Curl_multi *multi)
2171 {
2172   struct Curl_easy *data;
2173   struct Curl_easy *nextdata;
2174 
2175   if(GOOD_MULTI_HANDLE(multi)) {
2176     bool restore_pipe = FALSE;
2177     SIGPIPE_VARIABLE(pipe_st);
2178 
2179     multi->type = 0; /* not good anymore */
2180 
2181     /* Close all the connections in the connection cache */
2182     close_all_connections(multi);
2183 
2184     if(multi->closure_handle) {
2185       sigpipe_ignore(multi->closure_handle, &pipe_st);
2186       restore_pipe = TRUE;
2187 
2188       multi->closure_handle->dns.hostcache = &multi->hostcache;
2189       Curl_hostcache_clean(multi->closure_handle,
2190                            multi->closure_handle->dns.hostcache);
2191 
2192       Curl_close(multi->closure_handle);
2193     }
2194 
2195     Curl_hash_destroy(&multi->sockhash);
2196     Curl_conncache_destroy(&multi->conn_cache);
2197     Curl_llist_destroy(multi->msglist, NULL);
2198     Curl_llist_destroy(multi->pending, NULL);
2199 
2200     /* remove all easy handles */
2201     data = multi->easyp;
2202     while(data) {
2203       nextdata=data->next;
2204       if(data->dns.hostcachetype == HCACHE_MULTI) {
2205         /* clear out the usage of the shared DNS cache */
2206         Curl_hostcache_clean(data, data->dns.hostcache);
2207         data->dns.hostcache = NULL;
2208         data->dns.hostcachetype = HCACHE_NONE;
2209       }
2210 
2211       /* Clear the pointer to the connection cache */
2212       data->state.conn_cache = NULL;
2213       data->multi = NULL; /* clear the association */
2214 
2215       data = nextdata;
2216     }
2217 
2218     Curl_hash_destroy(&multi->hostcache);
2219 
2220     /* Free the blacklists by setting them to NULL */
2221     Curl_pipeline_set_site_blacklist(NULL, &multi->pipelining_site_bl);
2222     Curl_pipeline_set_server_blacklist(NULL, &multi->pipelining_server_bl);
2223 
2224     free(multi);
2225     if(restore_pipe)
2226       sigpipe_restore(&pipe_st);
2227 
2228     return CURLM_OK;
2229   }
2230   else
2231     return CURLM_BAD_HANDLE;
2232 }
2233 
2234 /*
2235  * curl_multi_info_read()
2236  *
2237  * This function is the primary way for a multi/multi_socket application to
2238  * figure out if a transfer has ended. We MUST make this function as fast as
2239  * possible as it will be polled frequently and we MUST NOT scan any lists in
2240  * here to figure out things. We must scale fine to thousands of handles and
2241  * beyond. The current design is fully O(1).
2242  */
2243 
curl_multi_info_read(struct Curl_multi * multi,int * msgs_in_queue)2244 CURLMsg *curl_multi_info_read(struct Curl_multi *multi, int *msgs_in_queue)
2245 {
2246   struct Curl_message *msg;
2247 
2248   *msgs_in_queue = 0; /* default to none */
2249 
2250   if(GOOD_MULTI_HANDLE(multi) && Curl_llist_count(multi->msglist)) {
2251     /* there is one or more messages in the list */
2252     struct curl_llist_element *e;
2253 
2254     /* extract the head of the list to return */
2255     e = multi->msglist->head;
2256 
2257     msg = e->ptr;
2258 
2259     /* remove the extracted entry */
2260     Curl_llist_remove(multi->msglist, e, NULL);
2261 
2262     *msgs_in_queue = curlx_uztosi(Curl_llist_count(multi->msglist));
2263 
2264     return &msg->extmsg;
2265   }
2266   else
2267     return NULL;
2268 }
2269 
2270 /*
2271  * singlesocket() checks what sockets we deal with and their "action state"
2272  * and if we have a different state in any of those sockets from last time we
2273  * call the callback accordingly.
2274  */
singlesocket(struct Curl_multi * multi,struct Curl_easy * data)2275 static void singlesocket(struct Curl_multi *multi,
2276                          struct Curl_easy *data)
2277 {
2278   curl_socket_t socks[MAX_SOCKSPEREASYHANDLE];
2279   int i;
2280   struct Curl_sh_entry *entry;
2281   curl_socket_t s;
2282   int num;
2283   unsigned int curraction;
2284 
2285   for(i=0; i< MAX_SOCKSPEREASYHANDLE; i++)
2286     socks[i] = CURL_SOCKET_BAD;
2287 
2288   /* Fill in the 'current' struct with the state as it is now: what sockets to
2289      supervise and for what actions */
2290   curraction = multi_getsock(data, socks, MAX_SOCKSPEREASYHANDLE);
2291 
2292   /* We have 0 .. N sockets already and we get to know about the 0 .. M
2293      sockets we should have from now on. Detect the differences, remove no
2294      longer supervised ones and add new ones */
2295 
2296   /* walk over the sockets we got right now */
2297   for(i=0; (i< MAX_SOCKSPEREASYHANDLE) &&
2298         (curraction & (GETSOCK_READSOCK(i) | GETSOCK_WRITESOCK(i)));
2299       i++) {
2300     int action = CURL_POLL_NONE;
2301 
2302     s = socks[i];
2303 
2304     /* get it from the hash */
2305     entry = sh_getentry(&multi->sockhash, s);
2306 
2307     if(curraction & GETSOCK_READSOCK(i))
2308       action |= CURL_POLL_IN;
2309     if(curraction & GETSOCK_WRITESOCK(i))
2310       action |= CURL_POLL_OUT;
2311 
2312     if(entry) {
2313       /* yeps, already present so check if it has the same action set */
2314       if(entry->action == action)
2315         /* same, continue */
2316         continue;
2317     }
2318     else {
2319       /* this is a socket we didn't have before, add it! */
2320       entry = sh_addentry(&multi->sockhash, s, data);
2321       if(!entry)
2322         /* fatal */
2323         return;
2324     }
2325 
2326     /* we know (entry != NULL) at this point, see the logic above */
2327     if(multi->socket_cb)
2328       multi->socket_cb(data,
2329                        s,
2330                        action,
2331                        multi->socket_userp,
2332                        entry->socketp);
2333 
2334     entry->action = action; /* store the current action state */
2335   }
2336 
2337   num = i; /* number of sockets */
2338 
2339   /* when we've walked over all the sockets we should have right now, we must
2340      make sure to detect sockets that are removed */
2341   for(i=0; i< data->numsocks; i++) {
2342     int j;
2343     s = data->sockets[i];
2344     for(j=0; j<num; j++) {
2345       if(s == socks[j]) {
2346         /* this is still supervised */
2347         s = CURL_SOCKET_BAD;
2348         break;
2349       }
2350     }
2351 
2352     entry = sh_getentry(&multi->sockhash, s);
2353     if(entry) {
2354       /* this socket has been removed. Tell the app to remove it */
2355       bool remove_sock_from_hash = TRUE;
2356 
2357       /* check if the socket to be removed serves a connection which has
2358          other easy-s in a pipeline. In this case the socket should not be
2359          removed. */
2360       struct connectdata *easy_conn = data->easy_conn;
2361       if(easy_conn) {
2362         if(easy_conn->recv_pipe && easy_conn->recv_pipe->size > 1) {
2363           /* the handle should not be removed from the pipe yet */
2364           remove_sock_from_hash = FALSE;
2365 
2366           /* Update the sockhash entry to instead point to the next in line
2367              for the recv_pipe, or the first (in case this particular easy
2368              isn't already) */
2369           if(entry->easy == data) {
2370             if(Curl_recvpipe_head(data, easy_conn))
2371               entry->easy = easy_conn->recv_pipe->head->next->ptr;
2372             else
2373               entry->easy = easy_conn->recv_pipe->head->ptr;
2374           }
2375         }
2376         if(easy_conn->send_pipe  && easy_conn->send_pipe->size > 1) {
2377           /* the handle should not be removed from the pipe yet */
2378           remove_sock_from_hash = FALSE;
2379 
2380           /* Update the sockhash entry to instead point to the next in line
2381              for the send_pipe, or the first (in case this particular easy
2382              isn't already) */
2383           if(entry->easy == data) {
2384             if(Curl_sendpipe_head(data, easy_conn))
2385               entry->easy = easy_conn->send_pipe->head->next->ptr;
2386             else
2387               entry->easy = easy_conn->send_pipe->head->ptr;
2388           }
2389         }
2390         /* Don't worry about overwriting recv_pipe head with send_pipe_head,
2391            when action will be asked on the socket (see multi_socket()), the
2392            head of the correct pipe will be taken according to the
2393            action. */
2394       }
2395 
2396       if(remove_sock_from_hash) {
2397         /* in this case 'entry' is always non-NULL */
2398         if(multi->socket_cb)
2399           multi->socket_cb(data,
2400                            s,
2401                            CURL_POLL_REMOVE,
2402                            multi->socket_userp,
2403                            entry->socketp);
2404         sh_delentry(&multi->sockhash, s);
2405       }
2406     } /* if sockhash entry existed */
2407   } /* for loop over numsocks */
2408 
2409   memcpy(data->sockets, socks, num*sizeof(curl_socket_t));
2410   data->numsocks = num;
2411 }
2412 
2413 /*
2414  * Curl_multi_closed()
2415  *
2416  * Used by the connect code to tell the multi_socket code that one of the
2417  * sockets we were using is about to be closed.  This function will then
2418  * remove it from the sockethash for this handle to make the multi_socket API
2419  * behave properly, especially for the case when libcurl will create another
2420  * socket again and it gets the same file descriptor number.
2421  */
2422 
Curl_multi_closed(struct connectdata * conn,curl_socket_t s)2423 void Curl_multi_closed(struct connectdata *conn, curl_socket_t s)
2424 {
2425   struct Curl_multi *multi = conn->data->multi;
2426   if(multi) {
2427     /* this is set if this connection is part of a handle that is added to
2428        a multi handle, and only then this is necessary */
2429     struct Curl_sh_entry *entry = sh_getentry(&multi->sockhash, s);
2430 
2431     if(entry) {
2432       if(multi->socket_cb)
2433         multi->socket_cb(conn->data, s, CURL_POLL_REMOVE,
2434                          multi->socket_userp,
2435                          entry->socketp);
2436 
2437       /* now remove it from the socket hash */
2438       sh_delentry(&multi->sockhash, s);
2439     }
2440   }
2441 }
2442 
2443 
2444 
2445 /*
2446  * add_next_timeout()
2447  *
2448  * Each Curl_easy has a list of timeouts. The add_next_timeout() is called
2449  * when it has just been removed from the splay tree because the timeout has
2450  * expired. This function is then to advance in the list to pick the next
2451  * timeout to use (skip the already expired ones) and add this node back to
2452  * the splay tree again.
2453  *
2454  * The splay tree only has each sessionhandle as a single node and the nearest
2455  * timeout is used to sort it on.
2456  */
add_next_timeout(struct timeval now,struct Curl_multi * multi,struct Curl_easy * d)2457 static CURLMcode add_next_timeout(struct timeval now,
2458                                   struct Curl_multi *multi,
2459                                   struct Curl_easy *d)
2460 {
2461   struct timeval *tv = &d->state.expiretime;
2462   struct curl_llist *list = d->state.timeoutlist;
2463   struct curl_llist_element *e;
2464 
2465   /* move over the timeout list for this specific handle and remove all
2466      timeouts that are now passed tense and store the next pending
2467      timeout in *tv */
2468   for(e = list->head; e;) {
2469     struct curl_llist_element *n = e->next;
2470     long diff = curlx_tvdiff(*(struct timeval *)e->ptr, now);
2471     if(diff <= 0)
2472       /* remove outdated entry */
2473       Curl_llist_remove(list, e, NULL);
2474     else
2475       /* the list is sorted so get out on the first mismatch */
2476       break;
2477     e = n;
2478   }
2479   e = list->head;
2480   if(!e) {
2481     /* clear the expire times within the handles that we remove from the
2482        splay tree */
2483     tv->tv_sec = 0;
2484     tv->tv_usec = 0;
2485   }
2486   else {
2487     /* copy the first entry to 'tv' */
2488     memcpy(tv, e->ptr, sizeof(*tv));
2489 
2490     /* remove first entry from list */
2491     Curl_llist_remove(list, e, NULL);
2492 
2493     /* insert this node again into the splay */
2494     multi->timetree = Curl_splayinsert(*tv, multi->timetree,
2495                                        &d->state.timenode);
2496   }
2497   return CURLM_OK;
2498 }
2499 
multi_socket(struct Curl_multi * multi,bool checkall,curl_socket_t s,int ev_bitmask,int * running_handles)2500 static CURLMcode multi_socket(struct Curl_multi *multi,
2501                               bool checkall,
2502                               curl_socket_t s,
2503                               int ev_bitmask,
2504                               int *running_handles)
2505 {
2506   CURLMcode result = CURLM_OK;
2507   struct Curl_easy *data = NULL;
2508   struct Curl_tree *t;
2509   struct timeval now = Curl_tvnow();
2510 
2511   if(checkall) {
2512     /* *perform() deals with running_handles on its own */
2513     result = curl_multi_perform(multi, running_handles);
2514 
2515     /* walk through each easy handle and do the socket state change magic
2516        and callbacks */
2517     if(result != CURLM_BAD_HANDLE) {
2518       data=multi->easyp;
2519       while(data) {
2520         singlesocket(multi, data);
2521         data = data->next;
2522       }
2523     }
2524 
2525     /* or should we fall-through and do the timer-based stuff? */
2526     return result;
2527   }
2528   else if(s != CURL_SOCKET_TIMEOUT) {
2529 
2530     struct Curl_sh_entry *entry = sh_getentry(&multi->sockhash, s);
2531 
2532     if(!entry)
2533       /* Unmatched socket, we can't act on it but we ignore this fact.  In
2534          real-world tests it has been proved that libevent can in fact give
2535          the application actions even though the socket was just previously
2536          asked to get removed, so thus we better survive stray socket actions
2537          and just move on. */
2538       ;
2539     else {
2540       SIGPIPE_VARIABLE(pipe_st);
2541 
2542       data = entry->easy;
2543 
2544       if(data->magic != CURLEASY_MAGIC_NUMBER)
2545         /* bad bad bad bad bad bad bad */
2546         return CURLM_INTERNAL_ERROR;
2547 
2548       /* If the pipeline is enabled, take the handle which is in the head of
2549          the pipeline. If we should write into the socket, take the send_pipe
2550          head.  If we should read from the socket, take the recv_pipe head. */
2551       if(data->easy_conn) {
2552         if((ev_bitmask & CURL_POLL_OUT) &&
2553            data->easy_conn->send_pipe &&
2554            data->easy_conn->send_pipe->head)
2555           data = data->easy_conn->send_pipe->head->ptr;
2556         else if((ev_bitmask & CURL_POLL_IN) &&
2557                 data->easy_conn->recv_pipe &&
2558                 data->easy_conn->recv_pipe->head)
2559           data = data->easy_conn->recv_pipe->head->ptr;
2560       }
2561 
2562       if(data->easy_conn &&
2563          !(data->easy_conn->handler->flags & PROTOPT_DIRLOCK))
2564         /* set socket event bitmask if they're not locked */
2565         data->easy_conn->cselect_bits = ev_bitmask;
2566 
2567       sigpipe_ignore(data, &pipe_st);
2568       result = multi_runsingle(multi, now, data);
2569       sigpipe_restore(&pipe_st);
2570 
2571       if(data->easy_conn &&
2572          !(data->easy_conn->handler->flags & PROTOPT_DIRLOCK))
2573         /* clear the bitmask only if not locked */
2574         data->easy_conn->cselect_bits = 0;
2575 
2576       if(CURLM_OK >= result)
2577         /* get the socket(s) and check if the state has been changed since
2578            last */
2579         singlesocket(multi, data);
2580 
2581       /* Now we fall-through and do the timer-based stuff, since we don't want
2582          to force the user to have to deal with timeouts as long as at least
2583          one connection in fact has traffic. */
2584 
2585       data = NULL; /* set data to NULL again to avoid calling
2586                       multi_runsingle() in case there's no need to */
2587       now = Curl_tvnow(); /* get a newer time since the multi_runsingle() loop
2588                              may have taken some time */
2589     }
2590   }
2591   else {
2592     /* Asked to run due to time-out. Clear the 'lastcall' variable to force
2593        update_timer() to trigger a callback to the app again even if the same
2594        timeout is still the one to run after this call. That handles the case
2595        when the application asks libcurl to run the timeout prematurely. */
2596     memset(&multi->timer_lastcall, 0, sizeof(multi->timer_lastcall));
2597   }
2598 
2599   /*
2600    * The loop following here will go on as long as there are expire-times left
2601    * to process in the splay and 'data' will be re-assigned for every expired
2602    * handle we deal with.
2603    */
2604   do {
2605     /* the first loop lap 'data' can be NULL */
2606     if(data) {
2607       SIGPIPE_VARIABLE(pipe_st);
2608 
2609       sigpipe_ignore(data, &pipe_st);
2610       result = multi_runsingle(multi, now, data);
2611       sigpipe_restore(&pipe_st);
2612 
2613       if(CURLM_OK >= result)
2614         /* get the socket(s) and check if the state has been changed since
2615            last */
2616         singlesocket(multi, data);
2617     }
2618 
2619     /* Check if there's one (more) expired timer to deal with! This function
2620        extracts a matching node if there is one */
2621 
2622     multi->timetree = Curl_splaygetbest(now, multi->timetree, &t);
2623     if(t) {
2624       data = t->payload; /* assign this for next loop */
2625       (void)add_next_timeout(now, multi, t->payload);
2626     }
2627 
2628   } while(t);
2629 
2630   *running_handles = multi->num_alive;
2631   return result;
2632 }
2633 
2634 #undef curl_multi_setopt
curl_multi_setopt(struct Curl_multi * multi,CURLMoption option,...)2635 CURLMcode curl_multi_setopt(struct Curl_multi *multi,
2636                             CURLMoption option, ...)
2637 {
2638   CURLMcode res = CURLM_OK;
2639   va_list param;
2640 
2641   if(!GOOD_MULTI_HANDLE(multi))
2642     return CURLM_BAD_HANDLE;
2643 
2644   va_start(param, option);
2645 
2646   switch(option) {
2647   case CURLMOPT_SOCKETFUNCTION:
2648     multi->socket_cb = va_arg(param, curl_socket_callback);
2649     break;
2650   case CURLMOPT_SOCKETDATA:
2651     multi->socket_userp = va_arg(param, void *);
2652     break;
2653   case CURLMOPT_PUSHFUNCTION:
2654     multi->push_cb = va_arg(param, curl_push_callback);
2655     break;
2656   case CURLMOPT_PUSHDATA:
2657     multi->push_userp = va_arg(param, void *);
2658     break;
2659   case CURLMOPT_PIPELINING:
2660     multi->pipelining = va_arg(param, long);
2661     break;
2662   case CURLMOPT_TIMERFUNCTION:
2663     multi->timer_cb = va_arg(param, curl_multi_timer_callback);
2664     break;
2665   case CURLMOPT_TIMERDATA:
2666     multi->timer_userp = va_arg(param, void *);
2667     break;
2668   case CURLMOPT_MAXCONNECTS:
2669     multi->maxconnects = va_arg(param, long);
2670     break;
2671   case CURLMOPT_MAX_HOST_CONNECTIONS:
2672     multi->max_host_connections = va_arg(param, long);
2673     break;
2674   case CURLMOPT_MAX_PIPELINE_LENGTH:
2675     multi->max_pipeline_length = va_arg(param, long);
2676     break;
2677   case CURLMOPT_CONTENT_LENGTH_PENALTY_SIZE:
2678     multi->content_length_penalty_size = va_arg(param, long);
2679     break;
2680   case CURLMOPT_CHUNK_LENGTH_PENALTY_SIZE:
2681     multi->chunk_length_penalty_size = va_arg(param, long);
2682     break;
2683   case CURLMOPT_PIPELINING_SITE_BL:
2684     res = Curl_pipeline_set_site_blacklist(va_arg(param, char **),
2685                                            &multi->pipelining_site_bl);
2686     break;
2687   case CURLMOPT_PIPELINING_SERVER_BL:
2688     res = Curl_pipeline_set_server_blacklist(va_arg(param, char **),
2689                                              &multi->pipelining_server_bl);
2690     break;
2691   case CURLMOPT_MAX_TOTAL_CONNECTIONS:
2692     multi->max_total_connections = va_arg(param, long);
2693     break;
2694   default:
2695     res = CURLM_UNKNOWN_OPTION;
2696     break;
2697   }
2698   va_end(param);
2699   return res;
2700 }
2701 
2702 /* we define curl_multi_socket() in the public multi.h header */
2703 #undef curl_multi_socket
2704 
curl_multi_socket(struct Curl_multi * multi,curl_socket_t s,int * running_handles)2705 CURLMcode curl_multi_socket(struct Curl_multi *multi, curl_socket_t s,
2706                             int *running_handles)
2707 {
2708   CURLMcode result = multi_socket(multi, FALSE, s, 0, running_handles);
2709   if(CURLM_OK >= result)
2710     update_timer(multi);
2711   return result;
2712 }
2713 
curl_multi_socket_action(struct Curl_multi * multi,curl_socket_t s,int ev_bitmask,int * running_handles)2714 CURLMcode curl_multi_socket_action(struct Curl_multi *multi, curl_socket_t s,
2715                                    int ev_bitmask, int *running_handles)
2716 {
2717   CURLMcode result = multi_socket(multi, FALSE, s,
2718                                   ev_bitmask, running_handles);
2719   if(CURLM_OK >= result)
2720     update_timer(multi);
2721   return result;
2722 }
2723 
curl_multi_socket_all(struct Curl_multi * multi,int * running_handles)2724 CURLMcode curl_multi_socket_all(struct Curl_multi *multi, int *running_handles)
2725 
2726 {
2727   CURLMcode result = multi_socket(multi, TRUE, CURL_SOCKET_BAD, 0,
2728                                   running_handles);
2729   if(CURLM_OK >= result)
2730     update_timer(multi);
2731   return result;
2732 }
2733 
multi_timeout(struct Curl_multi * multi,long * timeout_ms)2734 static CURLMcode multi_timeout(struct Curl_multi *multi,
2735                                long *timeout_ms)
2736 {
2737   static struct timeval tv_zero = {0, 0};
2738 
2739   if(multi->timetree) {
2740     /* we have a tree of expire times */
2741     struct timeval now = Curl_tvnow();
2742 
2743     /* splay the lowest to the bottom */
2744     multi->timetree = Curl_splay(tv_zero, multi->timetree);
2745 
2746     if(Curl_splaycomparekeys(multi->timetree->key, now) > 0) {
2747       /* some time left before expiration */
2748       *timeout_ms = curlx_tvdiff(multi->timetree->key, now);
2749       if(!*timeout_ms)
2750         /*
2751          * Since we only provide millisecond resolution on the returned value
2752          * and the diff might be less than one millisecond here, we don't
2753          * return zero as that may cause short bursts of busyloops on fast
2754          * processors while the diff is still present but less than one
2755          * millisecond! instead we return 1 until the time is ripe.
2756          */
2757         *timeout_ms=1;
2758     }
2759     else
2760       /* 0 means immediately */
2761       *timeout_ms = 0;
2762   }
2763   else
2764     *timeout_ms = -1;
2765 
2766   return CURLM_OK;
2767 }
2768 
curl_multi_timeout(struct Curl_multi * multi,long * timeout_ms)2769 CURLMcode curl_multi_timeout(struct Curl_multi *multi,
2770                              long *timeout_ms)
2771 {
2772   /* First, make some basic checks that the CURLM handle is a good handle */
2773   if(!GOOD_MULTI_HANDLE(multi))
2774     return CURLM_BAD_HANDLE;
2775 
2776   return multi_timeout(multi, timeout_ms);
2777 }
2778 
2779 /*
2780  * Tell the application it should update its timers, if it subscribes to the
2781  * update timer callback.
2782  */
update_timer(struct Curl_multi * multi)2783 static int update_timer(struct Curl_multi *multi)
2784 {
2785   long timeout_ms;
2786 
2787   if(!multi->timer_cb)
2788     return 0;
2789   if(multi_timeout(multi, &timeout_ms)) {
2790     return -1;
2791   }
2792   if(timeout_ms < 0) {
2793     static const struct timeval none={0, 0};
2794     if(Curl_splaycomparekeys(none, multi->timer_lastcall)) {
2795       multi->timer_lastcall = none;
2796       /* there's no timeout now but there was one previously, tell the app to
2797          disable it */
2798       return multi->timer_cb(multi, -1, multi->timer_userp);
2799     }
2800     return 0;
2801   }
2802 
2803   /* When multi_timeout() is done, multi->timetree points to the node with the
2804    * timeout we got the (relative) time-out time for. We can thus easily check
2805    * if this is the same (fixed) time as we got in a previous call and then
2806    * avoid calling the callback again. */
2807   if(Curl_splaycomparekeys(multi->timetree->key, multi->timer_lastcall) == 0)
2808     return 0;
2809 
2810   multi->timer_lastcall = multi->timetree->key;
2811 
2812   return multi->timer_cb(multi, timeout_ms, multi->timer_userp);
2813 }
2814 
2815 /*
2816  * multi_freetimeout()
2817  *
2818  * Callback used by the llist system when a single timeout list entry is
2819  * destroyed.
2820  */
multi_freetimeout(void * user,void * entryptr)2821 static void multi_freetimeout(void *user, void *entryptr)
2822 {
2823   (void)user;
2824 
2825   /* the entry was plain malloc()'ed */
2826   free(entryptr);
2827 }
2828 
2829 /*
2830  * multi_addtimeout()
2831  *
2832  * Add a timestamp to the list of timeouts. Keep the list sorted so that head
2833  * of list is always the timeout nearest in time.
2834  *
2835  */
2836 static CURLMcode
multi_addtimeout(struct curl_llist * timeoutlist,struct timeval * stamp)2837 multi_addtimeout(struct curl_llist *timeoutlist,
2838                  struct timeval *stamp)
2839 {
2840   struct curl_llist_element *e;
2841   struct timeval *timedup;
2842   struct curl_llist_element *prev = NULL;
2843 
2844   timedup = malloc(sizeof(*timedup));
2845   if(!timedup)
2846     return CURLM_OUT_OF_MEMORY;
2847 
2848   /* copy the timestamp */
2849   memcpy(timedup, stamp, sizeof(*timedup));
2850 
2851   if(Curl_llist_count(timeoutlist)) {
2852     /* find the correct spot in the list */
2853     for(e = timeoutlist->head; e; e = e->next) {
2854       struct timeval *checktime = e->ptr;
2855       long diff = curlx_tvdiff(*checktime, *timedup);
2856       if(diff > 0)
2857         break;
2858       prev = e;
2859     }
2860 
2861   }
2862   /* else
2863      this is the first timeout on the list */
2864 
2865   if(!Curl_llist_insert_next(timeoutlist, prev, timedup)) {
2866     free(timedup);
2867     return CURLM_OUT_OF_MEMORY;
2868   }
2869 
2870   return CURLM_OK;
2871 }
2872 
2873 /*
2874  * Curl_expire()
2875  *
2876  * given a number of milliseconds from now to use to set the 'act before
2877  * this'-time for the transfer, to be extracted by curl_multi_timeout()
2878  *
2879  * Note that the timeout will be added to a queue of timeouts if it defines a
2880  * moment in time that is later than the current head of queue.
2881  *
2882  * Pass zero to clear all timeout values for this handle.
2883 */
Curl_expire(struct Curl_easy * data,long milli)2884 void Curl_expire(struct Curl_easy *data, long milli)
2885 {
2886   struct Curl_multi *multi = data->multi;
2887   struct timeval *nowp = &data->state.expiretime;
2888   int rc;
2889 
2890   /* this is only interesting while there is still an associated multi struct
2891      remaining! */
2892   if(!multi)
2893     return;
2894 
2895   if(!milli) {
2896     /* No timeout, clear the time data. */
2897     if(nowp->tv_sec || nowp->tv_usec) {
2898       /* Since this is an cleared time, we must remove the previous entry from
2899          the splay tree */
2900       struct curl_llist *list = data->state.timeoutlist;
2901 
2902       rc = Curl_splayremovebyaddr(multi->timetree,
2903                                   &data->state.timenode,
2904                                   &multi->timetree);
2905       if(rc)
2906         infof(data, "Internal error clearing splay node = %d\n", rc);
2907 
2908       /* flush the timeout list too */
2909       while(list->size > 0)
2910         Curl_llist_remove(list, list->tail, NULL);
2911 
2912 #ifdef DEBUGBUILD
2913       infof(data, "Expire cleared\n");
2914 #endif
2915       nowp->tv_sec = 0;
2916       nowp->tv_usec = 0;
2917     }
2918   }
2919   else {
2920     struct timeval set;
2921 
2922     set = Curl_tvnow();
2923     set.tv_sec += milli/1000;
2924     set.tv_usec += (milli%1000)*1000;
2925 
2926     if(set.tv_usec >= 1000000) {
2927       set.tv_sec++;
2928       set.tv_usec -= 1000000;
2929     }
2930 
2931     if(nowp->tv_sec || nowp->tv_usec) {
2932       /* This means that the struct is added as a node in the splay tree.
2933          Compare if the new time is earlier, and only remove-old/add-new if it
2934          is. */
2935       long diff = curlx_tvdiff(set, *nowp);
2936       if(diff > 0) {
2937         /* the new expire time was later so just add it to the queue
2938            and get out */
2939         multi_addtimeout(data->state.timeoutlist, &set);
2940         return;
2941       }
2942 
2943       /* the new time is newer than the presently set one, so add the current
2944          to the queue and update the head */
2945       multi_addtimeout(data->state.timeoutlist, nowp);
2946 
2947       /* Since this is an updated time, we must remove the previous entry from
2948          the splay tree first and then re-add the new value */
2949       rc = Curl_splayremovebyaddr(multi->timetree,
2950                                   &data->state.timenode,
2951                                   &multi->timetree);
2952       if(rc)
2953         infof(data, "Internal error removing splay node = %d\n", rc);
2954     }
2955 
2956     *nowp = set;
2957     data->state.timenode.payload = data;
2958     multi->timetree = Curl_splayinsert(*nowp,
2959                                        multi->timetree,
2960                                        &data->state.timenode);
2961   }
2962 #if 0
2963   Curl_splayprint(multi->timetree, 0, TRUE);
2964 #endif
2965 }
2966 
2967 /*
2968  * Curl_expire_latest()
2969  *
2970  * This is like Curl_expire() but will only add a timeout node to the list of
2971  * timers if there is no timeout that will expire before the given time.
2972  *
2973  * Use this function if the code logic risks calling this function many times
2974  * or if there's no particular conditional wait in the code for this specific
2975  * time-out period to expire.
2976  *
2977  */
Curl_expire_latest(struct Curl_easy * data,long milli)2978 void Curl_expire_latest(struct Curl_easy *data, long milli)
2979 {
2980   struct timeval *expire = &data->state.expiretime;
2981 
2982   struct timeval set;
2983 
2984   set = Curl_tvnow();
2985   set.tv_sec += milli / 1000;
2986   set.tv_usec += (milli % 1000) * 1000;
2987 
2988   if(set.tv_usec >= 1000000) {
2989     set.tv_sec++;
2990     set.tv_usec -= 1000000;
2991   }
2992 
2993   if(expire->tv_sec || expire->tv_usec) {
2994     /* This means that the struct is added as a node in the splay tree.
2995        Compare if the new time is earlier, and only remove-old/add-new if it
2996          is. */
2997     long diff = curlx_tvdiff(set, *expire);
2998     if(diff > 0)
2999       /* the new expire time was later than the top time, so just skip this */
3000       return;
3001   }
3002 
3003   /* Just add the timeout like normal */
3004   Curl_expire(data, milli);
3005 }
3006 
curl_multi_assign(struct Curl_multi * multi,curl_socket_t s,void * hashp)3007 CURLMcode curl_multi_assign(struct Curl_multi *multi, curl_socket_t s,
3008                             void *hashp)
3009 {
3010   struct Curl_sh_entry *there = NULL;
3011 
3012   there = sh_getentry(&multi->sockhash, s);
3013 
3014   if(!there)
3015     return CURLM_BAD_SOCKET;
3016 
3017   there->socketp = hashp;
3018 
3019   return CURLM_OK;
3020 }
3021 
Curl_multi_max_host_connections(struct Curl_multi * multi)3022 size_t Curl_multi_max_host_connections(struct Curl_multi *multi)
3023 {
3024   return multi ? multi->max_host_connections : 0;
3025 }
3026 
Curl_multi_max_total_connections(struct Curl_multi * multi)3027 size_t Curl_multi_max_total_connections(struct Curl_multi *multi)
3028 {
3029   return multi ? multi->max_total_connections : 0;
3030 }
3031 
Curl_multi_content_length_penalty_size(struct Curl_multi * multi)3032 curl_off_t Curl_multi_content_length_penalty_size(struct Curl_multi *multi)
3033 {
3034   return multi ? multi->content_length_penalty_size : 0;
3035 }
3036 
Curl_multi_chunk_length_penalty_size(struct Curl_multi * multi)3037 curl_off_t Curl_multi_chunk_length_penalty_size(struct Curl_multi *multi)
3038 {
3039   return multi ? multi->chunk_length_penalty_size : 0;
3040 }
3041 
Curl_multi_pipelining_site_bl(struct Curl_multi * multi)3042 struct curl_llist *Curl_multi_pipelining_site_bl(struct Curl_multi *multi)
3043 {
3044   return multi->pipelining_site_bl;
3045 }
3046 
Curl_multi_pipelining_server_bl(struct Curl_multi * multi)3047 struct curl_llist *Curl_multi_pipelining_server_bl(struct Curl_multi *multi)
3048 {
3049   return multi->pipelining_server_bl;
3050 }
3051 
Curl_multi_process_pending_handles(struct Curl_multi * multi)3052 void Curl_multi_process_pending_handles(struct Curl_multi *multi)
3053 {
3054   struct curl_llist_element *e = multi->pending->head;
3055 
3056   while(e) {
3057     struct Curl_easy *data = e->ptr;
3058     struct curl_llist_element *next = e->next;
3059 
3060     if(data->mstate == CURLM_STATE_CONNECT_PEND) {
3061       multistate(data, CURLM_STATE_CONNECT);
3062 
3063       /* Remove this node from the list */
3064       Curl_llist_remove(multi->pending, e, NULL);
3065 
3066       /* Make sure that the handle will be processed soonish. */
3067       Curl_expire_latest(data, 1);
3068     }
3069 
3070     e = next; /* operate on next handle */
3071   }
3072 }
3073 
3074 #ifdef DEBUGBUILD
Curl_multi_dump(struct Curl_multi * multi)3075 void Curl_multi_dump(struct Curl_multi *multi)
3076 {
3077   struct Curl_easy *data;
3078   int i;
3079   fprintf(stderr, "* Multi status: %d handles, %d alive\n",
3080           multi->num_easy, multi->num_alive);
3081   for(data=multi->easyp; data; data = data->next) {
3082     if(data->mstate < CURLM_STATE_COMPLETED) {
3083       /* only display handles that are not completed */
3084       fprintf(stderr, "handle %p, state %s, %d sockets\n",
3085               (void *)data,
3086               statename[data->mstate], data->numsocks);
3087       for(i=0; i < data->numsocks; i++) {
3088         curl_socket_t s = data->sockets[i];
3089         struct Curl_sh_entry *entry = sh_getentry(&multi->sockhash, s);
3090 
3091         fprintf(stderr, "%d ", (int)s);
3092         if(!entry) {
3093           fprintf(stderr, "INTERNAL CONFUSION\n");
3094           continue;
3095         }
3096         fprintf(stderr, "[%s %s] ",
3097                 entry->action&CURL_POLL_IN?"RECVING":"",
3098                 entry->action&CURL_POLL_OUT?"SENDING":"");
3099       }
3100       if(data->numsocks)
3101         fprintf(stderr, "\n");
3102     }
3103   }
3104 }
3105 #endif
3106