8d7dc098b1
setenv() is not thread-safe and must happen before starting threads. The g_setenv() calls in main() happened after meta_context_configure(), which will start a thread via meta_profiler_new() which in its init calls g_bus_get(). Closes: https://gitlab.gnome.org/GNOME/gnome-shell/-/issues/6974 Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell/-/merge_requests/3018>
743 lines
21 KiB
C
743 lines
21 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#include "config.h"
|
|
|
|
#if defined (HAVE_MALLINFO) || defined (HAVE_MALLINFO2)
|
|
#include <malloc.h>
|
|
#endif
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <cogl-pango/cogl-pango.h>
|
|
#include <clutter/clutter.h>
|
|
#include <glib-unix.h>
|
|
#include <glib/gi18n-lib.h>
|
|
#include <girepository.h>
|
|
#include <meta/meta-context.h>
|
|
#include <meta/meta-plugin.h>
|
|
#include <meta/prefs.h>
|
|
#include <atk-bridge.h>
|
|
#include <link.h>
|
|
|
|
#ifdef HAVE_EXE_INTROSPECTION
|
|
#include <dlfcn.h>
|
|
#include <elf.h>
|
|
#endif
|
|
|
|
#include "shell-global.h"
|
|
#include "shell-global-private.h"
|
|
#include "shell-perf-log.h"
|
|
#include "st.h"
|
|
|
|
extern GType gnome_shell_plugin_get_type (void);
|
|
|
|
#define SHELL_DBUS_SERVICE "org.gnome.Shell"
|
|
|
|
#define WM_NAME "GNOME Shell"
|
|
#define GNOME_WM_KEYBINDINGS "Mutter,GNOME Shell"
|
|
|
|
static gboolean is_gdm_mode = FALSE;
|
|
static char *session_mode = NULL;
|
|
static int caught_signal = 0;
|
|
static gboolean force_animations = FALSE;
|
|
static char *script_path = NULL;
|
|
|
|
#define DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER 1
|
|
#define DBUS_REQUEST_NAME_REPLY_ALREADY_OWNER 4
|
|
|
|
enum {
|
|
SHELL_DEBUG_BACKTRACE_WARNINGS = 1,
|
|
SHELL_DEBUG_BACKTRACE_SEGFAULTS = 2,
|
|
};
|
|
static int _shell_debug;
|
|
static gboolean _tracked_signals[NSIG] = { 0 };
|
|
|
|
static void
|
|
shell_dbus_acquire_name (GDBusProxy *bus,
|
|
guint32 request_name_flags,
|
|
guint32 *request_name_result,
|
|
const gchar *name,
|
|
gboolean fatal)
|
|
{
|
|
GError *error = NULL;
|
|
GVariant *request_name_variant;
|
|
|
|
if (!(request_name_variant = g_dbus_proxy_call_sync (bus,
|
|
"RequestName",
|
|
g_variant_new ("(su)", name, request_name_flags),
|
|
0, /* call flags */
|
|
-1, /* timeout */
|
|
NULL, /* cancellable */
|
|
&error)))
|
|
{
|
|
g_printerr ("failed to acquire %s: %s\n", name, error->message);
|
|
g_clear_error (&error);
|
|
if (!fatal)
|
|
return;
|
|
exit (1);
|
|
}
|
|
g_variant_get (request_name_variant, "(u)", request_name_result);
|
|
g_variant_unref (request_name_variant);
|
|
}
|
|
|
|
static void
|
|
shell_dbus_init (gboolean replace)
|
|
{
|
|
GDBusConnection *session;
|
|
GDBusProxy *bus;
|
|
GError *error = NULL;
|
|
guint32 request_name_flags;
|
|
guint32 request_name_result;
|
|
|
|
session = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
|
|
|
if (error) {
|
|
g_printerr ("Failed to connect to session bus: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
|
|
bus = g_dbus_proxy_new_sync (session,
|
|
G_DBUS_PROXY_FLAGS_NONE,
|
|
NULL, /* interface info */
|
|
"org.freedesktop.DBus",
|
|
"/org/freedesktop/DBus",
|
|
"org.freedesktop.DBus",
|
|
NULL, /* cancellable */
|
|
&error);
|
|
|
|
if (!bus)
|
|
{
|
|
g_printerr ("Failed to get a session bus proxy: %s\n", error->message);
|
|
exit (1);
|
|
}
|
|
|
|
request_name_flags = G_BUS_NAME_OWNER_FLAGS_ALLOW_REPLACEMENT;
|
|
if (replace)
|
|
request_name_flags |= G_BUS_NAME_OWNER_FLAGS_REPLACE;
|
|
|
|
shell_dbus_acquire_name (bus,
|
|
request_name_flags,
|
|
&request_name_result,
|
|
SHELL_DBUS_SERVICE, TRUE);
|
|
if (!(request_name_result == DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER
|
|
|| request_name_result == DBUS_REQUEST_NAME_REPLY_ALREADY_OWNER))
|
|
{
|
|
g_printerr (SHELL_DBUS_SERVICE " already exists on bus and --replace not specified\n");
|
|
exit (1);
|
|
}
|
|
|
|
g_object_unref (bus);
|
|
g_object_unref (session);
|
|
}
|
|
|
|
#ifdef HAVE_EXE_INTROSPECTION
|
|
static void
|
|
maybe_add_rpath_introspection_paths (void)
|
|
{
|
|
ElfW (Dyn) *dyn;
|
|
ElfW (Dyn) *rpath = NULL;
|
|
ElfW (Dyn) *runpath = NULL;
|
|
const char *strtab = NULL;
|
|
g_auto (GStrv) paths = NULL;
|
|
g_autofree char *exe_dir = NULL;
|
|
GStrv str;
|
|
Dl_info dl_info;
|
|
struct link_map *link_map = NULL;
|
|
|
|
for (dyn = _DYNAMIC; dyn->d_tag != DT_NULL; dyn++)
|
|
{
|
|
if (dyn->d_tag == DT_RPATH)
|
|
rpath = dyn;
|
|
else if (dyn->d_tag == DT_RUNPATH)
|
|
runpath = dyn;
|
|
else if (dyn->d_tag == DT_STRTAB)
|
|
strtab = (const char *) dyn->d_un.d_val;
|
|
}
|
|
|
|
if ((!rpath && !runpath) || !strtab)
|
|
return;
|
|
|
|
if (dladdr1 (_DYNAMIC, &dl_info, (void **) &link_map, RTLD_DL_LINKMAP))
|
|
{
|
|
/* Sanity check */
|
|
g_return_if_fail ((void *) _DYNAMIC == (void *) link_map->l_ld);
|
|
|
|
/* strtab should be at an offset above our load address. If it's not
|
|
* then this is a special architecture (riscv, mips) that has a
|
|
* readonly _DYNAMIC section that's not relocated. So in that case
|
|
* strtab is currently an offset rather than an address. Let's make it
|
|
* an address...
|
|
*/
|
|
if (strtab < (const char *) link_map->l_addr)
|
|
strtab += link_map->l_addr;
|
|
}
|
|
|
|
if (rpath)
|
|
paths = g_strsplit (strtab + rpath->d_un.d_val, ":", -1);
|
|
else
|
|
paths = g_strsplit (strtab + runpath->d_un.d_val, ":", -1);
|
|
|
|
if (!paths)
|
|
return;
|
|
|
|
for (str = paths; *str; str++)
|
|
{
|
|
g_autoptr (GError) error = NULL;
|
|
g_autoptr (GString) rpath_dir = NULL;
|
|
|
|
if (!strstr (*str, "$ORIGIN"))
|
|
continue;
|
|
|
|
if (!exe_dir)
|
|
{
|
|
g_autofree char *exe_path = NULL;
|
|
|
|
exe_path = g_file_read_link ("/proc/self/exe", &error);
|
|
if (!exe_path)
|
|
{
|
|
g_warning ("Failed to find directory of executable: %s",
|
|
error->message);
|
|
return;
|
|
}
|
|
|
|
exe_dir = g_path_get_dirname (exe_path);
|
|
}
|
|
|
|
rpath_dir = g_string_new (*str);
|
|
g_string_replace (rpath_dir, "$ORIGIN", exe_dir, 0);
|
|
|
|
g_debug ("Prepending RPATH directory '%s' "
|
|
"to introsepciton library search path",
|
|
rpath_dir->str);
|
|
g_irepository_prepend_search_path (rpath_dir->str);
|
|
g_irepository_prepend_library_path (rpath_dir->str);
|
|
}
|
|
}
|
|
#endif /* HAVE_EXE_INTROSPECTION */
|
|
|
|
static void
|
|
shell_introspection_init (void)
|
|
{
|
|
|
|
g_irepository_prepend_search_path (MUTTER_TYPELIB_DIR);
|
|
g_irepository_prepend_search_path (GNOME_SHELL_PKGLIBDIR);
|
|
|
|
/* We need to explicitly add the directories where the private libraries are
|
|
* installed to the GIR's library path, so that they can be found at runtime
|
|
* when linking using DT_RUNPATH (instead of DT_RPATH), which is the default
|
|
* for some linkers (e.g. gold) and in some distros (e.g. Debian).
|
|
*/
|
|
g_irepository_prepend_library_path (MUTTER_TYPELIB_DIR);
|
|
g_irepository_prepend_library_path (GNOME_SHELL_PKGLIBDIR);
|
|
|
|
#ifdef HAVE_EXE_INTROSPECTION
|
|
maybe_add_rpath_introspection_paths ();
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
shell_fonts_init (void)
|
|
{
|
|
CoglPangoFontMap *fontmap;
|
|
|
|
/* Disable text mipmapping; it causes problems on pre-GEM Intel
|
|
* drivers and we should just be rendering text at the right
|
|
* size rather than scaling it. If we do effects where we dynamically
|
|
* zoom labels, then we might want to reconsider.
|
|
*/
|
|
fontmap = COGL_PANGO_FONT_MAP (clutter_get_font_map ());
|
|
cogl_pango_font_map_set_use_mipmapping (fontmap, FALSE);
|
|
}
|
|
|
|
static void
|
|
shell_profiler_init (void)
|
|
{
|
|
ShellGlobal *global;
|
|
GjsProfiler *profiler;
|
|
GjsContext *context;
|
|
const char *enabled;
|
|
const char *fd_str;
|
|
int fd = -1;
|
|
|
|
/* Sysprof uses the "GJS_TRACE_FD=N" environment variable to connect GJS
|
|
* profiler data to the combined Sysprof capture. Since we are in control of
|
|
* the GjsContext, we need to proxy this FD across to the GJS profiler.
|
|
*/
|
|
|
|
fd_str = g_getenv ("GJS_TRACE_FD");
|
|
enabled = g_getenv ("GJS_ENABLE_PROFILER");
|
|
if (fd_str == NULL || enabled == NULL)
|
|
return;
|
|
|
|
global = shell_global_get ();
|
|
g_return_if_fail (global);
|
|
|
|
context = _shell_global_get_gjs_context (global);
|
|
g_return_if_fail (context);
|
|
|
|
profiler = gjs_context_get_profiler (context);
|
|
g_return_if_fail (profiler);
|
|
|
|
if (fd_str)
|
|
{
|
|
fd = atoi (fd_str);
|
|
|
|
if (fd > 2)
|
|
{
|
|
gjs_profiler_set_fd (profiler, fd);
|
|
gjs_profiler_start (profiler);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
shell_profiler_shutdown (void)
|
|
{
|
|
ShellGlobal *global;
|
|
GjsProfiler *profiler;
|
|
GjsContext *context;
|
|
|
|
global = shell_global_get ();
|
|
context = _shell_global_get_gjs_context (global);
|
|
profiler = gjs_context_get_profiler (context);
|
|
|
|
if (profiler)
|
|
gjs_profiler_stop (profiler);
|
|
}
|
|
|
|
static void
|
|
malloc_statistics_callback (ShellPerfLog *perf_log,
|
|
gpointer data)
|
|
{
|
|
#if defined (HAVE_MALLINFO) || defined (HAVE_MALLINFO2)
|
|
#ifdef HAVE_MALLINFO2
|
|
struct mallinfo2 info = mallinfo2 ();
|
|
#else
|
|
struct mallinfo info = mallinfo ();
|
|
#endif
|
|
|
|
shell_perf_log_update_statistic_i (perf_log,
|
|
"malloc.arenaSize",
|
|
info.arena);
|
|
shell_perf_log_update_statistic_i (perf_log,
|
|
"malloc.mmapSize",
|
|
info.hblkhd);
|
|
shell_perf_log_update_statistic_i (perf_log,
|
|
"malloc.usedSize",
|
|
info.uordblks);
|
|
#endif /* defined (HAVE_MALLINFO) || defined (HAVE_MALLINFO2) */
|
|
}
|
|
|
|
static void
|
|
shell_perf_log_init (void)
|
|
{
|
|
ShellPerfLog *perf_log = shell_perf_log_get_default ();
|
|
|
|
/* For probably historical reasons, mallinfo() defines the returned values,
|
|
* even those in bytes as int, not size_t. We're determined not to use
|
|
* more than 2G of malloc'ed memory, so are OK with that.
|
|
*/
|
|
shell_perf_log_define_statistic (perf_log,
|
|
"malloc.arenaSize",
|
|
"Amount of memory allocated by malloc() with brk(), in bytes",
|
|
"i");
|
|
shell_perf_log_define_statistic (perf_log,
|
|
"malloc.mmapSize",
|
|
"Amount of memory allocated by malloc() with mmap(), in bytes",
|
|
"i");
|
|
shell_perf_log_define_statistic (perf_log,
|
|
"malloc.usedSize",
|
|
"Amount of malloc'ed memory currently in use",
|
|
"i");
|
|
|
|
shell_perf_log_add_statistics_callback (perf_log,
|
|
malloc_statistics_callback,
|
|
NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
shell_a11y_init (void)
|
|
{
|
|
cally_accessibility_init ();
|
|
|
|
if (clutter_get_accessibility_enabled () == FALSE)
|
|
{
|
|
g_warning ("Accessibility: clutter has no accessibility enabled"
|
|
" skipping the atk-bridge load");
|
|
}
|
|
else
|
|
{
|
|
atk_bridge_adaptor_init (NULL, NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
shell_init_debug (const char *debug_env)
|
|
{
|
|
static const GDebugKey keys[] = {
|
|
{ "backtrace-warnings", SHELL_DEBUG_BACKTRACE_WARNINGS },
|
|
{ "backtrace-segfaults", SHELL_DEBUG_BACKTRACE_SEGFAULTS },
|
|
};
|
|
|
|
_shell_debug = g_parse_debug_string (debug_env, keys,
|
|
G_N_ELEMENTS (keys));
|
|
}
|
|
|
|
static GLogWriterOutput
|
|
default_log_writer (GLogLevelFlags log_level,
|
|
const GLogField *fields,
|
|
gsize n_fields,
|
|
gpointer user_data)
|
|
{
|
|
GLogWriterOutput output;
|
|
int i;
|
|
|
|
output = g_log_writer_default (log_level, fields, n_fields, user_data);
|
|
|
|
if ((_shell_debug & SHELL_DEBUG_BACKTRACE_WARNINGS) &&
|
|
((log_level & G_LOG_LEVEL_CRITICAL) ||
|
|
(log_level & G_LOG_LEVEL_WARNING)))
|
|
{
|
|
const char *log_domain = NULL;
|
|
|
|
for (i = 0; i < n_fields; i++)
|
|
{
|
|
if (g_strcmp0 (fields[i].key, "GLIB_DOMAIN") == 0)
|
|
{
|
|
log_domain = fields[i].value;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Filter out Gjs logs, those already have the stack */
|
|
if (g_strcmp0 (log_domain, "Gjs") != 0)
|
|
gjs_dumpstack ();
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
static GLogWriterOutput
|
|
shut_up (GLogLevelFlags log_level,
|
|
const GLogField *fields,
|
|
gsize n_fields,
|
|
gpointer user_data)
|
|
{
|
|
return (GLogWriterOutput) {0};
|
|
}
|
|
|
|
static void
|
|
dump_gjs_stack_alarm_sigaction (int signo)
|
|
{
|
|
g_log_set_writer_func (g_log_writer_default, NULL, NULL);
|
|
g_warning ("Failed to dump Javascript stack, got stuck");
|
|
g_log_set_writer_func (default_log_writer, NULL, NULL);
|
|
|
|
raise (caught_signal);
|
|
}
|
|
|
|
static void
|
|
dump_gjs_stack_on_signal_handler (int signo)
|
|
{
|
|
struct sigaction sa = { .sa_handler = dump_gjs_stack_alarm_sigaction };
|
|
gsize i;
|
|
|
|
/* Ignore all the signals starting this point, a part the one we'll raise
|
|
* (which is implicitly ignored here through SA_RESETHAND), this is needed
|
|
* not to get this handler being called by other signals that we were
|
|
* tracking and that might be emitted by code called starting from now.
|
|
*/
|
|
for (i = 0; i < G_N_ELEMENTS (_tracked_signals); ++i)
|
|
{
|
|
if (_tracked_signals[i] && i != signo)
|
|
signal (i, SIG_IGN);
|
|
}
|
|
|
|
/* Waiting at least 5 seconds for the dumpstack, if it fails, we raise the error */
|
|
caught_signal = signo;
|
|
sigemptyset (&sa.sa_mask);
|
|
sigaction (SIGALRM, &sa, NULL);
|
|
|
|
alarm (5);
|
|
gjs_dumpstack ();
|
|
alarm (0);
|
|
|
|
raise (signo);
|
|
}
|
|
|
|
static void
|
|
dump_gjs_stack_on_signal (int signo)
|
|
{
|
|
struct sigaction sa = {
|
|
.sa_flags = SA_RESETHAND | SA_NODEFER,
|
|
.sa_handler = dump_gjs_stack_on_signal_handler,
|
|
};
|
|
|
|
sigemptyset (&sa.sa_mask);
|
|
|
|
sigaction (signo, &sa, NULL);
|
|
_tracked_signals[signo] = TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
list_modes (const char *option_name,
|
|
const char *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
ShellGlobal *global;
|
|
GjsContext *context;
|
|
uint8_t status;
|
|
|
|
/* Many of our imports require global to be set, so rather than
|
|
* tayloring our imports carefully here to avoid that dependency,
|
|
* we just set it. */
|
|
g_log_set_writer_func (shut_up, NULL, NULL);
|
|
|
|
_shell_global_init (NULL);
|
|
global = shell_global_get ();
|
|
context = _shell_global_get_gjs_context (global);
|
|
|
|
shell_introspection_init ();
|
|
|
|
if (!gjs_context_eval_module_file (context,
|
|
"resource:///org/gnome/shell/ui/listModes.js",
|
|
&status,
|
|
NULL))
|
|
g_message ("Retrieving list of available modes failed.");
|
|
|
|
g_object_unref (context);
|
|
exit (status);
|
|
}
|
|
|
|
static gboolean
|
|
print_version (const gchar *option_name,
|
|
const gchar *value,
|
|
gpointer data,
|
|
GError **error)
|
|
{
|
|
g_print ("GNOME Shell %s\n", VERSION);
|
|
exit (0);
|
|
}
|
|
|
|
GOptionEntry gnome_shell_options[] = {
|
|
{
|
|
"version", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
|
|
print_version,
|
|
N_("Print version"),
|
|
NULL
|
|
},
|
|
{
|
|
"gdm-mode", 0, G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE,
|
|
&is_gdm_mode,
|
|
N_("Mode used by GDM for login screen"),
|
|
NULL
|
|
},
|
|
{
|
|
"mode", 0, 0, G_OPTION_ARG_STRING,
|
|
&session_mode,
|
|
N_("Use a specific mode, e.g. “gdm” for login screen"),
|
|
"MODE"
|
|
},
|
|
{
|
|
"list-modes", 0, G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
|
|
list_modes,
|
|
N_("List possible modes"),
|
|
NULL
|
|
},
|
|
{
|
|
"force-animations", 0, G_OPTION_FLAG_NONE, G_OPTION_ARG_NONE,
|
|
&force_animations,
|
|
N_("Force animations to be enabled"),
|
|
NULL
|
|
},
|
|
{
|
|
"automation-script", 0, G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_FILENAME,
|
|
&script_path,
|
|
"",
|
|
NULL
|
|
},
|
|
{ NULL }
|
|
};
|
|
|
|
static gboolean
|
|
on_sigterm (gpointer user_data)
|
|
{
|
|
MetaContext *context = META_CONTEXT (user_data);
|
|
|
|
meta_context_terminate (context);
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
static void
|
|
init_signal_handlers (MetaContext *context)
|
|
{
|
|
struct sigaction act = { 0 };
|
|
sigset_t empty_mask;
|
|
|
|
sigemptyset (&empty_mask);
|
|
act.sa_handler = SIG_IGN;
|
|
act.sa_mask = empty_mask;
|
|
act.sa_flags = 0;
|
|
if (sigaction (SIGPIPE, &act, NULL) < 0)
|
|
g_warning ("Failed to register SIGPIPE handler: %s", g_strerror (errno));
|
|
#ifdef SIGXFSZ
|
|
if (sigaction (SIGXFSZ, &act, NULL) < 0)
|
|
g_warning ("Failed to register SIGXFSZ handler: %s", g_strerror (errno));
|
|
#endif
|
|
|
|
g_unix_signal_add (SIGTERM, on_sigterm, context);
|
|
}
|
|
|
|
static void
|
|
change_to_home_directory (void)
|
|
{
|
|
const char *home_dir;
|
|
|
|
home_dir = g_get_home_dir ();
|
|
if (!home_dir)
|
|
return;
|
|
|
|
if (chdir (home_dir) < 0)
|
|
g_warning ("Could not change to home directory %s", home_dir);
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
g_autoptr (MetaContext) context = NULL;
|
|
g_autoptr (GFile) automation_script = NULL;
|
|
g_autofree char *cwd = NULL;
|
|
GError *error = NULL;
|
|
int ecode = EXIT_SUCCESS;
|
|
|
|
bindtextdomain (GETTEXT_PACKAGE, LOCALEDIR);
|
|
bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
|
|
textdomain (GETTEXT_PACKAGE);
|
|
|
|
/* FIXME: Add gjs API to set this stuff and don't depend on the
|
|
* environment. These propagate to child processes.
|
|
*/
|
|
g_setenv ("GJS_DEBUG_OUTPUT", "stderr", TRUE);
|
|
g_setenv ("GJS_DEBUG_TOPICS", "JS ERROR;JS LOG", TRUE);
|
|
|
|
context = meta_create_context (WM_NAME);
|
|
meta_context_add_option_entries (context, gnome_shell_options,
|
|
GETTEXT_PACKAGE);
|
|
meta_context_add_option_group (context, g_irepository_get_option_group ());
|
|
|
|
session_mode = (char *) g_getenv ("GNOME_SHELL_SESSION_MODE");
|
|
|
|
if (!meta_context_configure (context, &argc, &argv, &error))
|
|
{
|
|
g_printerr ("Failed to configure: %s\n", error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
meta_context_set_plugin_gtype (context, gnome_shell_plugin_get_type ());
|
|
meta_context_set_gnome_wm_keybindings (context, GNOME_WM_KEYBINDINGS);
|
|
|
|
init_signal_handlers (context);
|
|
cwd = g_get_current_dir ();
|
|
change_to_home_directory ();
|
|
|
|
if (session_mode == NULL)
|
|
session_mode = is_gdm_mode ? (char *)"gdm" : (char *)"user";
|
|
|
|
dump_gjs_stack_on_signal (SIGABRT);
|
|
dump_gjs_stack_on_signal (SIGFPE);
|
|
dump_gjs_stack_on_signal (SIGIOT);
|
|
dump_gjs_stack_on_signal (SIGTRAP);
|
|
|
|
if ((_shell_debug & SHELL_DEBUG_BACKTRACE_SEGFAULTS))
|
|
{
|
|
dump_gjs_stack_on_signal (SIGBUS);
|
|
dump_gjs_stack_on_signal (SIGSEGV);
|
|
}
|
|
|
|
if (script_path)
|
|
automation_script = g_file_new_for_commandline_arg_and_cwd (script_path, cwd);
|
|
|
|
/* Initialize the Shell global, including GjsContext
|
|
* GjsContext will iterate the default main loop to
|
|
* resolve internal modules.
|
|
*/
|
|
_shell_global_init ("session-mode", session_mode,
|
|
"force-animations", force_animations,
|
|
"automation-script", automation_script,
|
|
NULL);
|
|
|
|
/* Setup Meta _after_ the Shell global to avoid GjsContext
|
|
* iterating on the main loop once Meta starts adding events
|
|
*/
|
|
if (!meta_context_setup (context, &error))
|
|
{
|
|
g_printerr ("Failed to setup: %s\n", error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
shell_init_debug (g_getenv ("SHELL_DEBUG"));
|
|
|
|
shell_dbus_init (meta_context_is_replacing (context));
|
|
shell_a11y_init ();
|
|
shell_perf_log_init ();
|
|
shell_introspection_init ();
|
|
shell_fonts_init ();
|
|
|
|
g_log_set_writer_func (default_log_writer, NULL, NULL);
|
|
|
|
shell_profiler_init ();
|
|
|
|
if (meta_context_get_compositor_type (context) == META_COMPOSITOR_TYPE_WAYLAND)
|
|
meta_context_raise_rlimit_nofile (context, NULL);
|
|
|
|
if (!meta_context_start (context, &error))
|
|
{
|
|
g_printerr ("GNOME Shell failed to start: %s\n", error->message);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
/* init.js calls meta_context_start_main_loop(), gjs_context_eval_module_file()
|
|
* will not return until Mutter is exited.
|
|
*/
|
|
GjsContext *gjs_context = _shell_global_get_gjs_context (shell_global_get());
|
|
uint8_t status;
|
|
if (!gjs_context_eval_module_file (gjs_context,
|
|
"resource:///org/gnome/shell/ui/init.js",
|
|
&status,
|
|
&error))
|
|
{
|
|
g_message ("Execution of main.js threw exception: %s", error->message);
|
|
g_error_free (error);
|
|
/* We just exit() here, since in a development environment you'll get the
|
|
* error in your shell output, and it's way better than a busted WM,
|
|
* which typically manifests as a white screen.
|
|
*
|
|
* In production, we shouldn't crash =) But if we do, we should get
|
|
* restarted by the session infrastructure, which is likely going
|
|
* to be better than some undefined state.
|
|
*
|
|
* If there was a generic "hook into bug-buddy for non-C crashes"
|
|
* infrastructure, here would be the place to put it.
|
|
*/
|
|
g_object_unref (gjs_context);
|
|
exit (1);
|
|
}
|
|
|
|
g_message ("Shutting down GNOME Shell");
|
|
_shell_global_notify_shutdown (shell_global_get ());
|
|
shell_profiler_shutdown ();
|
|
|
|
g_debug ("Tearing down the gjs context");
|
|
_shell_global_destroy_gjs_context (shell_global_get ());
|
|
g_object_unref (shell_global_get ());
|
|
|
|
g_debug ("Tearing down the mutter context");
|
|
meta_context_destroy (g_steal_pointer (&context));
|
|
|
|
return ecode;
|
|
}
|