162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci#include <signal.h>
362306a36Sopenharmony_ci#include "subcmd-util.h"
462306a36Sopenharmony_ci#include "sigchain.h"
562306a36Sopenharmony_ci
662306a36Sopenharmony_ci#define SIGCHAIN_MAX_SIGNALS 32
762306a36Sopenharmony_ci
862306a36Sopenharmony_cistruct sigchain_signal {
962306a36Sopenharmony_ci	sigchain_fun *old;
1062306a36Sopenharmony_ci	int n;
1162306a36Sopenharmony_ci	int alloc;
1262306a36Sopenharmony_ci};
1362306a36Sopenharmony_cistatic struct sigchain_signal signals[SIGCHAIN_MAX_SIGNALS];
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_cistatic void check_signum(int sig)
1662306a36Sopenharmony_ci{
1762306a36Sopenharmony_ci	if (sig < 1 || sig >= SIGCHAIN_MAX_SIGNALS)
1862306a36Sopenharmony_ci		die("BUG: signal out of range: %d", sig);
1962306a36Sopenharmony_ci}
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_cistatic int sigchain_push(int sig, sigchain_fun f)
2262306a36Sopenharmony_ci{
2362306a36Sopenharmony_ci	struct sigchain_signal *s = signals + sig;
2462306a36Sopenharmony_ci	check_signum(sig);
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_ci	ALLOC_GROW(s->old, s->n + 1, s->alloc);
2762306a36Sopenharmony_ci	s->old[s->n] = signal(sig, f);
2862306a36Sopenharmony_ci	if (s->old[s->n] == SIG_ERR)
2962306a36Sopenharmony_ci		return -1;
3062306a36Sopenharmony_ci	s->n++;
3162306a36Sopenharmony_ci	return 0;
3262306a36Sopenharmony_ci}
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ciint sigchain_pop(int sig)
3562306a36Sopenharmony_ci{
3662306a36Sopenharmony_ci	struct sigchain_signal *s = signals + sig;
3762306a36Sopenharmony_ci	check_signum(sig);
3862306a36Sopenharmony_ci	if (s->n < 1)
3962306a36Sopenharmony_ci		return 0;
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_ci	if (signal(sig, s->old[s->n - 1]) == SIG_ERR)
4262306a36Sopenharmony_ci		return -1;
4362306a36Sopenharmony_ci	s->n--;
4462306a36Sopenharmony_ci	return 0;
4562306a36Sopenharmony_ci}
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_civoid sigchain_push_common(sigchain_fun f)
4862306a36Sopenharmony_ci{
4962306a36Sopenharmony_ci	sigchain_push(SIGINT, f);
5062306a36Sopenharmony_ci	sigchain_push(SIGHUP, f);
5162306a36Sopenharmony_ci	sigchain_push(SIGTERM, f);
5262306a36Sopenharmony_ci	sigchain_push(SIGQUIT, f);
5362306a36Sopenharmony_ci	sigchain_push(SIGPIPE, f);
5462306a36Sopenharmony_ci}
55