mirror of
https://codeberg.org/noisytoot/notnotdnethack.git
synced 2025-04-04 07:30:42 +01:00
unixsound: Reap zombie child in SIGCHLD handler
So we don't end up with lots of zombie processes. Also only unregister self if nowplayingpid was actually unset (otherwise getting a SIGCHLD for a different process means that the handler won't get called when we get a SIGCHLD for the right process).
This commit is contained in:
parent
2e7b8dc55e
commit
e737a3ff26
1 changed files with 10 additions and 5 deletions
|
@ -88,14 +88,19 @@ play_usersound_blocking(const char *filename, int volume)
|
|||
}
|
||||
|
||||
static pid_t nowplayingpid = 0; /* PID of the process currently playing sound */
|
||||
static struct sigaction orig_handler;
|
||||
|
||||
static void
|
||||
usersound_sigchld_handler(int sig, siginfo_t *info, void *context)
|
||||
{
|
||||
/* If called on nowplayingpid, unset it */
|
||||
if (info->si_pid == nowplayingpid)
|
||||
/* Reap process */
|
||||
waitpid(info->si_pid, NULL, WNOHANG);
|
||||
/* If called on nowplayingpid, unregister self and unset it */
|
||||
if (sig == SIGCHLD && info->si_pid == nowplayingpid) {
|
||||
sigaction(sig, &orig_handler, NULL);
|
||||
nowplayingpid = 0;
|
||||
/* Otherwise ignore (nothing else handles SIGCHLD so this is fine) */
|
||||
}
|
||||
/* Otherwise, ignore (nothing else handles SIGCHLD so this is fine) */
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -120,10 +125,10 @@ play_usersound(const char *filename, int volume)
|
|||
kill(nowplayingpid, SIGTERM);
|
||||
/* Register signal handler */
|
||||
struct sigaction handler = {
|
||||
.sa_flags = SA_SIGINFO|SA_RESETHAND,
|
||||
.sa_flags = SA_SIGINFO,
|
||||
.sa_sigaction = usersound_sigchld_handler
|
||||
};
|
||||
sigaction(SIGCHLD, &handler, NULL);
|
||||
sigaction(SIGCHLD, &handler, &orig_handler);
|
||||
/* Fork before playing the sound so it doesn't block */
|
||||
pid_t pid = nowplayingpid = fork();
|
||||
if (pid != 0) { /* We are the parent */
|
||||
|
|
Loading…
Add table
Reference in a new issue