From 6218b5f7cba0e91ae863fc79b192c5261492d198 Mon Sep 17 00:00:00 2001 From: Daniel Black Date: Wed, 22 Nov 2023 18:40:53 +1100 Subject: [PATCH] MDEV-32567 Remove thr_alarm from server codebase - socket activation fix Systemd socket activation cannot handle a shutdown on the file descriptor[1]. Enumerate past the socket activation descriptors. If there was no shutdown to trigger the breaking of the event loop, then write to the termination_event_fd that was setup during the socket activation code for this purpose. As abort_loop= true is already set at the top of break_connect_loop, and this is checked in loop before sockets are processed, no additional checking to read from the termination_event_fd is needed. Without socket activation defined, or used, termination_event_fd keeps its -1 default value. Close the eventfd outside the while loop so retries can happen if the write fails for some reason. ref[1]: https://www.freedesktop.org/software/systemd/man/latest/sd_listen_fds.html Reviewed by: Vladislav Vaintroub --- include/my_service_manager.h | 1 + sql/mysqld.cc | 45 +++++++++++++++++++++++++++++++++--- 2 files changed, 43 insertions(+), 3 deletions(-) diff --git a/include/my_service_manager.h b/include/my_service_manager.h index 498fc762048..5b797718f83 100644 --- a/include/my_service_manager.h +++ b/include/my_service_manager.h @@ -37,6 +37,7 @@ #endif #else +#define sd_listen_fds(FD) (0) #define sd_listen_fds_with_names(FD, NAMES) (0) #define sd_is_socket_unix(FD, TYPE, LISTENING, PATH, SIZE) (0) #define sd_is_socket_inet(FD, FAMILY, TYPE, LISTENING, PORT) (0) diff --git a/sql/mysqld.cc b/sql/mysqld.cc index 2552334d71f..fb897d0326e 100644 --- a/sql/mysqld.cc +++ b/sql/mysqld.cc @@ -133,6 +133,9 @@ #endif #include +#ifdef __linux__ +#include +#endif #include @@ -1386,6 +1389,7 @@ struct my_rnd_struct sql_rand; ///< used by sql_class.cc:THD::THD() Dynamic_array listen_sockets(PSI_INSTRUMENT_MEM, 0); bool unix_sock_is_online= false; static int systemd_sock_activation; /* systemd socket activation */ +static int termination_event_fd= -1; C_MODE_START @@ -1669,10 +1673,28 @@ static void break_connect_loop() int UNINIT_VAR(error); DBUG_PRINT("info",("Waiting for select thread")); - for(size_t i=0; i < listen_sockets.size(); i++) { - int fd=mysql_socket_getfd(listen_sockets.at(i)); - shutdown(fd, SHUT_RDWR); + /* + Cannot call shutdown on systemd socket activated descriptors + as their clone in the pid 1 is reused. + */ + int lowest_fd, shutdowns= 0; + lowest_fd= sd_listen_fds(0) + SD_LISTEN_FDS_START; + for(size_t i= 0; i < listen_sockets.size(); i++) + { + int fd= mysql_socket_getfd(listen_sockets.at(i)); + if (fd >= lowest_fd) + { + shutdowns++; + mysql_socket_shutdown(listen_sockets.at(i), SHUT_RDWR); + } + } + if (!shutdowns && termination_event_fd >=0) + { + uint64_t u= 1; + if (write(termination_event_fd, &u, sizeof(uint64_t)) != sizeof(uint64_t)) + sql_print_error("Couldn't send event to terminate listen loop"); + } } set_timespec(abstime, 2); for (uint tmp=0 ; tmp < 10 && select_thread_in_use; tmp++) @@ -1687,6 +1709,8 @@ static void break_connect_loop() sql_print_error("Got error %d from mysql_cond_timedwait", error); #endif } + if (termination_event_fd >= 0) + close(termination_event_fd); mysql_mutex_unlock(&LOCK_start_thread); #endif /* _WIN32 */ } @@ -2644,6 +2668,9 @@ static void use_systemd_activated_sockets() listen_sockets.push(sock); } systemd_sock_activation= 1; + termination_event_fd= eventfd(0, EFD_CLOEXEC|EFD_NONBLOCK); + if (termination_event_fd == -1) + sql_print_warning("eventfd failed %d", errno); free(names); DBUG_VOID_RETURN; @@ -6263,6 +6290,18 @@ void handle_connections_sockets() set_non_blocking_if_supported(listen_sockets.at(i)); } #endif + if (termination_event_fd >= 0) + { +#ifdef HAVE_POLL + struct pollfd event_fd; + event_fd.fd= termination_event_fd; + event_fd.events= POLLIN; + fds.push(event_fd); +#else + FD_SET(termination_event_fd, &clientFDs); +#endif + /* no need to read this fd, abrt_loop is set before it gets a chance */ + } sd_notify(0, "READY=1\n" "STATUS=Taking your SQL requests now...\n");