2001-05-30 11:36:31 -04:00
|
|
|
/* Metacity main() */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2001 Havoc Pennington
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2001-12-09 17:41:12 -05:00
|
|
|
#include <config.h>
|
2001-05-30 11:36:31 -04:00
|
|
|
#include "main.h"
|
|
|
|
#include "util.h"
|
|
|
|
#include "display.h"
|
2001-05-30 23:30:58 -04:00
|
|
|
#include "errors.h"
|
2001-06-17 15:53:45 -04:00
|
|
|
#include "ui.h"
|
2001-06-22 02:21:44 -04:00
|
|
|
#include "session.h"
|
2001-12-09 17:41:12 -05:00
|
|
|
#include "prefs.h"
|
2001-05-30 11:36:31 -04:00
|
|
|
|
2001-06-19 23:01:26 -04:00
|
|
|
#include <glib-object.h>
|
2001-06-18 02:11:53 -04:00
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
#include <stdlib.h>
|
2001-06-11 01:47:51 -04:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <unistd.h>
|
2001-10-15 00:14:58 -04:00
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
2002-05-11 01:47:04 -04:00
|
|
|
#include <locale.h>
|
2001-05-30 11:36:31 -04:00
|
|
|
|
|
|
|
static MetaExitCode meta_exit_code = META_EXIT_SUCCESS;
|
|
|
|
static GMainLoop *meta_main_loop = NULL;
|
2001-10-15 00:14:58 -04:00
|
|
|
static gboolean meta_restart_after_quit = FALSE;
|
2001-05-30 11:36:31 -04:00
|
|
|
|
2002-02-06 22:07:56 -05:00
|
|
|
static void prefs_changed_callback (MetaPreference pref,
|
|
|
|
gpointer data);
|
|
|
|
|
2001-09-17 01:50:02 -04:00
|
|
|
static void
|
|
|
|
log_handler (const gchar *log_domain,
|
|
|
|
GLogLevelFlags log_level,
|
|
|
|
const gchar *message,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
2001-12-09 22:55:26 -05:00
|
|
|
meta_warning ("Log level %d: %s\n", log_level, message);
|
2001-09-17 01:50:02 -04:00
|
|
|
}
|
|
|
|
|
2001-06-22 02:21:44 -04:00
|
|
|
static void
|
|
|
|
usage (void)
|
|
|
|
{
|
2002-03-13 23:09:42 -05:00
|
|
|
g_print ("metacity [--disable-sm] [--sm-save-file=FILENAME] [--display=DISPLAY]\n");
|
2001-09-15 20:30:45 -04:00
|
|
|
exit (1);
|
2001-06-22 02:21:44 -04:00
|
|
|
}
|
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
int
|
|
|
|
main (int argc, char **argv)
|
|
|
|
{
|
2001-06-11 01:47:51 -04:00
|
|
|
struct sigaction act;
|
|
|
|
sigset_t empty_mask;
|
2001-06-12 00:38:24 -04:00
|
|
|
char *display_name;
|
2001-06-22 02:21:44 -04:00
|
|
|
int i;
|
|
|
|
const char *client_id;
|
|
|
|
gboolean disable_sm;
|
|
|
|
const char *prev_arg;
|
2002-03-13 23:09:42 -05:00
|
|
|
const char *save_file;
|
2002-04-22 14:18:38 -04:00
|
|
|
|
|
|
|
g_set_prgname (argv[0]);
|
|
|
|
|
|
|
|
if (setlocale (LC_ALL, "") == NULL)
|
|
|
|
meta_warning ("Locale not understood by C library, internationalization will not work\n");
|
2001-06-22 02:21:44 -04:00
|
|
|
|
2001-06-11 01:47:51 -04:00
|
|
|
sigemptyset (&empty_mask);
|
|
|
|
act.sa_handler = SIG_IGN;
|
|
|
|
act.sa_mask = empty_mask;
|
|
|
|
act.sa_flags = 0;
|
2002-03-04 21:43:22 -05:00
|
|
|
if (sigaction (SIGPIPE, &act, 0) < 0)
|
2002-05-10 23:37:51 -04:00
|
|
|
g_printerr ("Failed to register SIGPIPE handler: %s\n",
|
|
|
|
g_strerror (errno));
|
2002-03-04 21:43:22 -05:00
|
|
|
#ifdef SIGXFSZ
|
|
|
|
if (sigaction (SIGXFSZ, &act, 0) < 0)
|
2002-05-10 23:37:51 -04:00
|
|
|
g_printerr ("Failed to register SIGXFSZ handler: %s\n",
|
|
|
|
g_strerror (errno));
|
2002-03-04 21:43:22 -05:00
|
|
|
#endif
|
2002-03-06 17:27:24 -05:00
|
|
|
|
|
|
|
if (g_getenv ("METACITY_VERBOSE"))
|
|
|
|
meta_set_verbose (TRUE);
|
|
|
|
if (g_getenv ("METACITY_DEBUG"))
|
|
|
|
meta_set_debugging (TRUE);
|
2001-05-31 02:42:58 -04:00
|
|
|
meta_set_syncing (g_getenv ("METACITY_SYNC") != NULL);
|
2002-04-22 14:18:38 -04:00
|
|
|
|
|
|
|
{
|
|
|
|
const char *charset;
|
|
|
|
g_get_charset (&charset);
|
|
|
|
meta_verbose ("Running in locale \"%s\" with encoding \"%s\"\n",
|
|
|
|
setlocale (LC_ALL, NULL), charset);
|
|
|
|
}
|
2001-05-31 23:00:01 -04:00
|
|
|
|
2002-04-22 14:18:38 -04:00
|
|
|
bindtextdomain (GETTEXT_PACKAGE, METACITY_LOCALEDIR);
|
|
|
|
bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
|
|
|
|
textdomain (GETTEXT_PACKAGE);
|
|
|
|
|
2001-06-22 02:21:44 -04:00
|
|
|
/* Parse options lamely */
|
|
|
|
|
|
|
|
display_name = NULL;
|
|
|
|
client_id = NULL;
|
2002-03-13 23:09:42 -05:00
|
|
|
save_file = NULL;
|
2001-06-22 02:21:44 -04:00
|
|
|
disable_sm = FALSE;
|
|
|
|
prev_arg = NULL;
|
|
|
|
i = 1;
|
|
|
|
while (i < argc)
|
|
|
|
{
|
|
|
|
const char *arg = argv[i];
|
|
|
|
|
|
|
|
if (strcmp (arg, "--help") == 0 ||
|
|
|
|
strcmp (arg, "-h") == 0 ||
|
|
|
|
strcmp (arg, "-?") == 0)
|
|
|
|
usage ();
|
|
|
|
else if (strcmp (arg, "--sm-disable") == 0)
|
2002-06-08 19:55:27 -04:00
|
|
|
disable_sm = TRUE;
|
|
|
|
else if (strcmp (arg, "--replace") == 0)
|
|
|
|
meta_set_replace_current_wm (TRUE);
|
2001-06-22 02:21:44 -04:00
|
|
|
else if (strstr (arg, "--display=") == arg)
|
|
|
|
{
|
|
|
|
const char *disp;
|
|
|
|
|
|
|
|
if (display_name != NULL)
|
|
|
|
meta_fatal ("Can't specify display twice\n");
|
|
|
|
|
|
|
|
disp = strchr (arg, '=');
|
|
|
|
++disp;
|
|
|
|
|
|
|
|
display_name =
|
|
|
|
g_strconcat ("DISPLAY=", disp, NULL);
|
|
|
|
}
|
|
|
|
else if (prev_arg &&
|
|
|
|
strcmp (prev_arg, "--display") == 0)
|
|
|
|
{
|
|
|
|
if (display_name != NULL)
|
|
|
|
meta_fatal ("Can't specify display twice\n");
|
|
|
|
|
|
|
|
display_name = g_strconcat ("DISPLAY=", arg, NULL);
|
|
|
|
}
|
|
|
|
else if (strcmp (arg, "--display") == 0)
|
|
|
|
; /* wait for next arg */
|
2002-03-13 23:36:02 -05:00
|
|
|
else if (strstr (arg, "--sm-client-id=") == arg)
|
|
|
|
{
|
|
|
|
const char *id;
|
|
|
|
|
|
|
|
if (client_id)
|
|
|
|
meta_fatal ("Can't specify client ID twice\n");
|
|
|
|
|
|
|
|
id = strchr (arg, '=');
|
|
|
|
++id;
|
|
|
|
|
|
|
|
client_id = g_strdup (id);
|
|
|
|
}
|
|
|
|
else if (prev_arg &&
|
|
|
|
strcmp (prev_arg, "--sm-client-id") == 0)
|
|
|
|
{
|
|
|
|
if (client_id)
|
|
|
|
meta_fatal ("Can't specify client ID twice\n");
|
|
|
|
|
|
|
|
client_id = g_strdup (arg);
|
|
|
|
}
|
|
|
|
else if (strcmp (arg, "--sm-client-id") == 0)
|
|
|
|
; /* wait for next arg */
|
2002-03-13 23:09:42 -05:00
|
|
|
else if (strstr (arg, "--sm-save-file=") == arg)
|
2001-06-22 02:21:44 -04:00
|
|
|
{
|
2002-03-13 23:09:42 -05:00
|
|
|
const char *file;
|
2001-06-22 02:21:44 -04:00
|
|
|
|
2002-03-13 23:09:42 -05:00
|
|
|
if (save_file)
|
|
|
|
meta_fatal ("Can't specify SM save file twice\n");
|
2001-06-22 02:21:44 -04:00
|
|
|
|
2002-03-13 23:09:42 -05:00
|
|
|
file = strchr (arg, '=');
|
|
|
|
++file;
|
2001-06-22 02:21:44 -04:00
|
|
|
|
2002-03-13 23:09:42 -05:00
|
|
|
save_file = g_strdup (file);
|
2001-06-22 02:21:44 -04:00
|
|
|
}
|
|
|
|
else if (prev_arg &&
|
2002-03-13 23:09:42 -05:00
|
|
|
strcmp (prev_arg, "--sm-save-file") == 0)
|
2001-06-22 02:21:44 -04:00
|
|
|
{
|
2002-03-13 23:09:42 -05:00
|
|
|
if (save_file)
|
|
|
|
meta_fatal ("Can't specify SM save file twice\n");
|
2001-06-22 02:21:44 -04:00
|
|
|
|
2002-03-13 23:09:42 -05:00
|
|
|
save_file = g_strdup (arg);
|
2001-06-22 02:21:44 -04:00
|
|
|
}
|
2002-03-13 23:09:42 -05:00
|
|
|
else if (strcmp (arg, "--sm-save-file") == 0)
|
2001-06-22 02:21:44 -04:00
|
|
|
; /* wait for next arg */
|
|
|
|
else
|
|
|
|
usage ();
|
|
|
|
|
|
|
|
prev_arg = arg;
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
2002-03-13 23:36:02 -05:00
|
|
|
|
|
|
|
if (save_file && client_id)
|
|
|
|
meta_fatal ("Can't specify both SM save file and SM client id\n");
|
|
|
|
|
2001-06-22 02:21:44 -04:00
|
|
|
meta_main_loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
|
|
|
|
if (display_name == NULL &&
|
|
|
|
g_getenv ("METACITY_DISPLAY"))
|
2001-06-12 00:38:24 -04:00
|
|
|
{
|
|
|
|
meta_verbose ("Using METACITY_DISPLAY %s\n",
|
|
|
|
g_getenv ("METACITY_DISPLAY"));
|
|
|
|
display_name =
|
|
|
|
g_strconcat ("DISPLAY=", g_getenv ("METACITY_DISPLAY"), NULL);
|
2001-06-22 02:21:44 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (display_name)
|
|
|
|
{
|
2001-06-12 00:38:24 -04:00
|
|
|
putenv (display_name);
|
|
|
|
/* DO NOT FREE display_name, putenv() sucks */
|
2001-09-01 01:53:07 -04:00
|
|
|
}
|
2001-06-20 23:40:14 -04:00
|
|
|
|
2001-06-29 11:33:21 -04:00
|
|
|
g_type_init ();
|
2001-06-22 02:21:44 -04:00
|
|
|
|
2002-05-30 20:02:54 -04:00
|
|
|
#ifdef HAVE_SHAPE
|
|
|
|
meta_verbose ("Compiled with shape extension\n");
|
|
|
|
#else
|
|
|
|
meta_verbose ("Compiled without shape extension\n");
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_XINERAMA
|
|
|
|
meta_verbose ("Compiled with Xinerama extension\n");
|
|
|
|
#else
|
|
|
|
meta_verbose ("Compiled without Xinerama extension\n");
|
|
|
|
#endif
|
|
|
|
|
2001-12-09 17:41:12 -05:00
|
|
|
/* Load prefs */
|
|
|
|
meta_prefs_init ();
|
2002-02-06 22:07:56 -05:00
|
|
|
meta_prefs_add_listener (prefs_changed_callback, NULL);
|
2001-06-22 02:21:44 -04:00
|
|
|
|
|
|
|
meta_ui_init (&argc, &argv);
|
2001-09-01 01:53:07 -04:00
|
|
|
|
|
|
|
/* must be after UI init so we can override GDK handlers */
|
|
|
|
meta_errors_init ();
|
2001-09-17 01:50:02 -04:00
|
|
|
|
2002-02-06 22:07:56 -05:00
|
|
|
#if 0
|
2001-09-17 01:50:02 -04:00
|
|
|
g_log_set_handler (NULL,
|
|
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
|
|
log_handler, NULL);
|
2001-12-09 22:55:26 -05:00
|
|
|
g_log_set_handler ("Gtk",
|
|
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
|
|
log_handler, NULL);
|
|
|
|
g_log_set_handler ("Gdk",
|
|
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
|
|
log_handler, NULL);
|
|
|
|
g_log_set_handler ("GLib",
|
|
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
|
|
log_handler, NULL);
|
|
|
|
g_log_set_handler ("Pango",
|
|
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
|
|
log_handler, NULL);
|
|
|
|
g_log_set_handler ("GLib-GObject",
|
|
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
|
|
log_handler, NULL);
|
|
|
|
g_log_set_handler ("GThread",
|
|
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
|
|
log_handler, NULL);
|
2001-12-09 17:41:12 -05:00
|
|
|
|
2002-02-06 22:07:56 -05:00
|
|
|
#endif
|
2002-05-12 10:50:05 -04:00
|
|
|
|
|
|
|
if (g_getenv ("METACITY_G_FATAL_WARNINGS") != NULL)
|
|
|
|
g_log_set_always_fatal (G_LOG_LEVEL_MASK);
|
2002-02-06 22:07:56 -05:00
|
|
|
|
|
|
|
meta_ui_set_current_theme (meta_prefs_get_theme (), FALSE);
|
|
|
|
|
|
|
|
/* Try some panic stuff, this is lame but we really
|
|
|
|
* don't want users to lose their WM :-/
|
|
|
|
*/
|
|
|
|
if (!meta_ui_have_a_theme ())
|
|
|
|
meta_ui_set_current_theme ("Atlanta", FALSE);
|
|
|
|
|
|
|
|
if (!meta_ui_have_a_theme ())
|
|
|
|
meta_ui_set_current_theme ("Crux", FALSE);
|
|
|
|
|
|
|
|
if (!meta_ui_have_a_theme ())
|
2002-06-16 09:49:37 -04:00
|
|
|
meta_fatal (_("Could not find a theme! Be sure %s exists and contains the usual themes."),
|
2002-02-06 22:07:56 -05:00
|
|
|
METACITY_PKGDATADIR"/themes");
|
2001-12-09 22:55:26 -05:00
|
|
|
|
2001-12-09 17:41:12 -05:00
|
|
|
/* 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 (!disable_sm)
|
2002-03-13 23:36:02 -05:00
|
|
|
meta_session_init (client_id, save_file);
|
2001-05-30 23:30:58 -04:00
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
if (!meta_display_open (NULL))
|
|
|
|
meta_exit (META_EXIT_ERROR);
|
|
|
|
|
|
|
|
g_main_run (meta_main_loop);
|
|
|
|
|
2001-06-22 02:21:44 -04:00
|
|
|
{
|
|
|
|
GSList *displays;
|
|
|
|
GSList *tmp;
|
2001-10-15 00:14:58 -04:00
|
|
|
|
|
|
|
/* we need a copy since closing the display removes it
|
|
|
|
* from the list
|
|
|
|
*/
|
|
|
|
displays = g_slist_copy (meta_displays_list ());
|
2001-06-22 02:21:44 -04:00
|
|
|
tmp = displays;
|
|
|
|
while (tmp != NULL)
|
|
|
|
{
|
|
|
|
meta_display_close (tmp->data);
|
|
|
|
tmp = tmp->next;
|
|
|
|
}
|
2001-10-15 00:14:58 -04:00
|
|
|
g_slist_free (displays);
|
2001-06-22 02:21:44 -04:00
|
|
|
}
|
2001-10-15 00:14:58 -04:00
|
|
|
|
2002-06-08 18:04:59 -04:00
|
|
|
meta_session_shutdown ();
|
|
|
|
|
2001-10-15 00:14:58 -04:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
2001-06-22 02:21:44 -04:00
|
|
|
|
2001-05-30 11:36:31 -04:00
|
|
|
return meta_exit_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
GMainLoop*
|
|
|
|
meta_get_main_loop (void)
|
|
|
|
{
|
|
|
|
return meta_main_loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_quit (MetaExitCode code)
|
|
|
|
{
|
|
|
|
meta_exit_code = code;
|
2001-06-22 02:21:44 -04:00
|
|
|
|
|
|
|
if (g_main_is_running (meta_main_loop))
|
|
|
|
g_main_quit (meta_main_loop);
|
2001-05-30 11:36:31 -04:00
|
|
|
}
|
|
|
|
|
2001-10-15 00:14:58 -04:00
|
|
|
void
|
|
|
|
meta_restart (void)
|
|
|
|
{
|
|
|
|
meta_restart_after_quit = TRUE;
|
|
|
|
meta_quit (META_EXIT_SUCCESS);
|
|
|
|
}
|
2002-02-06 22:07:56 -05:00
|
|
|
|
|
|
|
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);
|
2002-02-06 22:25:34 -05:00
|
|
|
meta_display_retheme_all ();
|
2002-02-06 22:07:56 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* handled elsewhere or otherwise */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|