/* 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. */ #include #include "main.h" #include "util.h" #include "display.h" #include "errors.h" #include "ui.h" #include "session.h" #include "prefs.h" #include #include #ifdef HAVE_GCONF #include #endif #include #include #include #include #include #include #include #include #include #include static MetaExitCode meta_exit_code = META_EXIT_SUCCESS; static GMainLoop *meta_main_loop = NULL; static gboolean meta_restart_after_quit = FALSE; static void prefs_changed_callback (MetaPreference pref, gpointer data); 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 (); } static void usage (void) { g_print (_("metacity [--sm-disable] [--sm-client-id=ID] [--sm-save-file=FILENAME] [--display=DISPLAY] [--replace] [--version]\n")); exit (1); } static void version (void) { g_print (_("metacity %s\n" "Copyright (C) 2001-2002 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); } #define GNOME_ACCESSIBILITY_KEY "/desktop/gnome/interface/accessibility" static char * find_accessibility_module (const char *libname) { char *path; char *fname; char *retval; retval = NULL; fname = g_strconcat (libname, "." G_MODULE_SUFFIX, NULL); path = g_strconcat (METACITY_LIBDIR"/gtk-2.0/modules", G_DIR_SEPARATOR_S, fname, NULL); if (g_file_test (path, G_FILE_TEST_EXISTS)) retval = path; if (path) retval = path; else g_free (path); g_free (fname); return retval; } static gboolean accessibility_invoke_module (const char *libname, gboolean init) { GModule *handle; void (*invoke_fn) (void); const char *method; gboolean retval = FALSE; char *module_name; if (init) method = "gnome_accessibility_module_init"; else method = "gnome_accessibility_module_shutdown"; module_name = find_accessibility_module (libname); if (!module_name) { g_warning ("Accessibility: failed to find module '%s' which " "is needed to make this application accessible", libname); } else if (!(handle = g_module_open (module_name, G_MODULE_BIND_LAZY))) { g_warning ("Accessibility: failed to load module '%s': '%s'", libname, g_module_error ()); } else if (!g_module_symbol (handle, method, (gpointer *)&invoke_fn)) { g_warning ("Accessibility: error library '%s' does not include " "method '%s' required for accessibility support", libname, method); g_module_close (handle); } else { retval = TRUE; invoke_fn (); } g_free (module_name); return retval; } static gboolean accessibility_invoke (gboolean init) { accessibility_invoke_module ("libgail", init); accessibility_invoke_module ("libatk-bridge", init); return TRUE; } int main (int argc, char **argv) { struct sigaction act; sigset_t empty_mask; char *display_name; int i; const char *client_id; gboolean disable_sm; const char *prev_arg; const char *save_file; g_set_prgname (argv[0]); if (setlocale (LC_ALL, "") == NULL) meta_warning ("Locale not understood by C library, internationalization will not work\n"); 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 if (g_getenv ("METACITY_VERBOSE")) meta_set_verbose (TRUE); if (g_getenv ("METACITY_DEBUG")) meta_set_debugging (TRUE); meta_set_syncing (g_getenv ("METACITY_SYNC") != NULL); if (g_get_home_dir ()) chdir (g_get_home_dir ()); { char buf[256]; GDate d; g_date_clear (&d, 1); g_date_set_time (&d, time (NULL)); g_date_strftime (buf, sizeof (buf), "%x", &d); meta_verbose ("Metacity version %s running on %s\n", VERSION, buf); } { const char *charset; g_get_charset (&charset); meta_verbose ("Running in locale \"%s\" with encoding \"%s\"\n", setlocale (LC_ALL, NULL), charset); } bindtextdomain (GETTEXT_PACKAGE, METACITY_LOCALEDIR); bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8"); textdomain (GETTEXT_PACKAGE); /* Parse options lamely */ display_name = NULL; client_id = NULL; save_file = NULL; 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, "--version") == 0) version (); else if (strcmp (arg, "--sm-disable") == 0) disable_sm = TRUE; else if (strcmp (arg, "--replace") == 0) meta_set_replace_current_wm (TRUE); 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 */ 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 */ else if (strstr (arg, "--sm-save-file=") == arg) { const char *file; if (save_file) meta_fatal ("Can't specify SM save file twice\n"); file = strchr (arg, '='); ++file; save_file = g_strdup (file); } else if (prev_arg && strcmp (prev_arg, "--sm-save-file") == 0) { if (save_file) meta_fatal ("Can't specify SM save file twice\n"); save_file = g_strdup (arg); } else if (strcmp (arg, "--sm-save-file") == 0) ; /* wait for next arg */ else usage (); prev_arg = arg; ++i; } if (save_file && 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); if (display_name == NULL && g_getenv ("METACITY_DISPLAY")) { meta_verbose ("Using METACITY_DISPLAY %s\n", g_getenv ("METACITY_DISPLAY")); display_name = g_strconcat ("DISPLAY=", g_getenv ("METACITY_DISPLAY"), NULL); } if (display_name) { putenv (display_name); /* DO NOT FREE display_name, putenv() sucks */ } g_type_init (); #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 #ifdef HAVE_COMPOSITE_EXTENSIONS meta_verbose ("Compiled with composite extensions\n"); #else meta_verbose ("Compiled without composite extensions\n"); #endif /* Load prefs */ meta_prefs_init (); meta_prefs_add_listener (prefs_changed_callback, NULL); meta_ui_init (&argc, &argv); /* must be after UI init so we can override GDK handlers */ meta_errors_init (); #if 1 g_log_set_handler (NULL, G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION, log_handler, NULL); g_log_set_handler (G_LOG_DOMAIN, G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION, log_handler, NULL); 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); #endif if (g_getenv ("METACITY_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 (METACITY_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."), METACITY_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 (!disable_sm) meta_session_init (client_id, save_file); if (!meta_display_open (NULL)) meta_exit (META_EXIT_ERROR); { gboolean do_init_a11y; do_init_a11y = FALSE; #ifdef HAVE_GCONF do_init_a11y = gconf_client_get_bool ( gconf_client_get_default (), GNOME_ACCESSIBILITY_KEY, NULL); #endif if (do_init_a11y) accessibility_invoke (TRUE); } g_main_run (meta_main_loop); { GSList *displays; GSList *tmp; /* we need a copy since closing the display removes it * from the list */ displays = g_slist_copy (meta_displays_list ()); tmp = displays; while (tmp != NULL) { meta_display_close (tmp->data); tmp = tmp->next; } g_slist_free (displays); } meta_session_shutdown (); 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; } GMainLoop* meta_get_main_loop (void) { return meta_main_loop; } void meta_quit (MetaExitCode code) { meta_exit_code = code; if (g_main_is_running (meta_main_loop)) g_main_quit (meta_main_loop); } void meta_restart (void) { meta_restart_after_quit = TRUE; meta_quit (META_EXIT_SUCCESS); } 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; default: /* handled elsewhere or otherwise */ break; } }