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:
parent
5ffdff685a
commit
14be1d30ec
3 changed files with 29 additions and 16 deletions
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
31
src/tincd.c
31
src/tincd.c
|
@ -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) {
|
||||||
|
|
Loading…
Reference in a new issue