Skip to content

Commit

Permalink
flatpak-spawn: Make signal handling async-signal-safe
Browse files Browse the repository at this point in the history
signal(2) is annoying to use because the signal handler is restricted
to a limited set of async-signal-safe functions (see signal-safety(7)).
For example, the signal handler might be called while the same thread
is halfway through a call to malloc(), so it is not necessarily safe
for it to allocate memory - which is a problem, because g_idle_add()
almost certainly allocates memory.

Solving this portably would require tricky Unix code like the
implementation that's behind g_unix_signal_source_new() (which we
can't use directly here, because it is documented not to support
SIGQUIT, SIGCONT or SIGTSTP). However, Flatpak is Linux-specific, and
Linux since at least 2.6.27 (2008) implements signalfd(), which delivers
signals into a poll()-based main loop - exactly what we want here.

Resolves: #29
Signed-off-by: Simon McVittie <[email protected]>
  • Loading branch information
smcv committed Aug 3, 2020
1 parent 4fdd2e2 commit cde4382
Show file tree
Hide file tree
Showing 3 changed files with 130 additions and 15 deletions.
1 change: 1 addition & 0 deletions meson.build
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ conf.set('_GNU_SOURCE', 1)
config_h = configure_file(output: 'config.h', configuration: conf)

gio_unix = dependency('gio-unix-2.0')
threads = dependency('threads')

srcinc = include_directories('src')

Expand Down
142 changes: 128 additions & 14 deletions src/flatpak-spawn.c
Original file line number Diff line number Diff line change
Expand Up @@ -23,14 +23,22 @@
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/signalfd.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <glib-unix.h>
#include <gio/gio.h>
#include <gio/gunixfdlist.h>

#include "backport-autoptr.h"

/* Change to #if 1 to check backwards-compatibility code paths */
#if 0
#undef GLIB_CHECK_VERSION
#define GLIB_CHECK_VERSION(x, y, z) (0)
#endif

typedef enum {
FLATPAK_SPAWN_FLAGS_CLEAR_ENV = 1 << 0,
FLATPAK_SPAWN_FLAGS_LATEST_VERSION = 1 << 1,
Expand Down Expand Up @@ -136,14 +144,37 @@ message_handler (G_GNUC_UNUSED const gchar *log_domain,
g_printerr ("%s: %s\n", g_get_prgname (), message);
}

static gboolean
forward_signal_idle_cb (gpointer user_data)
static void
forward_signal (int sig)
{
int sig = GPOINTER_TO_INT(user_data);
g_autoptr(GVariant) reply = NULL;
gboolean to_process_group = FALSE;
g_autoptr(GError) error = NULL;

if (child_pid == 0)
{
/* We are not monitoring a child yet, so let the signal act on
* this main process instead */
if (sig == SIGTSTP || sig == SIGSTOP || sig == SIGTTIN || sig == SIGTTOU)
{
raise (SIGSTOP);
}
else if (sig != SIGCONT)
{
sigset_t mask;

sigemptyset (&mask);
sigaddset (&mask, sig);
/* Unblock it, so that it will be delivered properly this time.
* Use pthread_sigmask instead of sigprocmask because the latter
* has unspecified behaviour in a multi-threaded process. */
pthread_sigmask (SIG_UNBLOCK, &mask, NULL);
raise (sig);
}

return;
}

g_debug ("Forwarding signal: %d", sig);

/* We forward stop requests as real stop, because the default doesn't
Expand Down Expand Up @@ -174,26 +205,97 @@ forward_signal_idle_cb (gpointer user_data)
g_debug ("SIGSTOP:ing flatpak-spawn");
raise (SIGSTOP);
}

return G_SOURCE_REMOVE;
}

static void
forward_signal_handler (int sig)
static gboolean
forward_signal_handler (
#if GLIB_CHECK_VERSION (2, 36, 0)
int sfd,
#else
GIOChannel *source,
#endif
G_GNUC_UNUSED GIOCondition condition,
G_GNUC_UNUSED gpointer data)
{
g_idle_add (forward_signal_idle_cb, GINT_TO_POINTER(sig));
struct signalfd_siginfo info;
ssize_t size;

#if !GLIB_CHECK_VERSION (2, 36, 0)
int sfd;

sfd = g_io_channel_unix_get_fd (source);
g_return_val_if_fail (sfd >= 0, G_SOURCE_CONTINUE);
#endif

size = read (sfd, &info, sizeof (info));

if (size < 0)
{
if (errno != EINTR && errno != EAGAIN)
g_warning ("Unable to read struct signalfd_siginfo: %s",
g_strerror (errno));
}
else if (size != sizeof (info))
{
g_warning ("Expected struct signalfd_siginfo of size %"
G_GSIZE_FORMAT ", got %" G_GSSIZE_FORMAT,
sizeof (info), size);
}
else
{
forward_signal (info.ssi_signo);
}

return G_SOURCE_CONTINUE;
}

static void
static guint
forward_signals (void)
{
int forward[] = {
static int forward[] = {
SIGHUP, SIGINT, SIGQUIT, SIGTERM, SIGCONT, SIGTSTP, SIGUSR1, SIGUSR2
};
sigset_t mask;
guint i;
int sfd;

sigemptyset (&mask);

for (i = 0; i < G_N_ELEMENTS (forward); i++)
sigaddset (&mask, forward[i]);

sfd = signalfd (-1, &mask, SFD_NONBLOCK | SFD_CLOEXEC);

for (i = 0; i < G_N_ELEMENTS(forward); i++)
signal (forward[i], forward_signal_handler);
if (sfd < 0)
{
g_warning ("Unable to watch signals: %s", g_strerror (errno));
return 0;
}

/*
* We have to block the signals, for two reasons:
* - If we didn't, most of them would kill our process.
* Listening for a signal with a signalfd does not prevent the signal's
* default disposition from being acted on.
* - Reading from a signalfd only returns information about the signals
* that are still pending for the process. If we ignored them instead
* of blocking them, they would no longer be pending by the time the
* main loop wakes up and reads from the signalfd.
*/
pthread_sigmask (SIG_BLOCK, &mask, NULL);

#if GLIB_CHECK_VERSION (2, 36, 0)
return g_unix_fd_add (sfd, G_IO_IN, forward_signal_handler, NULL);
#else
GIOChannel *channel = g_io_channel_unix_new (sfd);
guint ret;

/* Disable text recoding, treat it as a bytestream */
g_io_channel_set_encoding (channel, NULL, NULL);
ret = g_io_add_watch (channel, G_IO_IN, forward_signal_handler, NULL);
g_io_channel_unref (channel);
return ret;
#endif
}

static void
Expand Down Expand Up @@ -473,6 +575,7 @@ main (int argc,
{ "directory", 0, 0, G_OPTION_ARG_FILENAME, &opt_directory, "Working directory in which to run the command", "DIR" },
{ NULL }
};
guint signal_source = 0;

setlocale (LC_ALL, "");

Expand Down Expand Up @@ -517,6 +620,16 @@ main (int argc,
if (verbose)
g_log_set_handler (G_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, message_handler, NULL);

/* We have to block the signals we want to forward before we start any
* other thread, and in particular the GDBus worker thread, because
* the signal mask is per-thread. We need all threads to have the same
* mask, otherwise a thread that doesn't have the mask will receive
* process-directed signals, causing the whole process to exit. */
signal_source = forward_signals ();

if (signal_source == 0)
return 1;

session_bus = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
if (session_bus == NULL)
{
Expand Down Expand Up @@ -820,13 +933,14 @@ main (int argc,

g_debug ("child_pid: %d", child_pid);

forward_signals ();

loop = g_main_loop_new (NULL, FALSE);

g_signal_connect (session_bus, "closed", G_CALLBACK (session_bus_closed_cb), loop);

g_main_loop_run (loop);

if (signal_source != 0)
g_source_remove (signal_source);

return 0;
}
2 changes: 1 addition & 1 deletion src/meson.build
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
flatpak_spawn = executable(
'flatpak-spawn',
sources: 'flatpak-spawn.c',
dependencies: [gio_unix],
dependencies: [gio_unix, threads],
c_args: ['-include', '@0@'.format(config_h)],
install: true,
)
Expand Down

0 comments on commit cde4382

Please sign in to comment.