Unix: Don't send quit events during signal handler. Make note to send it, and send next time we SDL_PumpEvents(). Otherwise, we might be trying to use malloc() to push a new event on the queue while a signal is interrupting malloc() elsewhere, usually causing a crash. Fixes Bugzilla #2870.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
diff --git a/src/events/SDL_events.c b/src/events/SDL_events.c
index edfc3ed..4f5bb2d 100644
--- a/src/events/SDL_events.c
+++ b/src/events/SDL_events.c
@@ -406,6 +406,8 @@ SDL_PumpEvents(void)
SDL_JoystickUpdate();
}
#endif
+
+ SDL_SendPendingQuit(); /* in case we had a signal handler fire, etc. */
}
/* Public functions */
diff --git a/src/events/SDL_events_c.h b/src/events/SDL_events_c.h
index 0289590..ebd983d 100644
--- a/src/events/SDL_events_c.h
+++ b/src/events/SDL_events_c.h
@@ -43,6 +43,8 @@ extern int SDL_QuitInit(void);
extern int SDL_SendQuit(void);
extern void SDL_QuitQuit(void);
+extern void SDL_SendPendingQuit(void);
+
/* The event filter function */
extern SDL_EventFilter SDL_EventOK;
extern void *SDL_EventOKParam;
diff --git a/src/events/SDL_quit.c b/src/events/SDL_quit.c
index 914b5bf..d2b215b 100644
--- a/src/events/SDL_quit.c
+++ b/src/events/SDL_quit.c
@@ -20,6 +20,7 @@
*/
#include "../SDL_internal.h"
#include "SDL_hints.h"
+#include "SDL_assert.h"
/* General quit handling code for SDL */
@@ -30,8 +31,8 @@
#include "SDL_events.h"
#include "SDL_events_c.h"
-
static SDL_bool disable_signals = SDL_FALSE;
+static SDL_bool send_quit_pending = SDL_FALSE;
#ifdef HAVE_SIGNAL_H
static void
@@ -40,8 +41,9 @@ SDL_HandleSIG(int sig)
/* Reset the signal handler */
signal(sig, SDL_HandleSIG);
- /* Signal a quit interrupt */
- SDL_SendQuit();
+ /* Send a quit event next time the event loop pumps. */
+ /* We can't send it in signal handler; malloc() might be interrupted! */
+ send_quit_pending = SDL_TRUE;
}
#endif /* HAVE_SIGNAL_H */
@@ -136,7 +138,17 @@ SDL_QuitQuit(void)
int
SDL_SendQuit(void)
{
+ send_quit_pending = SDL_FALSE;
return SDL_SendAppEvent(SDL_QUIT);
}
+void
+SDL_SendPendingQuit(void)
+{
+ if (send_quit_pending) {
+ SDL_SendQuit();
+ SDL_assert(!send_quit_pending);
+ }
+}
+
/* vi: set ts=4 sw=4 expandtab: */