aboutsummaryrefslogtreecommitdiff
path: root/src/microhttpd/daemon.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/microhttpd/daemon.c')
-rw-r--r--src/microhttpd/daemon.c342
1 files changed, 182 insertions, 160 deletions
diff --git a/src/microhttpd/daemon.c b/src/microhttpd/daemon.c
index 2bce6d15..622ab34e 100644
--- a/src/microhttpd/daemon.c
+++ b/src/microhttpd/daemon.c
@@ -23,6 +23,7 @@
23 * @brief A minimal-HTTP server library 23 * @brief A minimal-HTTP server library
24 * @author Daniel Pittman 24 * @author Daniel Pittman
25 * @author Christian Grothoff 25 * @author Christian Grothoff
26 * @author Karlson2k (Evgeny Grin)
26 */ 27 */
27#include "platform.h" 28#include "platform.h"
28#include "mhd_threads.h" 29#include "mhd_threads.h"
@@ -688,6 +689,7 @@ urh_to_fdset (struct MHD_UpgradeResponseHandle *urh,
688 fd_setsize)) ) 689 fd_setsize)) )
689 return MHD_NO; 690 return MHD_NO;
690 if ( (0 == (MHD_EPOLL_STATE_WRITE_READY & urh->mhd.celi)) && 691 if ( (0 == (MHD_EPOLL_STATE_WRITE_READY & urh->mhd.celi)) &&
692 (MHD_NO == urh->was_closed) &&
691 (MHD_INVALID_SOCKET != urh->mhd.socket) && 693 (MHD_INVALID_SOCKET != urh->mhd.socket) &&
692 (! MHD_add_to_fd_set_ (urh->mhd.socket, 694 (! MHD_add_to_fd_set_ (urh->mhd.socket,
693 ws, 695 ws,
@@ -695,6 +697,7 @@ urh_to_fdset (struct MHD_UpgradeResponseHandle *urh,
695 fd_setsize)) ) 697 fd_setsize)) )
696 return MHD_NO; 698 return MHD_NO;
697 if ( (urh->in_buffer_used < urh->in_buffer_size) && 699 if ( (urh->in_buffer_used < urh->in_buffer_size) &&
700 (MHD_NO == urh->was_closed) &&
698 (MHD_INVALID_SOCKET != urh->connection->socket_fd) && 701 (MHD_INVALID_SOCKET != urh->connection->socket_fd) &&
699 (! MHD_add_to_fd_set_ (urh->connection->socket_fd, 702 (! MHD_add_to_fd_set_ (urh->connection->socket_fd,
700 rs, 703 rs,
@@ -934,73 +937,19 @@ call_handlers (struct MHD_Connection *con,
934void 937void
935MHD_cleanup_upgraded_connection_ (struct MHD_Connection *connection) 938MHD_cleanup_upgraded_connection_ (struct MHD_Connection *connection)
936{ 939{
937 struct MHD_Daemon *daemon = connection->daemon;
938 struct MHD_UpgradeResponseHandle *urh = connection->urh; 940 struct MHD_UpgradeResponseHandle *urh = connection->urh;
939 941
940#if HTTPS_SUPPORT 942 /* Signal remote client the end of TLS connection by
941 if ( (NULL != urh) && 943 * gracefully closing TLS session. */
942 (0 != (daemon->options & MHD_USE_TLS)) ) 944 if (0 != (connection->daemon->options & MHD_USE_TLS))
943 { 945 gnutls_bye (connection->tls_session,
944 if (0 == (daemon->options & MHD_USE_THREAD_PER_CONNECTION)) 946 GNUTLS_SHUT_WR);
945 DLL_remove (daemon->urh_head, 947
946 daemon->urh_tail, 948 if (MHD_INVALID_SOCKET != urh->mhd.socket)
947 urh); 949 MHD_socket_close_chk_ (urh->mhd.socket);
948#if EPOLL_SUPPORT 950
949 if (0 != (daemon->options & MHD_USE_EPOLL)) 951 if (MHD_INVALID_SOCKET != urh->app.socket)
950 { 952 MHD_socket_close_chk_ (urh->app.socket);
951 /* epoll documentation suggests that closing a FD
952 automatically removes it from the epoll set; however,
953 this is not true as if we fail to do manually remove it,
954 we are still seeing an event for this fd in epoll,
955 causing grief (use-after-free...) --- at least on my
956 system. */
957 if (0 != epoll_ctl (daemon->epoll_upgrade_fd,
958 EPOLL_CTL_DEL,
959 connection->socket_fd,
960 NULL))
961 MHD_PANIC (_("Failed to remove FD from epoll set\n"));
962 }
963#endif /* EPOLL_SUPPORT */
964 if (MHD_INVALID_SOCKET != urh->mhd.socket)
965 {
966 /* epoll documentation suggests that closing a FD
967 automatically removes it from the epoll set; however,
968 this is not true as if we fail to do manually remove it,
969 we are still seeing an event for this fd in epoll,
970 causing grief (use-after-free...) --- at least on my
971 system. */
972#if EPOLL_SUPPORT
973 if ( (0 != (daemon->options & MHD_USE_EPOLL)) &&
974 (0 != epoll_ctl (daemon->epoll_upgrade_fd,
975 EPOLL_CTL_DEL,
976 urh->mhd.socket,
977 NULL)) )
978 MHD_PANIC (_("Failed to remove FD from epoll set\n"));
979#endif /* EPOLL_SUPPORT */
980 MHD_socket_close_chk_ (urh->mhd.socket);
981 }
982 if (MHD_INVALID_SOCKET != urh->app.socket)
983 MHD_socket_close_chk_ (urh->app.socket);
984 }
985#endif /* HTTPS_SUPPORT */
986 if (0 == (daemon->options & MHD_USE_THREAD_PER_CONNECTION))
987 {
988 /* resuming the connection will indirectly close it */
989 MHD_resume_connection (connection);
990#if HTTPS_SUPPORT
991 if (0 != (daemon->options & MHD_USE_TLS))
992 {
993 gnutls_bye (connection->tls_session,
994 GNUTLS_SHUT_RDWR);
995 }
996#endif
997 }
998 else
999 {
1000 /* the thread would no longer close it */
1001 MHD_connection_close_ (connection,
1002 MHD_REQUEST_TERMINATED_COMPLETED_OK);
1003 }
1004 953
1005 connection->urh = NULL; 954 connection->urh = NULL;
1006 if (NULL != urh) 955 if (NULL != urh)
@@ -1259,14 +1208,6 @@ process_urh (struct MHD_UpgradeResponseHandle *urh)
1259 urh->app.celi &= ~MHD_EPOLL_STATE_WRITE_READY; 1208 urh->app.celi &= ~MHD_EPOLL_STATE_WRITE_READY;
1260 urh->mhd.celi &= ~MHD_EPOLL_STATE_READ_READY; 1209 urh->mhd.celi &= ~MHD_EPOLL_STATE_READ_READY;
1261 } 1210 }
1262
1263 /* cleanup connection if it was closed and all data was sent */
1264 if ( (MHD_NO != urh->was_closed) &&
1265 (0 == urh->out_buffer_size) &&
1266 (0 == urh->out_buffer_used) )
1267 {
1268 MHD_cleanup_upgraded_connection_ (urh->connection);
1269 }
1270} 1211}
1271#endif 1212#endif
1272 1213
@@ -1292,7 +1233,8 @@ thread_main_connection_upgrade (struct MHD_Connection *con)
1292 if ( (0 != (daemon->options & MHD_USE_TLS)) && 1233 if ( (0 != (daemon->options & MHD_USE_TLS)) &&
1293 (0 == (daemon->options & MHD_USE_POLL))) 1234 (0 == (daemon->options & MHD_USE_POLL)))
1294 { 1235 {
1295 while (MHD_CONNECTION_UPGRADE == con->state) 1236 while ( (MHD_CONNECTION_UPGRADE == con->state) ||
1237 (0 != urh->out_buffer_used) )
1296 { 1238 {
1297 /* use select */ 1239 /* use select */
1298 fd_set rs; 1240 fd_set rs;
@@ -1356,7 +1298,8 @@ thread_main_connection_upgrade (struct MHD_Connection *con)
1356 /* use poll() */ 1298 /* use poll() */
1357 const unsigned int timeout = UINT_MAX; 1299 const unsigned int timeout = UINT_MAX;
1358 1300
1359 while (MHD_CONNECTION_UPGRADE == con->state) 1301 while ( (MHD_CONNECTION_UPGRADE == con->state) ||
1302 (0 != urh->out_buffer_used) )
1360 { 1303 {
1361 struct pollfd p[2]; 1304 struct pollfd p[2];
1362 1305
@@ -1410,6 +1353,10 @@ thread_main_connection_upgrade (struct MHD_Connection *con)
1410#endif 1353#endif
1411 /* end HTTPS */ 1354 /* end HTTPS */
1412#endif 1355#endif
1356 /* TLS forwarding was finished. Cleanup socketpair. */
1357 MHD_connection_finish_forward_ (con);
1358 /* Do not set 'urh->clean_ready' yet as 'urh' will be used
1359 * in connection thread for a little while. */
1413} 1360}
1414 1361
1415 1362
@@ -1456,7 +1403,7 @@ thread_main_handle_connection (void *data)
1456 const unsigned int timeout = daemon->connection_timeout; 1403 const unsigned int timeout = daemon->connection_timeout;
1457 _MHD_bool was_suspended = 0; 1404 _MHD_bool was_suspended = 0;
1458 1405
1459 if (MHD_NO != con->suspended) 1406 if (MHD_NO != con->suspended && NULL == con->urh)
1460 { 1407 {
1461 /* Connection was suspended, wait for resume. */ 1408 /* Connection was suspended, wait for resume. */
1462 was_suspended = !0; 1409 was_suspended = !0;
@@ -1721,7 +1668,11 @@ thread_main_handle_connection (void *data)
1721 goto exit; 1668 goto exit;
1722 } 1669 }
1723#endif 1670#endif
1724 if (MHD_CONNECTION_UPGRADE == con->state) 1671 /* Check for 'MHD_CONNECTION_UPGRADE_CLOSED' too:
1672 * application can finish with "upgraded" connection
1673 * before this thread process it for the first time. */
1674 if ( (MHD_CONNECTION_UPGRADE == con->state) ||
1675 (MHD_CONNECTION_UPGRADE_CLOSED == con->state) )
1725 { 1676 {
1726 /* Normal HTTP processing is finished, 1677 /* Normal HTTP processing is finished,
1727 * notify application. */ 1678 * notify application. */
@@ -1734,13 +1685,17 @@ thread_main_handle_connection (void *data)
1734 con->client_aware = MHD_NO; 1685 con->client_aware = MHD_NO;
1735 1686
1736 thread_main_connection_upgrade (con); 1687 thread_main_connection_upgrade (con);
1737#if HTTPS_SUPPORT 1688 /* MHD_connection_finish_forward_() was called by thread_main_connection_upgrade(). */
1738 if (0 != (daemon->options & MHD_USE_TLS) ) 1689
1739 break; 1690 /* "Upgraded" data will not be used in this thread from this point. */
1740#endif 1691 con->urh->clean_ready = MHD_YES;
1692 /* If 'urh->was_closed' set to MHD_YES, connection will be
1693 * moved immediately to cleanup list. Otherwise connection
1694 * will stay in suspended list until 'urh' will be marked
1695 * with 'was_closed' by application. */
1696 MHD_resume_connection(con);
1741 1697
1742 /* skip usual clean up EXCEPT for the completion 1698 /* skip usual clean up */
1743 notification (which must be done in this thread)! */
1744 return (MHD_THRD_RTRN_TYPE_) 0; 1699 return (MHD_THRD_RTRN_TYPE_) 0;
1745 } 1700 }
1746 } 1701 }
@@ -2459,6 +2414,8 @@ MHD_resume_connection (struct MHD_Connection *connection)
2459/** 2414/**
2460 * Run through the suspended connections and move any that are no 2415 * Run through the suspended connections and move any that are no
2461 * longer suspended back to the active state. 2416 * longer suspended back to the active state.
2417 * @remark To be called only from thread that process
2418 * daemon's select()/poll()/etc.
2462 * 2419 *
2463 * @param daemon daemon context 2420 * @param daemon daemon context
2464 * @return #MHD_YES if a connection was actually resumed 2421 * @return #MHD_YES if a connection was actually resumed
@@ -2490,40 +2447,55 @@ resume_suspended_connections (struct MHD_Daemon *daemon)
2490 while (NULL != (pos = next)) 2447 while (NULL != (pos = next))
2491 { 2448 {
2492 next = pos->next; 2449 next = pos->next;
2493 if (MHD_NO == pos->resuming) 2450 if ( (MHD_NO == pos->resuming) ||
2451 ((NULL != pos->urh) &&
2452 ((MHD_NO == pos->urh->was_closed) || (MHD_NO == pos->urh->clean_ready))) )
2494 continue; 2453 continue;
2495 ret = MHD_YES; 2454 ret = MHD_YES;
2496 DLL_remove (daemon->suspended_connections_head, 2455 DLL_remove (daemon->suspended_connections_head,
2497 daemon->suspended_connections_tail, 2456 daemon->suspended_connections_tail,
2498 pos); 2457 pos);
2499 DLL_insert (daemon->connections_head, 2458 if (NULL == pos->urh)
2500 daemon->connections_tail,
2501 pos);
2502 if (0 == (daemon->options & MHD_USE_THREAD_PER_CONNECTION))
2503 { 2459 {
2504 if (pos->connection_timeout == daemon->connection_timeout) 2460 DLL_insert (daemon->connections_head,
2505 XDLL_insert (daemon->normal_timeout_head, 2461 daemon->connections_tail,
2506 daemon->normal_timeout_tail, 2462 pos);
2507 pos); 2463 if (0 == (daemon->options & MHD_USE_THREAD_PER_CONNECTION))
2508 else 2464 {
2509 XDLL_insert (daemon->manual_timeout_head, 2465 if (pos->connection_timeout == daemon->connection_timeout)
2510 daemon->manual_timeout_tail, 2466 XDLL_insert (daemon->normal_timeout_head,
2511 pos); 2467 daemon->normal_timeout_tail,
2512 } 2468 pos);
2469 else
2470 XDLL_insert (daemon->manual_timeout_head,
2471 daemon->manual_timeout_tail,
2472 pos);
2473 }
2513#ifdef EPOLL_SUPPORT 2474#ifdef EPOLL_SUPPORT
2514 if (0 != (daemon->options & MHD_USE_EPOLL)) 2475 if (0 != (daemon->options & MHD_USE_EPOLL))
2476 {
2477 if (0 != (pos->epoll_state & MHD_EPOLL_STATE_IN_EREADY_EDLL))
2478 MHD_PANIC ("Resumed connection was already in EREADY set\n");
2479 /* we always mark resumed connections as ready, as we
2480 might have missed the edge poll event during suspension */
2481 EDLL_insert (daemon->eready_head,
2482 daemon->eready_tail,
2483 pos);
2484 pos->epoll_state |= MHD_EPOLL_STATE_IN_EREADY_EDLL;
2485 pos->epoll_state &= ~MHD_EPOLL_STATE_SUSPENDED;
2486 }
2487#endif
2488 }
2489 else
2515 { 2490 {
2516 if (0 != (pos->epoll_state & MHD_EPOLL_STATE_IN_EREADY_EDLL)) 2491 /* Data forwarding was finished (for TLS connections) AND
2517 MHD_PANIC ("Resumed connection was already in EREADY set\n"); 2492 * application was closed upgraded connection.
2518 /* we always mark resumed connections as ready, as we 2493 * Insert connection into cleanup list. */
2519 might have missed the edge poll event during suspension */ 2494 DLL_insert (daemon->cleanup_head,
2520 EDLL_insert (daemon->eready_head, 2495 daemon->cleanup_tail,
2521 daemon->eready_tail, 2496 pos);
2522 pos); 2497
2523 pos->epoll_state |= MHD_EPOLL_STATE_IN_EREADY_EDLL;
2524 pos->epoll_state &= ~MHD_EPOLL_STATE_SUSPENDED;
2525 } 2498 }
2526#endif
2527 pos->suspended = MHD_NO; 2499 pos->suspended = MHD_NO;
2528 pos->resuming = MHD_NO; 2500 pos->resuming = MHD_NO;
2529 } 2501 }
@@ -2735,6 +2707,8 @@ MHD_accept_connection (struct MHD_Daemon *daemon)
2735 * Free resources associated with all closed connections. 2707 * Free resources associated with all closed connections.
2736 * (destroy responses, free buffers, etc.). All closed 2708 * (destroy responses, free buffers, etc.). All closed
2737 * connections are kept in the "cleanup" doubly-linked list. 2709 * connections are kept in the "cleanup" doubly-linked list.
2710 * @remark To be called only from thread that
2711 * process daemon's select()/poll()/etc.
2738 * 2712 *
2739 * @param daemon daemon to clean up 2713 * @param daemon daemon to clean up
2740 */ 2714 */
@@ -2758,6 +2732,8 @@ MHD_cleanup_connections (struct MHD_Daemon *daemon)
2758 (MHD_NO == pos->thread_joined) && 2732 (MHD_NO == pos->thread_joined) &&
2759 (! MHD_join_thread_ (pos->pid)) ) 2733 (! MHD_join_thread_ (pos->pid)) )
2760 MHD_PANIC (_("Failed to join a thread\n")); 2734 MHD_PANIC (_("Failed to join a thread\n"));
2735 if (NULL != pos->urh)
2736 MHD_cleanup_upgraded_connection_ (pos);
2761 MHD_pool_destroy (pos->pool); 2737 MHD_pool_destroy (pos->pool);
2762#if HTTPS_SUPPORT 2738#if HTTPS_SUPPORT
2763 if (NULL != pos->tls_session) 2739 if (NULL != pos->tls_session)
@@ -3014,6 +2990,17 @@ MHD_run_from_select (struct MHD_Daemon *daemon,
3014 write_fd_set); 2990 write_fd_set);
3015 /* call generic forwarding function for passing data */ 2991 /* call generic forwarding function for passing data */
3016 process_urh (urh); 2992 process_urh (urh);
2993 /* Finished forwarding? */
2994 if ( (0 == urh->in_buffer_size) &&
2995 (0 == urh->out_buffer_size) &&
2996 (0 == urh->in_buffer_used) &&
2997 (0 == urh->out_buffer_used) )
2998 {
2999 MHD_connection_finish_forward_ (urh->connection);
3000 urh->clean_ready = MHD_YES;
3001 /* Resuming will move connection to cleanup list. */
3002 MHD_resume_connection(urh->connection);
3003 }
3017 } 3004 }
3018#endif 3005#endif
3019 MHD_cleanup_connections (daemon); 3006 MHD_cleanup_connections (daemon);
@@ -3367,28 +3354,45 @@ MHD_poll_all (struct MHD_Daemon *daemon,
3367 { 3354 {
3368 if (i >= num_connections) 3355 if (i >= num_connections)
3369 break; /* connection list changed somehow, retry later ... */ 3356 break; /* connection list changed somehow, retry later ... */
3357
3358 /* Get next connection here as connection can be removed
3359 * from 'daemon->urh_head' list. */
3370 urhn = urh->next; 3360 urhn = urh->next;
3371 if (p[poll_server+i].fd != urh->connection->socket_fd) 3361 /* Check for fd mismatch. FIXME: required for safety? */
3372 continue; /* fd mismatch, something else happened, retry later ... */ 3362 if (p[poll_server+i].fd == urh->connection->socket_fd)
3373 if (0 != (p[poll_server+i].revents & POLLIN)) 3363 {
3374 urh->app.celi |= MHD_EPOLL_STATE_READ_READY; 3364 if (0 != (p[poll_server+i].revents & POLLIN))
3375 if (0 != (p[poll_server+i].revents & POLLOUT)) 3365 urh->app.celi |= MHD_EPOLL_STATE_READ_READY;
3376 urh->app.celi |= MHD_EPOLL_STATE_WRITE_READY; 3366 if (0 != (p[poll_server+i].revents & POLLOUT))
3367 urh->app.celi |= MHD_EPOLL_STATE_WRITE_READY;
3368 }
3377 i++; 3369 i++;
3378 if (p[poll_server+i].fd != urh->mhd.socket) 3370 /* Check for fd mismatch. FIXME: required for safety? */
3371 if (p[poll_server+i].fd == urh->mhd.socket)
3379 { 3372 {
3380 /* fd mismatch, something else happened, retry later ... */ 3373 if (0 != (p[poll_server+i].revents & POLLIN))
3381 /* may still be able to do something based on updates 3374 urh->mhd.celi |= MHD_EPOLL_STATE_READ_READY;
3382 to socket_fd availability */ 3375 if (0 != (p[poll_server+i].revents & POLLOUT))
3383 process_urh (urh); 3376 urh->mhd.celi |= MHD_EPOLL_STATE_WRITE_READY;
3384 continue;
3385 } 3377 }
3386 if (0 != (p[poll_server+i].revents & POLLIN))
3387 urh->mhd.celi |= MHD_EPOLL_STATE_READ_READY;
3388 if (0 != (p[poll_server+i].revents & POLLOUT))
3389 urh->mhd.celi |= MHD_EPOLL_STATE_WRITE_READY;
3390 i++; 3378 i++;
3391 process_urh (urh); 3379 process_urh (urh);
3380 /* Finished forwarding? */
3381 if ( (0 == urh->in_buffer_size) &&
3382 (0 == urh->out_buffer_size) &&
3383 (0 == urh->in_buffer_used) &&
3384 (0 == urh->out_buffer_used) )
3385 {
3386 /* MHD_connection_finish_forward_() will remove connection from
3387 * 'daemon->urh_head' list. */
3388 MHD_connection_finish_forward_ (urh->connection);
3389 urh->clean_ready = MHD_YES;
3390 /* If 'urh->was_closed' set to MHD_YES, connection will be
3391 * moved immediately to cleanup list. Otherwise connection
3392 * will stay in suspended list until 'urh' will be marked
3393 * with 'was_closed' by application. */
3394 MHD_resume_connection(urh->connection);
3395 }
3392 } 3396 }
3393#endif 3397#endif
3394 /* handle 'listen' FD */ 3398 /* handle 'listen' FD */
@@ -3530,12 +3534,11 @@ run_epoll_for_upgrade (struct MHD_Daemon *daemon)
3530{ 3534{
3531 struct epoll_event events[MAX_EVENTS]; 3535 struct epoll_event events[MAX_EVENTS];
3532 int num_events; 3536 int num_events;
3533 unsigned int i;
3534 unsigned int j;
3535 3537
3536 num_events = MAX_EVENTS; 3538 num_events = MAX_EVENTS;
3537 while (MAX_EVENTS == num_events) 3539 while (MAX_EVENTS == num_events)
3538 { 3540 {
3541 unsigned int i;
3539 /* update event masks */ 3542 /* update event masks */
3540 num_events = epoll_wait (daemon->epoll_upgrade_fd, 3543 num_events = epoll_wait (daemon->epoll_upgrade_fd,
3541 events, 3544 events,
@@ -3555,35 +3558,8 @@ run_epoll_for_upgrade (struct MHD_Daemon *daemon)
3555 } 3558 }
3556 for (i=0;i<(unsigned int) num_events;i++) 3559 for (i=0;i<(unsigned int) num_events;i++)
3557 { 3560 {
3558 struct UpgradeEpollHandle *ueh = events[i].data.ptr; 3561 struct UpgradeEpollHandle * const ueh = events[i].data.ptr;
3559 struct MHD_UpgradeResponseHandle *urh; 3562 struct MHD_UpgradeResponseHandle * const urh = ueh->urh;
3560
3561 if (NULL == ueh)
3562 continue; /* was killed, see below */
3563 urh = ueh->urh;
3564
3565 /* In case we get two events for the same upgrade handle,
3566 squash them together (otherwise the first one may
3567 cause us to free the 'urh', and the second one then
3568 causes a use-after-free). */
3569 for (j=i+1;j< (unsigned int) num_events;j++)
3570 {
3571 struct UpgradeEpollHandle *uehj = events[j].data.ptr;
3572 struct MHD_UpgradeResponseHandle *urhj;
3573
3574 if (NULL == uehj)
3575 continue; /* was killed, see below */
3576 urhj = uehj->urh;
3577
3578 if (urh == urhj) /* yep, indeed the same! */
3579 {
3580 if (0 != (events[j].events & EPOLLIN))
3581 uehj->celi |= MHD_EPOLL_STATE_READ_READY;
3582 if (0 != (events[j].events & EPOLLOUT))
3583 uehj->celi |= MHD_EPOLL_STATE_WRITE_READY;
3584 }
3585 events[j].data.ptr = NULL; /* kill this one */
3586 }
3587 3563
3588 /* Update our state based on what is ready according to epoll() */ 3564 /* Update our state based on what is ready according to epoll() */
3589 if (0 != (events[i].events & EPOLLIN)) 3565 if (0 != (events[i].events & EPOLLIN))
@@ -3591,8 +3567,21 @@ run_epoll_for_upgrade (struct MHD_Daemon *daemon)
3591 if (0 != (events[i].events & EPOLLOUT)) 3567 if (0 != (events[i].events & EPOLLOUT))
3592 ueh->celi |= MHD_EPOLL_STATE_WRITE_READY; 3568 ueh->celi |= MHD_EPOLL_STATE_WRITE_READY;
3593 3569
3594 /* shuffle data based on buffers and FD readyness */
3595 process_urh (urh); 3570 process_urh (urh);
3571 /* Finished forwarding? */
3572 if ( (0 == urh->in_buffer_size) &&
3573 (0 == urh->out_buffer_size) &&
3574 (0 == urh->in_buffer_used) &&
3575 (0 == urh->out_buffer_used) )
3576 {
3577 MHD_connection_finish_forward_ (urh->connection);
3578 urh->clean_ready = MHD_YES;
3579 /* If 'urh->was_closed' set to MHD_YES, connection will be
3580 * moved immediately to cleanup list. Otherwise connection
3581 * will stay in suspended list until 'urh' will be marked
3582 * with 'was_closed' by application. */
3583 MHD_resume_connection(urh->connection);
3584 }
3596 } 3585 }
3597 } 3586 }
3598 return MHD_YES; 3587 return MHD_YES;
@@ -5403,18 +5392,23 @@ close_all_connections (struct MHD_Daemon *daemon)
5403#ifdef HTTPS_SUPPORT 5392#ifdef HTTPS_SUPPORT
5404 struct MHD_UpgradeResponseHandle *urh; 5393 struct MHD_UpgradeResponseHandle *urh;
5405 struct MHD_UpgradeResponseHandle *urhn; 5394 struct MHD_UpgradeResponseHandle *urhn;
5395 const _MHD_bool used_tls = (0 != (daemon->options & MHD_USE_TLS));
5396 const _MHD_bool used_thr_p_c = (0 != (daemon->options & MHD_USE_THREAD_PER_CONNECTION));
5406#endif /* HTTPS_SUPPORT */ 5397#endif /* HTTPS_SUPPORT */
5407 5398
5399#ifdef HTTPS_SUPPORT
5408 /* give upgraded HTTPS connections a chance to finish */ 5400 /* give upgraded HTTPS connections a chance to finish */
5409#if HTTPS_SUPPORT 5401 /* 'daemon->urh_head' is not used in thread-per-connection mode. */
5410 for (urh = daemon->urh_head; NULL != urh; urh = urhn) 5402 for (urh = daemon->urh_head; NULL != urh; urh = urhn)
5411 { 5403 {
5412 urhn = urh->next; 5404 urhn = urh->next;
5413 /* call generic forwarding function for passing data 5405 /* call generic forwarding function for passing data
5414 with chance to detect that application is done; 5406 with chance to detect that application is done. */
5415 fake read readyness just to be sure. */
5416 urh->mhd.celi |= MHD_EPOLL_STATE_READ_READY;
5417 process_urh (urh); 5407 process_urh (urh);
5408 MHD_connection_finish_forward_ (urh->connection);
5409 urh->clean_ready = MHD_YES;
5410 /* Resuming will move connection to cleanup list. */
5411 MHD_resume_connection(urh->connection);
5418 } 5412 }
5419#endif 5413#endif
5420 5414
@@ -5431,6 +5425,23 @@ close_all_connections (struct MHD_Daemon *daemon)
5431 traverse DLLs in peace... */ 5425 traverse DLLs in peace... */
5432 if (0 != (daemon->options & MHD_USE_THREAD_PER_CONNECTION)) 5426 if (0 != (daemon->options & MHD_USE_THREAD_PER_CONNECTION))
5433 MHD_mutex_lock_chk_ (&daemon->cleanup_connection_mutex); 5427 MHD_mutex_lock_chk_ (&daemon->cleanup_connection_mutex);
5428#ifdef HTTPS_SUPPORT
5429 if (used_tls && used_thr_p_c)
5430 {
5431 struct MHD_Connection * susp;
5432
5433 susp = daemon->suspended_connections_head;
5434 while (NULL != susp)
5435 {
5436 if (NULL == susp->urh) /* "Upgraded" connection? */
5437 MHD_PANIC (_("MHD_stop_daemon() called while we have suspended connections.\n"));
5438 else if (MHD_NO == susp->urh->clean_ready)
5439 shutdown (urh->app.socket, SHUT_RDWR); /* Wake thread by shutdown of app socket. */
5440 susp = susp->next;
5441 }
5442 }
5443 else /* This 'else' is combined with next 'if'. */
5444#endif /* HTTPS_SUPPORT */
5434 if (NULL != daemon->suspended_connections_head) 5445 if (NULL != daemon->suspended_connections_head)
5435 MHD_PANIC (_("MHD_stop_daemon() called while we have suspended connections.\n")); 5446 MHD_PANIC (_("MHD_stop_daemon() called while we have suspended connections.\n"));
5436 for (pos = daemon->connections_head; NULL != pos; pos = pos->next) 5447 for (pos = daemon->connections_head; NULL != pos; pos = pos->next)
@@ -5465,6 +5476,17 @@ close_all_connections (struct MHD_Daemon *daemon)
5465 pos = pos->next; 5476 pos = pos->next;
5466 } 5477 }
5467 } 5478 }
5479
5480#ifdef HTTPS_SUPPORT
5481 /* Finished threads with "upgraded" connections need to be moved
5482 * to cleanup list by resume_suspended_connections(). */
5483 if (used_tls && used_thr_p_c)
5484 {
5485 daemon->resuming = MHD_YES; /* Force check for pending resume. */
5486 resume_suspended_connections (daemon);
5487 }
5488#endif /* HTTPS_SUPPORT */
5489
5468 /* now that we're alone, move everyone to cleanup */ 5490 /* now that we're alone, move everyone to cleanup */
5469 while (NULL != (pos = daemon->connections_head)) 5491 while (NULL != (pos = daemon->connections_head))
5470 { 5492 {