d58c9a57c6
If we are replaced via the window manager selection, then we close the MetaDisplay before meta_finalize() runs. We should be careful not to try to close the display again (and access already freed memory) in that case, so clear the global 'the_display' variable. See Metacity bug http://bugzilla.gnome.org/show_bug.cgi?id=588119 http://bugzilla.gnome.org/show_bug.cgi?id=592742
816 lines
22 KiB
C
816 lines
22 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
/* Mutter main() */
|
|
|
|
/*
|
|
* Copyright (C) 2001 Havoc Pennington
|
|
* Copyright (C) 2006 Elijah Newren
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*/
|
|
|
|
/**
|
|
* \file
|
|
* Program startup.
|
|
* Functions which parse the command-line arguments, create the display,
|
|
* kick everything off and then close down Mutter when it's time to go.
|
|
*/
|
|
|
|
/**
|
|
* \mainpage
|
|
* Mutter - a boring window manager for the adult in you
|
|
*
|
|
* Many window managers are like Marshmallow Froot Loops; Mutter
|
|
* is like Cheerios.
|
|
*
|
|
* The best way to get a handle on how the whole system fits together
|
|
* is discussed in doc/code-overview.txt; if you're looking for functions
|
|
* to investigate, read main(), meta_display_open(), and event_callback().
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
#define _SVID_SOURCE /* for putenv() and some signal-related functions */
|
|
|
|
#include <config.h>
|
|
#include "main.h"
|
|
#include "util.h"
|
|
#include "display-private.h"
|
|
#include "errors.h"
|
|
#include "ui.h"
|
|
#include "session.h"
|
|
#include "prefs.h"
|
|
#include "compositor.h"
|
|
|
|
#include <glib-object.h>
|
|
#include <gdk/gdkx.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <sys/types.h>
|
|
#include <wait.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <signal.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <locale.h>
|
|
#include <time.h>
|
|
|
|
#include <clutter/clutter.h>
|
|
#include <clutter/x11/clutter-x11.h>
|
|
|
|
#ifdef HAVE_INTROSPECTION
|
|
#include <girepository.h>
|
|
#include "compositor/mutter-plugin-manager.h"
|
|
#endif
|
|
|
|
/**
|
|
* The exit code we'll return to our parent process when we eventually die.
|
|
*/
|
|
static MetaExitCode meta_exit_code = META_EXIT_SUCCESS;
|
|
|
|
/**
|
|
* Handle on the main loop, so that we have an easy way of shutting Mutter
|
|
* down.
|
|
*/
|
|
static GMainLoop *meta_main_loop = NULL;
|
|
|
|
/**
|
|
* If set, Mutter will spawn an identical copy of itself immediately
|
|
* before quitting.
|
|
*/
|
|
static gboolean meta_restart_after_quit = FALSE;
|
|
|
|
static void prefs_changed_callback (MetaPreference pref,
|
|
gpointer data);
|
|
|
|
/**
|
|
* Prints log messages. If Mutter was compiled with backtrace support,
|
|
* also prints a backtrace (see meta_print_backtrace()).
|
|
*
|
|
* \param log_domain the domain the error occurred in (we ignore this)
|
|
* \param log_level the log level so that we can filter out less
|
|
* important messages
|
|
* \param message the message to log
|
|
* \param user_data arbitrary data (we ignore this)
|
|
*/
|
|
static void
|
|
log_handler (const gchar *log_domain,
|
|
GLogLevelFlags log_level,
|
|
const gchar *message,
|
|
gpointer user_data)
|
|
{
|
|
meta_warning ("Log level %d: %s\n", log_level, message);
|
|
meta_print_backtrace ();
|
|
}
|
|
|
|
/**
|
|
* Prints the version notice. This is shown when Mutter is called
|
|
* with the --version switch.
|
|
*/
|
|
static void
|
|
version (void)
|
|
{
|
|
g_print (_("mutter %s\n"
|
|
"Copyright (C) 2001-2008 Havoc Pennington, Red Hat, Inc., and others\n"
|
|
"This is free software; see the source for copying conditions.\n"
|
|
"There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n"),
|
|
VERSION);
|
|
exit (0);
|
|
}
|
|
|
|
/**
|
|
* Prints a list of which configure script options were used to
|
|
* build this copy of Mutter. This is actually always called
|
|
* on startup, but it's all no-op unless we're in verbose mode
|
|
* (see meta_set_verbose).
|
|
*/
|
|
static void
|
|
meta_print_compilation_info (void)
|
|
{
|
|
#ifdef HAVE_SHAPE
|
|
meta_verbose ("Compiled with shape extension\n");
|
|
#else
|
|
meta_verbose ("Compiled without shape extension\n");
|
|
#endif
|
|
#ifdef HAVE_XINERAMA
|
|
meta_topic (META_DEBUG_XINERAMA, "Compiled with Xinerama extension\n");
|
|
#else
|
|
meta_topic (META_DEBUG_XINERAMA, "Compiled without Xinerama extension\n");
|
|
#endif
|
|
#ifdef HAVE_XFREE_XINERAMA
|
|
meta_topic (META_DEBUG_XINERAMA, " (using XFree86 Xinerama)\n");
|
|
#else
|
|
meta_topic (META_DEBUG_XINERAMA, " (not using XFree86 Xinerama)\n");
|
|
#endif
|
|
#ifdef HAVE_SOLARIS_XINERAMA
|
|
meta_topic (META_DEBUG_XINERAMA, " (using Solaris Xinerama)\n");
|
|
#else
|
|
meta_topic (META_DEBUG_XINERAMA, " (not using Solaris Xinerama)\n");
|
|
#endif
|
|
#ifdef HAVE_XSYNC
|
|
meta_verbose ("Compiled with sync extension\n");
|
|
#else
|
|
meta_verbose ("Compiled without sync extension\n");
|
|
#endif
|
|
#ifdef HAVE_RANDR
|
|
meta_verbose ("Compiled with randr extension\n");
|
|
#else
|
|
meta_verbose ("Compiled without randr extension\n");
|
|
#endif
|
|
#ifdef HAVE_STARTUP_NOTIFICATION
|
|
meta_verbose ("Compiled with startup notification\n");
|
|
#else
|
|
meta_verbose ("Compiled without startup notification\n");
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* Prints the version number, the current timestamp (not the
|
|
* build date), the locale, the character encoding, and a list
|
|
* of configure script options that were used to build this
|
|
* copy of Mutter. This is actually always called
|
|
* on startup, but it's all no-op unless we're in verbose mode
|
|
* (see meta_set_verbose).
|
|
*/
|
|
static void
|
|
meta_print_self_identity (void)
|
|
{
|
|
char buf[256];
|
|
GDate d;
|
|
const char *charset;
|
|
|
|
/* Version and current date. */
|
|
g_date_clear (&d, 1);
|
|
g_date_set_time_t (&d, time (NULL));
|
|
g_date_strftime (buf, sizeof (buf), "%x", &d);
|
|
meta_verbose ("Mutter version %s running on %s\n",
|
|
VERSION, buf);
|
|
|
|
/* Locale and encoding. */
|
|
g_get_charset (&charset);
|
|
meta_verbose ("Running in locale \"%s\" with encoding \"%s\"\n",
|
|
setlocale (LC_ALL, NULL), charset);
|
|
|
|
/* Compilation settings. */
|
|
meta_print_compilation_info ();
|
|
}
|
|
|
|
/**
|
|
* The set of possible options that can be set on Mutter's
|
|
* command line. This type exists so that meta_parse_options() can
|
|
* write to an instance of it.
|
|
*/
|
|
typedef struct
|
|
{
|
|
gchar *save_file;
|
|
gchar *display_name;
|
|
gchar *client_id;
|
|
gchar *mutter_plugins;
|
|
gboolean replace_wm;
|
|
gboolean disable_sm;
|
|
gboolean print_version;
|
|
gboolean sync;
|
|
gboolean composite;
|
|
gboolean no_composite;
|
|
gboolean no_tab_popup;
|
|
gchar *introspect;
|
|
} MetaArguments;
|
|
|
|
#define COMPOSITE_OPTS_FLAGS 0
|
|
|
|
/**
|
|
* Parses argc and argv and returns the
|
|
* arguments that Mutter understands in meta_args.
|
|
*
|
|
* The strange call signature has to be written like it is so
|
|
* that g_option_context_parse() gets a chance to modify argc and
|
|
* argv.
|
|
*
|
|
* \param argc Pointer to the number of arguments Mutter was given
|
|
* \param argv Pointer to the array of arguments Mutter was given
|
|
* \param meta_args The result of parsing the arguments.
|
|
**/
|
|
static GOptionContext *
|
|
meta_parse_options (int *argc, char ***argv,
|
|
MetaArguments *meta_args)
|
|
{
|
|
MetaArguments my_args = {NULL, NULL, NULL, NULL,
|
|
FALSE, FALSE, FALSE, FALSE, FALSE};
|
|
GOptionEntry options[] = {
|
|
{
|
|
"sm-disable", 0, 0, G_OPTION_ARG_NONE,
|
|
&my_args.disable_sm,
|
|
N_("Disable connection to session manager"),
|
|
NULL
|
|
},
|
|
{
|
|
"replace", 0, 0, G_OPTION_ARG_NONE,
|
|
&my_args.replace_wm,
|
|
N_("Replace the running window manager with Mutter"),
|
|
NULL
|
|
},
|
|
{
|
|
"sm-client-id", 0, 0, G_OPTION_ARG_STRING,
|
|
&my_args.client_id,
|
|
N_("Specify session management ID"),
|
|
"ID"
|
|
},
|
|
{
|
|
"display", 'd', 0, G_OPTION_ARG_STRING,
|
|
&my_args.display_name, N_("X Display to use"),
|
|
"DISPLAY"
|
|
},
|
|
{
|
|
"sm-save-file", 0, 0, G_OPTION_ARG_FILENAME,
|
|
&my_args.save_file,
|
|
N_("Initialize session from savefile"),
|
|
"FILE"
|
|
},
|
|
{
|
|
"version", 0, 0, G_OPTION_ARG_NONE,
|
|
&my_args.print_version,
|
|
N_("Print version"),
|
|
NULL
|
|
},
|
|
{
|
|
"sync", 0, 0, G_OPTION_ARG_NONE,
|
|
&my_args.sync,
|
|
N_("Make X calls synchronous"),
|
|
NULL
|
|
},
|
|
{
|
|
"composite", 'c', COMPOSITE_OPTS_FLAGS, G_OPTION_ARG_NONE,
|
|
&my_args.composite,
|
|
N_("Turn compositing on"),
|
|
NULL
|
|
},
|
|
{
|
|
"no-composite", 0, COMPOSITE_OPTS_FLAGS, G_OPTION_ARG_NONE,
|
|
&my_args.no_composite,
|
|
N_("Turn compositing off"),
|
|
NULL
|
|
},
|
|
{
|
|
"mutter-plugins", 0, 0, G_OPTION_ARG_STRING,
|
|
&my_args.mutter_plugins,
|
|
N_("Comma-separated list of compositor plugins"),
|
|
"PLUGINS"
|
|
},
|
|
{
|
|
"no-tab-popup", 0, 0, G_OPTION_ARG_NONE,
|
|
&my_args.no_tab_popup,
|
|
N_("Whether window popup/frame should be shown when cycling windows."),
|
|
NULL
|
|
},
|
|
#ifdef HAVE_INTROSPECTION
|
|
{
|
|
"introspect-dump", 0, 0, G_OPTION_ARG_STRING,
|
|
&my_args.introspect,
|
|
N_("Internal argument for GObject introspection"), "INTROSPECT"
|
|
},
|
|
#endif
|
|
{NULL}
|
|
};
|
|
GOptionContext *ctx;
|
|
GError *error = NULL;
|
|
|
|
ctx = g_option_context_new (NULL);
|
|
g_option_context_add_main_entries (ctx, options, "mutter");
|
|
g_option_context_add_group (ctx, clutter_get_option_group_without_init ());
|
|
|
|
if (!g_option_context_parse (ctx, argc, argv, &error))
|
|
{
|
|
g_print ("mutter: %s\n", error->message);
|
|
exit(1);
|
|
}
|
|
|
|
/* Return the parsed options through the meta_args param. */
|
|
*meta_args = my_args;
|
|
return ctx;
|
|
}
|
|
|
|
/* Mutter is responsible for pulling events off the X queue, so Clutter
|
|
* doesn't need (and shouldn't) run its normal event source which polls
|
|
* the X fd, but we do have to deal with dispatching events that accumulate
|
|
* in the clutter queue. This happens, for example, when clutter generate
|
|
* enter/leave events on mouse motion - several events are queued in the
|
|
* clutter queue but only one dispatched. It could also happen because of
|
|
* explicit calls to clutter_event_put(). We add a very simple custom
|
|
* event loop source which is simply responsible for pulling events off
|
|
* of the queue and dispatching them before we block for new events.
|
|
*/
|
|
|
|
static gboolean
|
|
event_prepare (GSource *source,
|
|
gint *timeout_)
|
|
{
|
|
*timeout_ = -1;
|
|
|
|
return clutter_events_pending ();
|
|
}
|
|
|
|
static gboolean
|
|
event_check (GSource *source)
|
|
{
|
|
return clutter_events_pending ();
|
|
}
|
|
|
|
static gboolean
|
|
event_dispatch (GSource *source,
|
|
GSourceFunc callback,
|
|
gpointer user_data)
|
|
{
|
|
ClutterEvent *event = clutter_event_get ();
|
|
if (event)
|
|
clutter_do_event (event);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GSourceFuncs event_funcs = {
|
|
event_prepare,
|
|
event_check,
|
|
event_dispatch
|
|
};
|
|
|
|
static void
|
|
meta_clutter_init (GOptionContext *ctx, int *argc, char ***argv)
|
|
{
|
|
clutter_x11_set_display (gdk_display);
|
|
clutter_x11_disable_event_retrieval ();
|
|
|
|
if (CLUTTER_INIT_SUCCESS == clutter_init (argc, argv))
|
|
{
|
|
GSource *source = g_source_new (&event_funcs, sizeof (GSource));
|
|
g_source_attach (source, NULL);
|
|
g_source_unref (source);
|
|
}
|
|
else
|
|
{
|
|
meta_fatal ("Unable to initialize Clutter.\n");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Selects which display Mutter should use. It first tries to use
|
|
* display_name as the display. If display_name is NULL then
|
|
* try to use the environment variable MUTTER_DISPLAY. If that
|
|
* also is NULL, use the default - :0.0
|
|
*/
|
|
static void
|
|
meta_select_display (gchar *display_name)
|
|
{
|
|
gchar *envVar = "";
|
|
if (display_name)
|
|
envVar = g_strconcat ("DISPLAY=", display_name, NULL);
|
|
else if (g_getenv ("MUTTER_DISPLAY"))
|
|
envVar = g_strconcat ("DISPLAY=",
|
|
g_getenv ("MUTTER_DISPLAY"), NULL);
|
|
/* DO NOT FREE envVar, putenv() sucks */
|
|
putenv (envVar);
|
|
}
|
|
|
|
static void
|
|
meta_finalize (void)
|
|
{
|
|
MetaDisplay *display = meta_get_display ();
|
|
|
|
if (display)
|
|
meta_display_close (display,
|
|
CurrentTime); /* I doubt correct timestamps matter here */
|
|
|
|
meta_session_shutdown ();
|
|
}
|
|
|
|
static void
|
|
sigterm_handler (int signum)
|
|
{
|
|
meta_finalize ();
|
|
|
|
exit (meta_exit_code);
|
|
}
|
|
|
|
static guint sigchld_signal_id = 0;
|
|
|
|
static void
|
|
sigchld_handler (int signum, siginfo_t *info, void *context)
|
|
{
|
|
int stat;
|
|
|
|
if (info->si_code == CLD_EXITED)
|
|
{
|
|
g_signal_emit (sigchld_nexus, sigchld_signal_id, 0,
|
|
info->si_status,
|
|
GINT_TO_POINTER (info->si_pid));
|
|
}
|
|
|
|
g_signal_handlers_disconnect_matched (sigchld_nexus,
|
|
G_SIGNAL_MATCH_DATA,
|
|
sigchld_signal_id,
|
|
0, NULL, NULL,
|
|
GINT_TO_POINTER (info->si_pid));
|
|
|
|
waitpid (info->si_pid, &stat, WNOHANG);
|
|
}
|
|
|
|
/**
|
|
* This is where the story begins. It parses commandline options and
|
|
* environment variables, sets up the screen, hands control off to
|
|
* GTK, and cleans up afterwards.
|
|
*
|
|
* \param argc Number of arguments (as usual)
|
|
* \param argv Array of arguments (as usual)
|
|
*
|
|
* \bug It's a bit long. It would be good to split it out into separate
|
|
* functions.
|
|
*/
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
struct sigaction act;
|
|
sigset_t empty_mask;
|
|
MetaArguments meta_args;
|
|
const gchar *log_domains[] = {
|
|
NULL, G_LOG_DOMAIN, "Gtk", "Gdk", "GLib",
|
|
"Pango", "GLib-GObject", "GThread"
|
|
};
|
|
guint i;
|
|
GOptionContext *ctx;
|
|
|
|
if (!g_thread_supported ())
|
|
g_thread_init (NULL);
|
|
|
|
if (setlocale (LC_ALL, "") == NULL)
|
|
meta_warning ("Locale not understood by C library, internationalization will not work\n");
|
|
|
|
g_type_init ();
|
|
|
|
sigemptyset (&empty_mask);
|
|
act.sa_handler = SIG_IGN;
|
|
act.sa_mask = empty_mask;
|
|
act.sa_flags = 0;
|
|
if (sigaction (SIGPIPE, &act, NULL) < 0)
|
|
g_printerr ("Failed to register SIGPIPE handler: %s\n",
|
|
g_strerror (errno));
|
|
#ifdef SIGXFSZ
|
|
if (sigaction (SIGXFSZ, &act, NULL) < 0)
|
|
g_printerr ("Failed to register SIGXFSZ handler: %s\n",
|
|
g_strerror (errno));
|
|
#endif
|
|
|
|
act.sa_handler = &sigterm_handler;
|
|
if (sigaction (SIGTERM, &act, NULL) < 0)
|
|
g_printerr ("Failed to register SIGTERM handler: %s\n",
|
|
g_strerror (errno));
|
|
|
|
sigchld_nexus = g_object_new (META_TYPE_NEXUS, NULL);
|
|
|
|
sigchld_signal_id =
|
|
g_signal_new ("sigchld", META_TYPE_NEXUS,
|
|
G_SIGNAL_RUN_LAST,
|
|
0, NULL, NULL,
|
|
g_cclosure_marshal_VOID__UINT_POINTER,
|
|
G_TYPE_NONE,
|
|
2,
|
|
G_TYPE_UINT, G_TYPE_POINTER);
|
|
|
|
act.sa_flags = SA_NOCLDSTOP | SA_SIGINFO;
|
|
act.sa_handler = SIG_DFL;
|
|
act.sa_sigaction = &sigchld_handler;
|
|
if (sigaction (SIGCHLD, &act, NULL) < 0)
|
|
g_printerr ("Failed to register SIGCHLD handler: %s\n",
|
|
g_strerror (errno));
|
|
|
|
if (g_getenv ("MUTTER_VERBOSE"))
|
|
meta_set_verbose (TRUE);
|
|
if (g_getenv ("MUTTER_DEBUG"))
|
|
meta_set_debugging (TRUE);
|
|
|
|
if (g_get_home_dir ())
|
|
if (chdir (g_get_home_dir ()) < 0)
|
|
meta_warning ("Could not change to home directory %s.\n",
|
|
g_get_home_dir ());
|
|
|
|
g_type_init ();
|
|
|
|
meta_print_self_identity ();
|
|
|
|
bindtextdomain (GETTEXT_PACKAGE, MUTTER_LOCALEDIR);
|
|
bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
|
|
textdomain (GETTEXT_PACKAGE);
|
|
|
|
/* Parse command line arguments.*/
|
|
ctx = meta_parse_options (&argc, &argv, &meta_args);
|
|
|
|
/* This must come before the introspect below, so we load all the plugins
|
|
* in order to get their get_type functions.
|
|
*/
|
|
if (meta_args.mutter_plugins)
|
|
{
|
|
char **plugins = g_strsplit (meta_args.mutter_plugins, ",", -1);
|
|
char **plugin;
|
|
GSList *plugins_list = NULL;
|
|
|
|
for (plugin = plugins; *plugin; plugin++)
|
|
{
|
|
g_strstrip (*plugin);
|
|
plugins_list = g_slist_prepend (plugins_list, *plugin);
|
|
}
|
|
|
|
plugins_list = g_slist_reverse (plugins_list);
|
|
meta_prefs_override_clutter_plugins (plugins_list);
|
|
|
|
g_slist_free(plugins_list);
|
|
g_strfreev (plugins);
|
|
}
|
|
|
|
#ifdef HAVE_INTROSPECTION
|
|
g_irepository_prepend_search_path (MUTTER_PKGLIBDIR);
|
|
if (meta_args.introspect)
|
|
{
|
|
GError *error = NULL;
|
|
if (meta_args.mutter_plugins)
|
|
{
|
|
/* We need to load all plugins so that we can call their
|
|
* get_type functions. We do not call
|
|
* mutter_plugin_manager_initialize because almost nothing else
|
|
* is initialized at this point, and we don't plan to run any real
|
|
* plugin code.
|
|
*/
|
|
MutterPluginManager *mgr = mutter_plugin_manager_new (NULL);
|
|
if (!mutter_plugin_manager_load (mgr))
|
|
g_critical ("failed to load plugins");
|
|
}
|
|
if (!g_irepository_dump (meta_args.introspect, &error))
|
|
{
|
|
g_printerr ("failed to dump: %s\n", error->message);
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
meta_set_syncing (meta_args.sync || (g_getenv ("MUTTER_SYNC") != NULL));
|
|
|
|
if (meta_args.print_version)
|
|
version ();
|
|
|
|
meta_select_display (meta_args.display_name);
|
|
|
|
if (meta_args.replace_wm)
|
|
meta_set_replace_current_wm (TRUE);
|
|
|
|
if (meta_args.save_file && meta_args.client_id)
|
|
meta_fatal ("Can't specify both SM save file and SM client id\n");
|
|
|
|
meta_main_loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
meta_ui_init (&argc, &argv);
|
|
|
|
/*
|
|
* Clutter can only be initialized after the UI.
|
|
*/
|
|
meta_clutter_init (ctx, &argc, &argv);
|
|
|
|
g_option_context_free (ctx);
|
|
|
|
/* must be after UI init so we can override GDK handlers */
|
|
meta_errors_init ();
|
|
|
|
/* Load prefs */
|
|
meta_prefs_init ();
|
|
meta_prefs_add_listener (prefs_changed_callback, NULL);
|
|
|
|
|
|
#if 1
|
|
|
|
for (i=0; i<G_N_ELEMENTS(log_domains); i++)
|
|
g_log_set_handler (log_domains[i],
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
log_handler, NULL);
|
|
|
|
#endif
|
|
|
|
if (g_getenv ("MUTTER_G_FATAL_WARNINGS") != NULL)
|
|
g_log_set_always_fatal (G_LOG_LEVEL_MASK);
|
|
|
|
meta_ui_set_current_theme (meta_prefs_get_theme (), FALSE);
|
|
|
|
/* Try to find some theme that'll work if the theme preference
|
|
* doesn't exist. First try Simple (the default theme) then just
|
|
* try anything in the themes directory.
|
|
*/
|
|
if (!meta_ui_have_a_theme ())
|
|
meta_ui_set_current_theme ("Simple", FALSE);
|
|
|
|
if (!meta_ui_have_a_theme ())
|
|
{
|
|
const char *dir_entry = NULL;
|
|
GError *err = NULL;
|
|
GDir *themes_dir = NULL;
|
|
|
|
if (!(themes_dir = g_dir_open (MUTTER_DATADIR"/themes", 0, &err)))
|
|
{
|
|
meta_fatal (_("Failed to scan themes directory: %s\n"), err->message);
|
|
g_error_free (err);
|
|
}
|
|
else
|
|
{
|
|
while (((dir_entry = g_dir_read_name (themes_dir)) != NULL) &&
|
|
(!meta_ui_have_a_theme ()))
|
|
{
|
|
meta_ui_set_current_theme (dir_entry, FALSE);
|
|
}
|
|
|
|
g_dir_close (themes_dir);
|
|
}
|
|
}
|
|
|
|
if (!meta_ui_have_a_theme ())
|
|
meta_fatal (_("Could not find a theme! Be sure %s exists and contains the usual themes.\n"),
|
|
MUTTER_DATADIR"/themes");
|
|
|
|
/* Connect to SM as late as possible - but before managing display,
|
|
* or we might try to manage a window before we have the session
|
|
* info
|
|
*/
|
|
if (!meta_args.disable_sm)
|
|
{
|
|
if (meta_args.client_id == NULL)
|
|
{
|
|
const gchar *desktop_autostart_id;
|
|
|
|
desktop_autostart_id = g_getenv ("DESKTOP_AUTOSTART_ID");
|
|
|
|
if (desktop_autostart_id != NULL)
|
|
meta_args.client_id = g_strdup (desktop_autostart_id);
|
|
}
|
|
|
|
/* Unset DESKTOP_AUTOSTART_ID in order to avoid child processes to
|
|
* use the same client id. */
|
|
g_unsetenv ("DESKTOP_AUTOSTART_ID");
|
|
|
|
meta_session_init (meta_args.client_id, meta_args.save_file);
|
|
}
|
|
/* Free memory possibly allocated by the argument parsing which are
|
|
* no longer needed.
|
|
*/
|
|
g_free (meta_args.save_file);
|
|
g_free (meta_args.display_name);
|
|
g_free (meta_args.client_id);
|
|
|
|
if (meta_args.composite || meta_args.no_composite)
|
|
meta_prefs_set_compositing_manager (meta_args.composite);
|
|
|
|
if (meta_args.no_tab_popup)
|
|
{
|
|
meta_prefs_override_no_tab_popup (TRUE);
|
|
}
|
|
|
|
if (!meta_display_open ())
|
|
meta_exit (META_EXIT_ERROR);
|
|
|
|
g_main_loop_run (meta_main_loop);
|
|
|
|
meta_finalize ();
|
|
|
|
if (meta_restart_after_quit)
|
|
{
|
|
GError *err;
|
|
|
|
err = NULL;
|
|
if (!g_spawn_async (NULL,
|
|
argv,
|
|
NULL,
|
|
G_SPAWN_SEARCH_PATH,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
&err))
|
|
{
|
|
meta_fatal (_("Failed to restart: %s\n"),
|
|
err->message);
|
|
g_error_free (err); /* not reached anyhow */
|
|
meta_exit_code = META_EXIT_ERROR;
|
|
}
|
|
}
|
|
|
|
return meta_exit_code;
|
|
}
|
|
|
|
/**
|
|
* Stops Mutter. This tells the event loop to stop processing; it is rather
|
|
* dangerous to use this rather than meta_restart() because this will leave
|
|
* the user with no window manager. We generally do this only if, for example,
|
|
* the session manager asks us to; we assume the session manager knows what
|
|
* it's talking about.
|
|
*
|
|
* \param code The success or failure code to return to the calling process.
|
|
*/
|
|
void
|
|
meta_quit (MetaExitCode code)
|
|
{
|
|
meta_exit_code = code;
|
|
|
|
if (g_main_loop_is_running (meta_main_loop))
|
|
g_main_loop_quit (meta_main_loop);
|
|
}
|
|
|
|
/**
|
|
* Restarts Mutter. In practice, this tells the event loop to stop
|
|
* processing, having first set the meta_restart_after_quit flag which
|
|
* tells Mutter to spawn an identical copy of itself before quitting.
|
|
* This happens on receipt of a _MUTTER_RESTART_MESSAGE client event.
|
|
*/
|
|
void
|
|
meta_restart (void)
|
|
{
|
|
meta_restart_after_quit = TRUE;
|
|
meta_quit (META_EXIT_SUCCESS);
|
|
}
|
|
|
|
/**
|
|
* Called on pref changes. (One of several functions of its kind and purpose.)
|
|
*
|
|
* \bug Why are these particular prefs handled in main.c and not others?
|
|
* Should they be?
|
|
*
|
|
* \param pref Which preference has changed
|
|
* \param data Arbitrary data (which we ignore)
|
|
*/
|
|
static void
|
|
prefs_changed_callback (MetaPreference pref,
|
|
gpointer data)
|
|
{
|
|
switch (pref)
|
|
{
|
|
case META_PREF_THEME:
|
|
meta_ui_set_current_theme (meta_prefs_get_theme (), FALSE);
|
|
meta_display_retheme_all ();
|
|
break;
|
|
|
|
case META_PREF_CURSOR_THEME:
|
|
case META_PREF_CURSOR_SIZE:
|
|
meta_display_set_cursor_theme (meta_prefs_get_cursor_theme (),
|
|
meta_prefs_get_cursor_size ());
|
|
break;
|
|
default:
|
|
/* handled elsewhere or otherwise */
|
|
break;
|
|
}
|
|
}
|