Fix missing comments

refs #5230
This commit is contained in:
Alexander A. Klimov 2019-07-16 11:11:10 +02:00
parent 372ecd8a72
commit 4ee9ac16b4
3 changed files with 75 additions and 1 deletions

View File

@ -443,6 +443,11 @@ void Application::RequestReopenLogs()
} }
#ifndef _WIN32 #ifndef _WIN32
/**
* Sets the PID of the Icinga umbrella process.
*
* @param pid The PID of the Icinga umbrella process.
*/
void Application::SetUmbrellaProcess(pid_t pid) void Application::SetUmbrellaProcess(pid_t pid)
{ {
m_UmbrellaProcess = pid; m_UmbrellaProcess = pid;

View File

@ -127,7 +127,7 @@ private:
static bool m_RequestReopenLogs; /**< Whether we should re-open log files. */ static bool m_RequestReopenLogs; /**< Whether we should re-open log files. */
#ifndef _WIN32 #ifndef _WIN32
static pid_t m_UmbrellaProcess; static pid_t m_UmbrellaProcess; /**< The PID of the Icinga umbrella process */
#endif /* _WIN32 */ #endif /* _WIN32 */
static int m_ArgC; /**< The number of command-line arguments. */ static int m_ArgC; /**< The number of command-line arguments. */

View File

@ -182,10 +182,20 @@ std::vector<String> DaemonCommand::GetArgumentSuggestions(const String& argument
} }
#ifndef _WIN32 #ifndef _WIN32
// The PID of the Icinga umbrella process
pid_t l_UmbrellaPid = 0; pid_t l_UmbrellaPid = 0;
// Whether the umbrella process allowed us to continue working beyond config validation
static std::atomic<bool> l_AllowedToWork (false); static std::atomic<bool> l_AllowedToWork (false);
#endif /* _WIN32 */ #endif /* _WIN32 */
/**
* Do the actual work (config loading, ...)
*
* @param configs Files to read config from
*
* @return Exit code
*/
static inline static inline
int RunWorker(const std::vector<std::string>& configs) int RunWorker(const std::vector<std::string>& configs)
{ {
@ -248,6 +258,9 @@ int RunWorker(const std::vector<std::string>& configs)
} }
#ifndef _WIN32 #ifndef _WIN32
/**
* The possible states of a seemless worker being started by StartUnixWorker().
*/
enum class UnixWorkerState : uint_fast8_t enum class UnixWorkerState : uint_fast8_t
{ {
Pending, Pending,
@ -255,6 +268,7 @@ enum class UnixWorkerState : uint_fast8_t
Failed Failed
}; };
// The signals to block temporarily in StartUnixWorker().
static const sigset_t l_UnixWorkerSignals = ([]() -> sigset_t { static const sigset_t l_UnixWorkerSignals = ([]() -> sigset_t {
sigset_t s; sigset_t s;
@ -269,32 +283,49 @@ static const sigset_t l_UnixWorkerSignals = ([]() -> sigset_t {
return s; return s;
})(); })();
// The PID of the seemless worker currently being started by StartUnixWorker()
static std::atomic<pid_t> l_CurrentlyStartingUnixWorkerPid (-1); static std::atomic<pid_t> l_CurrentlyStartingUnixWorkerPid (-1);
// The state of the seemless worker currently being started by StartUnixWorker()
static std::atomic<UnixWorkerState> l_CurrentlyStartingUnixWorkerState (UnixWorkerState::Pending); static std::atomic<UnixWorkerState> l_CurrentlyStartingUnixWorkerState (UnixWorkerState::Pending);
// The last temination signal we received
static std::atomic<int> l_TermSignal (-1); static std::atomic<int> l_TermSignal (-1);
// Whether someone requested to re-load config (and we didn't handle that request, yet)
static std::atomic<bool> l_RequestedReload (false); static std::atomic<bool> l_RequestedReload (false);
// Whether someone requested to re-open logs (and we didn't handle that request, yet)
static std::atomic<bool> l_RequestedReopenLogs (false); static std::atomic<bool> l_RequestedReopenLogs (false);
/**
* Umbrella process' signal handlers
*/
static void UmbrellaSignalHandler(int num, siginfo_t *info, void*) static void UmbrellaSignalHandler(int num, siginfo_t *info, void*)
{ {
switch (num) { switch (num) {
case SIGUSR1: case SIGUSR1:
// Someone requested to re-open logs
l_RequestedReopenLogs.store(true); l_RequestedReopenLogs.store(true);
break; break;
case SIGUSR2: case SIGUSR2:
if (l_CurrentlyStartingUnixWorkerState.load() == UnixWorkerState::Pending if (l_CurrentlyStartingUnixWorkerState.load() == UnixWorkerState::Pending
&& info->si_pid == l_CurrentlyStartingUnixWorkerPid.load()) { && info->si_pid == l_CurrentlyStartingUnixWorkerPid.load()) {
// The seemless worker currently being started by StartUnixWorker() successfully loaded its config
l_CurrentlyStartingUnixWorkerState.store(UnixWorkerState::LoadedConfig); l_CurrentlyStartingUnixWorkerState.store(UnixWorkerState::LoadedConfig);
} }
break; break;
case SIGCHLD: case SIGCHLD:
if (l_CurrentlyStartingUnixWorkerState.load() == UnixWorkerState::Pending if (l_CurrentlyStartingUnixWorkerState.load() == UnixWorkerState::Pending
&& info->si_pid == l_CurrentlyStartingUnixWorkerPid.load()) { && info->si_pid == l_CurrentlyStartingUnixWorkerPid.load()) {
// The seemless worker currently being started by StartUnixWorker() failed
l_CurrentlyStartingUnixWorkerState.store(UnixWorkerState::Failed); l_CurrentlyStartingUnixWorkerState.store(UnixWorkerState::Failed);
} }
break; break;
case SIGINT: case SIGINT:
case SIGTERM: case SIGTERM:
// Someone requested our termination
{ {
struct sigaction sa; struct sigaction sa;
memset(&sa, 0, sizeof(sa)); memset(&sa, 0, sizeof(sa));
@ -307,35 +338,47 @@ static void UmbrellaSignalHandler(int num, siginfo_t *info, void*)
l_TermSignal.store(num); l_TermSignal.store(num);
break; break;
case SIGHUP: case SIGHUP:
// Someone requested to re-load config
l_RequestedReload.store(true); l_RequestedReload.store(true);
break; break;
default: default:
// Programming error (or someone has broken the userspace)
VERIFY(!"Caught unexpected signal"); VERIFY(!"Caught unexpected signal");
} }
} }
/**
* Seemless worker's signal handlers
*/
static void WorkerSignalHandler(int num, siginfo_t *info, void*) static void WorkerSignalHandler(int num, siginfo_t *info, void*)
{ {
switch (num) { switch (num) {
case SIGUSR2: case SIGUSR2:
if (info->si_pid == l_UmbrellaPid) { if (info->si_pid == l_UmbrellaPid) {
// The umbrella process allowed us to continue working beyond config validation
l_AllowedToWork.store(true); l_AllowedToWork.store(true);
} }
break; break;
case SIGINT: case SIGINT:
case SIGTERM: case SIGTERM:
if (info->si_pid == l_UmbrellaPid) { if (info->si_pid == l_UmbrellaPid) {
// The umbrella process requested our termination
Application::RequestShutdown(); Application::RequestShutdown();
} }
break; break;
default: default:
// Programming error (or someone has broken the userspace)
VERIFY(!"Caught unexpected signal"); VERIFY(!"Caught unexpected signal");
} }
} }
#ifdef HAVE_SYSTEMD #ifdef HAVE_SYSTEMD
// When we last notified the watchdog.
static std::atomic<double> l_LastNotifiedWatchdog (0); static std::atomic<double> l_LastNotifiedWatchdog (0);
/**
* Notify the watchdog if not notified during the last 2.5s.
*/
static void NotifyWatchdog() static void NotifyWatchdog()
{ {
double now = Utility::GetTime(); double now = Utility::GetTime();
@ -347,6 +390,13 @@ static void NotifyWatchdog()
} }
#endif /* HAVE_SYSTEMD */ #endif /* HAVE_SYSTEMD */
/**
* Starts seemless worker process doing the actual work (config loading, ...)
*
* @param configs Files to read config from
*
* @return The worker's PID on success, -1 on failure (if the worker couldn't load its config)
*/
static pid_t StartUnixWorker(const std::vector<std::string>& configs) static pid_t StartUnixWorker(const std::vector<std::string>& configs)
{ {
Log(LogNotice, "cli") << "Spawning seemless worker process doing the actual work"; Log(LogNotice, "cli") << "Spawning seemless worker process doing the actual work";
@ -359,6 +409,9 @@ static pid_t StartUnixWorker(const std::vector<std::string>& configs)
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
/* Block the signal handlers we'd like to change in the child process until we changed them.
* Block SIGUSR2 and SIGCHLD handlers until we've set l_CurrentlyStartingUnixWorkerPid.
*/
(void)sigprocmask(SIG_BLOCK, &l_UnixWorkerSignals, nullptr); (void)sigprocmask(SIG_BLOCK, &l_UnixWorkerSignals, nullptr);
pid_t pid = fork(); pid_t pid = fork();
@ -415,6 +468,7 @@ static pid_t StartUnixWorker(const std::vector<std::string>& configs)
Log(LogNotice, "cli") << "Spawned worker process (PID " << pid << "), waiting for it to load its config"; Log(LogNotice, "cli") << "Spawned worker process (PID " << pid << "), waiting for it to load its config";
// Wait for the newly spawned process to either load its config or fail.
for (;;) { for (;;) {
#ifdef HAVE_SYSTEMD #ifdef HAVE_SYSTEMD
NotifyWatchdog(); NotifyWatchdog();
@ -442,6 +496,7 @@ static pid_t StartUnixWorker(const std::vector<std::string>& configs)
break; break;
} }
// Reset flags for the next time
l_CurrentlyStartingUnixWorkerPid.store(-1); l_CurrentlyStartingUnixWorkerPid.store(-1);
l_CurrentlyStartingUnixWorkerState.store(UnixWorkerState::Pending); l_CurrentlyStartingUnixWorkerState.store(UnixWorkerState::Pending);
@ -457,6 +512,9 @@ static pid_t StartUnixWorker(const std::vector<std::string>& configs)
return pid; return pid;
} }
/**
* Workaround to instantiate Application (which is abstract) in DaemonCommand#Run()
*/
class PidFileManagementApp : public Application class PidFileManagementApp : public Application
{ {
public: public:
@ -520,6 +578,10 @@ int DaemonCommand::Run(const po::variables_map& vm, const std::vector<std::strin
} }
#ifndef _WIN32 #ifndef _WIN32
/* The Application manages the PID file,
* but on *nix this process doesn't load any config
* so there's no central Application instance.
*/
PidFileManagementApp app; PidFileManagementApp app;
try { try {
@ -569,19 +631,24 @@ int DaemonCommand::Run(const po::variables_map& vm, const std::vector<std::strin
(void)sigaction(SIGHUP, &sa, nullptr); (void)sigaction(SIGHUP, &sa, nullptr);
} }
// The PID of the current seemless worker
pid_t currentWorker = StartUnixWorker(configs); pid_t currentWorker = StartUnixWorker(configs);
if (currentWorker == -1) { if (currentWorker == -1) {
return EXIT_FAILURE; return EXIT_FAILURE;
} }
// Immediately allow the first (non-reload) worker to continue working beyond config validation
(void)kill(currentWorker, SIGUSR2); (void)kill(currentWorker, SIGUSR2);
#ifdef HAVE_SYSTEMD #ifdef HAVE_SYSTEMD
sd_notify(0, "READY=1"); sd_notify(0, "READY=1");
#endif /* HAVE_SYSTEMD */ #endif /* HAVE_SYSTEMD */
// Whether we already forwarded a termination signal to the seemless worker
bool requestedTermination = false; bool requestedTermination = false;
// Whether we already notified systemd about our termination
bool notifiedTermination = false; bool notifiedTermination = false;
for (;;) { for (;;) {
@ -634,6 +701,7 @@ int DaemonCommand::Run(const po::variables_map& vm, const std::vector<std::strin
Log(LogNotice, "cli") << "Waited for " << Utility::FormatDuration(Utility::GetTime() - start) << " on old process to exit."; Log(LogNotice, "cli") << "Waited for " << Utility::FormatDuration(Utility::GetTime() - start) << " on old process to exit.";
} }
// Old instance shut down, allow the new one to continue working beyond config validation
(void)kill(nextWorker, SIGUSR2); (void)kill(nextWorker, SIGUSR2);
currentWorker = nextWorker; currentWorker = nextWorker;
@ -663,6 +731,7 @@ int DaemonCommand::Run(const po::variables_map& vm, const std::vector<std::strin
} }
#endif /* HAVE_SYSTEMD */ #endif /* HAVE_SYSTEMD */
// If killed by signal, forward it via the exit code (to be as seemless as possible)
return WIFSIGNALED(status) ? 128 + WTERMSIG(status) : WEXITSTATUS(status); return WIFSIGNALED(status) ? 128 + WTERMSIG(status) : WEXITSTATUS(status);
} }
} }