Win32Main.cxx: more clean shutdown

This change fixes two issues:

1) console_handler is called from a separate thread.
   Thread-safe primitive is required for correct operation.

2) If console_handler returns TRUE our process is immediately terminated.
   We use Sleep() to give main thread an opportunity to shutdown correctly.
This commit is contained in:
Denis Krjuchkov 2013-01-13 15:20:32 +06:00
parent eef4f33a29
commit abb0fcb203

View File

@ -25,6 +25,9 @@
#include "mpd_error.h" #include "mpd_error.h"
#include "GlobalEvents.hxx" #include "GlobalEvents.hxx"
#include <cstdlib>
#include <atomic>
#include <glib.h> #include <glib.h>
#include <windows.h> #include <windows.h>
@ -32,7 +35,7 @@
static int service_argc; static int service_argc;
static char **service_argv; static char **service_argv;
static char service_name[] = ""; static char service_name[] = "";
static BOOL ignore_console_events; static std::atomic_bool running;
static SERVICE_STATUS_HANDLE service_handle; static SERVICE_STATUS_HANDLE service_handle;
static void WINAPI static void WINAPI
@ -103,8 +106,22 @@ console_handler(DWORD event)
switch (event) { switch (event) {
case CTRL_C_EVENT: case CTRL_C_EVENT:
case CTRL_CLOSE_EVENT: case CTRL_CLOSE_EVENT:
if (!ignore_console_events) if (running.load()) {
// Recent msdn docs that process is terminated
// if this function returns TRUE.
// We initiate correct shutdown sequence (if possible).
// Once main() returns CRT will terminate our process
// regardless our thread is still active.
// If this did not happen within 3 seconds
// let's shutdown anyway.
GlobalEvents::Emit(GlobalEvents::SHUTDOWN); GlobalEvents::Emit(GlobalEvents::SHUTDOWN);
// Under debugger it's better to wait indefinitely
// to allow debugging of shutdown code.
Sleep(IsDebuggerPresent() ? INFINITE : 3000);
}
// If we're not running main loop there is no chance for
// clean shutdown.
std::exit(EXIT_FAILURE);
return TRUE; return TRUE;
default: default:
return FALSE; return FALSE;
@ -125,8 +142,8 @@ int win32_main(int argc, char *argv[])
error_code = GetLastError(); error_code = GetLastError();
if (error_code == ERROR_FAILED_SERVICE_CONTROLLER_CONNECT) { if (error_code == ERROR_FAILED_SERVICE_CONTROLLER_CONNECT) {
/* running as console app */ /* running as console app */
running.store(false);
SetConsoleTitle("Music Player Daemon"); SetConsoleTitle("Music Player Daemon");
ignore_console_events = TRUE;
SetConsoleCtrlHandler(console_handler, TRUE); SetConsoleCtrlHandler(console_handler, TRUE);
return mpd_main(argc, argv); return mpd_main(argc, argv);
} }
@ -140,7 +157,7 @@ void win32_app_started()
if (service_handle != 0) if (service_handle != 0)
service_notify_status(SERVICE_RUNNING); service_notify_status(SERVICE_RUNNING);
else else
ignore_console_events = FALSE; running.store(true);
} }
void win32_app_stopping() void win32_app_stopping()
@ -148,7 +165,7 @@ void win32_app_stopping()
if (service_handle != 0) if (service_handle != 0)
service_notify_status(SERVICE_STOP_PENDING); service_notify_status(SERVICE_STOP_PENDING);
else else
ignore_console_events = TRUE; running.store(false);
} }
#endif #endif