Shutdown cleanly when receiving a Windows console shutdown request.

This commit makes tinc exit cleanly on Windows when hitting CTRL+C at
the console or when the user logs off. This change has no effect when
running tinc as a service.
This commit is contained in:
Etienne Dechamps 2014-07-12 17:47:01 +01:00
parent 5ffdff685a
commit 14be1d30ec
3 changed files with 29 additions and 16 deletions

View file

@ -109,7 +109,7 @@ static bool install_service(void) {
return true; return true;
} }
static io_t stop_io; io_t stop_io;
DWORD WINAPI controlhandler(DWORD request, DWORD type, LPVOID boe, LPVOID bah) { DWORD WINAPI controlhandler(DWORD request, DWORD type, LPVOID boe, LPVOID bah) {
switch(request) { switch(request) {
@ -135,17 +135,9 @@ DWORD WINAPI controlhandler(DWORD request, DWORD type, LPVOID boe, LPVOID bah) {
return NO_ERROR; return NO_ERROR;
} }
static void stop_handler(void *data, int flags) {
event_exit();
}
VOID WINAPI run_service(DWORD argc, LPTSTR* argv) { VOID WINAPI run_service(DWORD argc, LPTSTR* argv) {
extern int main2(int argc, char **argv); extern int main2(int argc, char **argv);
io_add_event(&stop_io, stop_handler, NULL, WSACreateEvent());
if (stop_io.event == FALSE)
abort();
status.dwServiceType = SERVICE_WIN32; status.dwServiceType = SERVICE_WIN32;
status.dwControlsAccepted = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_SHUTDOWN; status.dwControlsAccepted = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_SHUTDOWN;
status.dwWin32ExitCode = 0; status.dwWin32ExitCode = 0;
@ -172,9 +164,6 @@ VOID WINAPI run_service(DWORD argc, LPTSTR* argv) {
SetServiceStatus(statushandle, &status); SetServiceStatus(statushandle, &status);
} }
if (WSACloseEvent(stop_io.event) == FALSE)
abort();
io_del(&stop_io);
return; return;
} }

View file

@ -29,6 +29,7 @@ extern bool detach(void);
extern bool kill_other(int); extern bool kill_other(int);
#ifdef HAVE_MINGW #ifdef HAVE_MINGW
extern io_t stop_io;
extern bool init_service(void); extern bool init_service(void);
#endif #endif

View file

@ -49,6 +49,7 @@
#include "control.h" #include "control.h"
#include "crypto.h" #include "crypto.h"
#include "device.h" #include "device.h"
#include "event.h"
#include "logger.h" #include "logger.h"
#include "names.h" #include "names.h"
#include "net.h" #include "net.h"
@ -303,6 +304,17 @@ static bool drop_privs(void) {
#ifdef HAVE_MINGW #ifdef HAVE_MINGW
# define setpriority(level) !SetPriorityClass(GetCurrentProcess(), (level)) # define setpriority(level) !SetPriorityClass(GetCurrentProcess(), (level))
static void stop_handler(void *data, int flags) {
event_exit();
}
static BOOL WINAPI console_ctrl_handler(DWORD type) {
logger(DEBUG_ALWAYS, LOG_NOTICE, "Got console shutdown request");
if (WSASetEvent(stop_io.event) == FALSE)
abort();
return TRUE;
}
#else #else
# define NORMAL_PRIORITY_CLASS 0 # define NORMAL_PRIORITY_CLASS 0
# define BELOW_NORMAL_PRIORITY_CLASS 10 # define BELOW_NORMAL_PRIORITY_CLASS 10
@ -371,10 +383,21 @@ int main(int argc, char **argv) {
#endif #endif
#ifdef HAVE_MINGW #ifdef HAVE_MINGW
if(!do_detach || !init_service()) io_add_event(&stop_io, stop_handler, NULL, WSACreateEvent());
return main2(argc, argv); if (stop_io.event == FALSE)
else abort();
return 1;
int result;
if(!do_detach || !init_service()) {
SetConsoleCtrlHandler(console_ctrl_handler, TRUE);
result = main2(argc, argv);
} else
result = 1;
if (WSACloseEvent(stop_io.event) == FALSE)
abort();
io_del(&stop_io);
return result;
} }
int main2(int argc, char **argv) { int main2(int argc, char **argv) {