diff src/audacious/main.c @ 2313:3149d4b1a9a9 trunk

[svn] - objective-make autodepend fixes - move all sourcecode into src/ and adjust Makefiles accordingly
author nenolod
date Fri, 12 Jan 2007 11:43:40 -0800
parents
children 49d285f6008b
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/audacious/main.c	Fri Jan 12 11:43:40 2007 -0800
@@ -0,0 +1,1286 @@
+/*  Audacious - Cross-platform multimedia player
+ *  Copyright (C) 2005-2007  Audacious development team.
+ *
+ *  Based on BMP:
+ *  Copyright (C) 2003-2004  BMP development team.
+ *
+ *  Based on XMMS:
+ *  Copyright (C) 1998-2003  XMMS development team.
+ *
+ *  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; under version 2 of the License.
+ *
+ *  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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+#  include "config.h"
+#endif
+
+#include "main.h"
+
+#include <glib.h>
+#include <glib/gi18n.h>
+#include <glib/gprintf.h>
+#include <gdk/gdk.h>
+#include <stdlib.h>
+#include <string.h>
+#include <getopt.h>
+#include <ctype.h>
+#include <time.h>
+
+#include <unistd.h>
+#include <errno.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <signal.h>
+
+#include "platform/smartinclude.h"
+
+#include "libaudacious/configdb.h"
+#include "libaudacious/beepctrl.h"
+#include "libaudacious/util.h"
+#include "vfs.h"
+
+#include "controlsocket.h"
+#include "dnd.h"
+#include "effect.h"
+#include "ui_equalizer.h"
+#include "general.h"
+#include "genevent.h"
+#include "hints.h"
+#include "input.h"
+#include "logger.h"
+#include "ui_main.h"
+#include "ui_manager.h"
+#include "output.h"
+#include "playback.h"
+#include "playlist.h"
+#include "ui_playlist.h"
+#include "ui_preferences.h"
+#include "pluginenum.h"
+#include "ui_skinselector.h"
+#include "util.h"
+#include "visualization.h"
+#include "build_stamp.h"
+#include "ui_fileinfo.h"
+#include "signals.h"
+
+#include "icons-csource.h"
+#include "icons-stock.h"
+#include "images/audacious_player.xpm"
+
+gboolean has_x11_connection = FALSE; 	/* do we have an X11 connection? */
+
+/* Translatable string for beep.desktop's comment field */
+const gchar *desktop_comment = N_("Audacious");
+
+const gchar *application_name = N_("Audacious");
+
+
+struct _BmpCmdLineOpt {
+    GList *filenames;
+    gint session;
+    gboolean play, stop, pause, fwd, rew, play_pause, playcd, show_jump_box;
+    gboolean enqueue, mainwin, remote, activate;
+    gboolean load_skins;
+    gboolean headless;
+    gboolean no_log;
+    gboolean enqueue_to_temp;
+    gchar *previous_session_id;
+};
+
+typedef struct _BmpCmdLineOpt BmpCmdLineOpt;
+
+BmpCmdLineOpt options;
+
+BmpConfig cfg;
+
+BmpConfig bmp_default_config = {
+    MAINWIN_DEFAULT_POS_X,      /* mainwin x position */
+    MAINWIN_DEFAULT_POS_Y,      /* mainwin y position */
+    EQUALIZER_DEFAULT_POS_X,    /* equalizer x position */
+    EQUALIZER_DEFAULT_POS_Y,    /* equalizer y position */
+    PLAYLISTWIN_DEFAULT_POS_X,  /* playlistwin x position */
+    PLAYLISTWIN_DEFAULT_POS_Y,  /* playlistwin y position */
+    PLAYLISTWIN_DEFAULT_WIDTH,  /* playlistwin width */
+    PLAYLISTWIN_DEFAULT_HEIGHT, /* playlistwin height */
+    10,                         /* snap distance */
+    FALSE,                      /* real-time priority */
+    FALSE, FALSE,               /* shuffle, repeat */
+    FALSE,                      /* UNUSED (double size) */
+    TRUE,                       /* autoscroll */
+    TRUE,                       /* analyzer peaks */
+    FALSE,                      /* equalizer autoload */
+    FALSE,                      /* easy move */
+    FALSE,                      /* equalizer active */
+    FALSE,                      /* playlistwin visible */
+    FALSE,                      /* equalizer visible */
+    TRUE,                       /* player visible */
+    FALSE,                      /* player shaded */
+    FALSE,                      /* playlistwin shaded */
+    FALSE,                      /* equalizer shaded */
+    FALSE,                      /* allow multiple instances */
+    TRUE,                       /* always show cb */
+    TRUE, TRUE, TRUE,           /* convert '_', %20 and '\' */
+    TRUE,                       /* show numbers in playlist */
+    TRUE,                       /* snap windows */
+    TRUE,                       /* save window positions */
+    TRUE,                       /* dim titlebar */
+    FALSE,                      /* get playlist info on load */
+    TRUE,                       /* get playlist info on demand */
+    TRUE,                       /* UNUSED (equalizer doublesize linked) */
+    FALSE,                      /* sort jump to file */
+    FALSE,                      /* use effect plugins */
+    FALSE,                      /* always on top */
+    FALSE,                      /* sticky */
+    FALSE,                      /* no playlist advance */
+    FALSE,                      /* stop after current song */
+    TRUE,                       /* refresh file list */
+    TRUE,                       /* UNUSED (smooth title scrolling) */
+    TRUE,                       /* use playlist metadata */
+    TRUE,                       /* warn about unplayables */
+    FALSE,                      /* use \ as directory delimiter */
+    FALSE,                      /* random skin on play */
+    FALSE,                      /* use fontsets */
+    FALSE,                      /* use X font for mainwin */
+    TRUE,			/* use custom cursors */
+    TRUE,			/* close dialog on open */
+    TRUE,			/* close dialog on add */
+    0.0,                        /* equalizer preamp */
+    {0, 0, 0, 0, 0,             /* equalizer bands */
+     0, 0, 0, 0, 0},
+    NULL,                       /* skin */
+    NULL,                       /* output plugin */
+    NULL,                       /* file selector path */
+    NULL,                       /* playlist path */
+    NULL,                       /* playlist font */
+    NULL,                       /* mainwin font */
+    NULL,                       /* disabled input plugins */
+    NULL,                       /* enabled general plugins */
+    NULL,                       /* enabled visualization plugins */
+    NULL,                       /* enabled effect plugins */
+    NULL,                       /* equalizer preset default file */
+    NULL,                       /* equalizer preset extension */
+    NULL,                       /* URL history */
+    0,                          /* timer mode */
+    VIS_ANALYZER,               /* visualizer type */
+    ANALYZER_NORMAL,            /* analyzer mode */
+    ANALYZER_BARS,              /* analyzer type */
+    SCOPE_DOT,                  /* scope mode */
+    VOICEPRINT_NORMAL,          /* voiceprint mode */
+    VU_SMOOTH,                  /* VU mode */
+    REFRESH_FULL,               /* visualizer refresh rate */
+    FALLOFF_FAST,               /* analyzer fall off rate */
+    FALLOFF_SLOW,               /* peaks fall off rate */
+    0,                          /* playlist position */
+    2,                          /* pause between songs time */
+    FALSE,                      /* pause between songs */
+    FALSE,                      /* show window decorations */
+    8,                          /* mouse wheel scroll step */
+    FALSE,                      /* playlist transparent */
+    2,                          /* 3rd preset (ARTIST - ALBUM - TITLE) */
+    NULL,                       /* title format */
+    FALSE,                      /* software volume control enabled */
+    TRUE,                       /* UNUSED (XMMS compatibility mode) */
+    TRUE,                       /* extra eq filtering */
+    3,                          /* scroll pl by */
+    FALSE,                      /* resume playback on startup */
+    -1,                         /* resume playback on startup time */
+    TRUE,			/* show seperators in pl */
+    NULL,
+    NULL,
+    3000,			/* audio buffer size */
+    FALSE,			/* whether or not to postpone format detection on initial add */
+    TRUE,			/* show filepopup for tuple */
+    NULL,			/* words identifying covers */
+    NULL,			/* words that might not show up in cover names */
+    FALSE,
+    0,
+    NULL,			/* default session uri base (non-NULL = custom session uri base) */
+    150,			/* short side length of the picture in the filepopup */
+    20,				/* delay until the filepopup comes up */
+    FALSE,			/* use filename.jpg for coverart */
+    FALSE,			/* use XMMS-style file selection */
+    TRUE,                       /* use extension probing         */
+    255, 255, 255,		/* colorize r, g, b */
+    FALSE,			/* internal: whether or not to terminate */
+};
+
+typedef struct bmp_cfg_boolent_t {
+    char const *be_vname;
+    gboolean *be_vloc;
+    gboolean be_wrt;
+} bmp_cfg_boolent;
+
+typedef struct bmp_cfg_nument_t {
+    char const *ie_vname;
+    gint *ie_vloc;
+    gboolean ie_wrt;
+} bmp_cfg_nument;
+
+typedef struct bmp_cfg_strent_t {
+    char const *se_vname;
+    char **se_vloc;
+    gboolean se_wrt;
+} bmp_cfg_strent;
+
+const gchar *bmp_titlestring_presets[] = {
+    "%t",
+    "%{p:%p - %}%t",
+    "%{p:%p - %}%{a:%a - %}%t",
+    "%{p:%p - %}%{a:%a - %}%{n:%n. %}%t",
+    "%{p:%p %}%{a:[ %a ] %}%{p:- %}%{n:%n. %}%{t:%t%}",
+    "%{a:%a - %}%t"
+};
+
+const guint n_titlestring_presets = G_N_ELEMENTS(bmp_titlestring_presets);
+
+const gchar *chardet_detector_presets[] = {
+	"None",
+	"Japanese",
+	"Taiwanese",
+	"Chinese",
+	"Korean",
+	"Russian",
+#ifdef HAVE_UDET
+	"Universal"
+#endif
+};
+
+const guint n_chardet_detector_presets = G_N_ELEMENTS(chardet_detector_presets);	
+
+static bmp_cfg_boolent bmp_boolents[] = {
+    {"allow_multiple_instances", &cfg.allow_multiple_instances, TRUE},
+    {"use_realtime", &cfg.use_realtime, TRUE},
+    {"always_show_cb", &cfg.always_show_cb, TRUE},
+    {"convert_underscore", &cfg.convert_underscore, TRUE},
+    {"convert_twenty", &cfg.convert_twenty, TRUE},
+    {"convert_slash", &cfg.convert_slash, TRUE },
+    {"show_numbers_in_pl", &cfg.show_numbers_in_pl, TRUE},
+    {"show_separator_in_pl", &cfg.show_separator_in_pl, TRUE},
+    {"snap_windows", &cfg.snap_windows, TRUE},
+    {"save_window_positions", &cfg.save_window_position, TRUE},
+    {"dim_titlebar", &cfg.dim_titlebar, TRUE},
+    {"get_info_on_load", &cfg.get_info_on_load, TRUE},
+    {"get_info_on_demand", &cfg.get_info_on_demand, TRUE},
+    {"eq_doublesize_linked", &cfg.eq_doublesize_linked, TRUE},
+    {"no_playlist_advance", &cfg.no_playlist_advance, TRUE},
+    {"refresh_file_list", &cfg.refresh_file_list, TRUE},
+    {"sort_jump_to_file", &cfg.sort_jump_to_file, TRUE},
+    {"use_pl_metadata", &cfg.use_pl_metadata, TRUE},
+    {"warn_about_unplayables", &cfg.warn_about_unplayables, TRUE},
+    {"use_backslash_as_dir_delimiter", &cfg.use_backslash_as_dir_delimiter, TRUE},
+    {"player_shaded", &cfg.player_shaded, TRUE},
+    {"player_visible", &cfg.player_visible, TRUE},
+    {"shuffle", &cfg.shuffle, TRUE},
+    {"repeat", &cfg.repeat, TRUE},
+    {"doublesize", &cfg.doublesize, TRUE},
+    {"autoscroll_songname", &cfg.autoscroll, TRUE},
+    {"stop_after_current_song", &cfg.stopaftersong, TRUE},
+    {"playlist_shaded", &cfg.playlist_shaded, TRUE},
+    {"playlist_visible", &cfg.playlist_visible, TRUE},
+    {"playlist_transparent", &cfg.playlist_transparent, TRUE},
+    {"use_fontsets", &cfg.use_fontsets, TRUE},
+    {"mainwin_use_xfont", &cfg.mainwin_use_xfont, FALSE},
+    {"equalizer_visible", &cfg.equalizer_visible, TRUE},
+    {"equalizer_active", &cfg.equalizer_active, TRUE},
+    {"equalizer_shaded", &cfg.equalizer_shaded, TRUE},
+    {"equalizer_autoload", &cfg.equalizer_autoload, TRUE},
+    {"easy_move", &cfg.easy_move, TRUE},
+    {"use_eplugins", &cfg.use_eplugins, TRUE},
+    {"always_on_top", &cfg.always_on_top, TRUE},
+    {"sticky", &cfg.sticky, TRUE},
+    {"random_skin_on_play", &cfg.random_skin_on_play, TRUE},
+    {"pause_between_songs", &cfg.pause_between_songs, TRUE},
+    {"show_wm_decorations", &cfg.show_wm_decorations, TRUE},
+    {"eq_extra_filtering", &cfg.eq_extra_filtering, TRUE},
+    {"analyzer_peaks", &cfg.analyzer_peaks, TRUE},
+    {"custom_cursors", &cfg.custom_cursors, TRUE},
+    {"close_dialog_open", &cfg.close_dialog_open, TRUE},
+    {"close_dialog_add", &cfg.close_dialog_add, TRUE},
+    {"resume_playback_on_startup", &cfg.resume_playback_on_startup, TRUE},
+    {"playlist_detect", &cfg.playlist_detect, TRUE},
+    {"show_filepopup_for_tuple", &cfg.show_filepopup_for_tuple, TRUE},
+    {"recurse_for_cover", &cfg.recurse_for_cover, TRUE},
+    {"use_file_cover", &cfg.use_file_cover, TRUE},
+    {"use_xmms_style_fileselector", &cfg.use_xmms_style_fileselector, TRUE},
+    {"use_extension_probing", &cfg.use_extension_probing, TRUE},
+};
+
+static gint ncfgbent = G_N_ELEMENTS(bmp_boolents);
+
+static bmp_cfg_nument bmp_numents[] = {
+    {"player_x", &cfg.player_x, TRUE},
+    {"player_y", &cfg.player_y, TRUE},
+    {"timer_mode", &cfg.timer_mode, TRUE},
+    {"vis_type", &cfg.vis_type, TRUE},
+    {"analyzer_mode", &cfg.analyzer_mode, TRUE},
+    {"analyzer_type", &cfg.analyzer_type, TRUE},
+    {"scope_mode", &cfg.scope_mode, TRUE},
+    {"vu_mode", &cfg.vu_mode, TRUE},
+    {"voiceprint_mode", &cfg.voiceprint_mode, TRUE},
+    {"vis_refresh_rate", &cfg.vis_refresh, TRUE},
+    {"analyzer_falloff", &cfg.analyzer_falloff, TRUE},
+    {"peaks_falloff", &cfg.peaks_falloff, TRUE},
+    {"playlist_x", &cfg.playlist_x, TRUE},
+    {"playlist_y", &cfg.playlist_y, TRUE},
+    {"playlist_width", &cfg.playlist_width, TRUE},
+    {"playlist_height", &cfg.playlist_height, TRUE},
+    {"playlist_position", &cfg.playlist_position, TRUE},
+    {"equalizer_x", &cfg.equalizer_x, TRUE},
+    {"equalizer_y", &cfg.equalizer_y, TRUE},
+    {"snap_distance", &cfg.snap_distance, TRUE},
+    {"pause_between_songs_time", &cfg.pause_between_songs_time, TRUE},
+    {"mouse_wheel_change", &cfg.mouse_change, TRUE},
+    {"scroll_pl_by", &cfg.scroll_pl_by, TRUE},
+    {"titlestring_preset", &cfg.titlestring_preset, TRUE},
+    {"resume_playback_on_startup_time", &cfg.resume_playback_on_startup_time, TRUE},
+    {"output_buffer_size", &cfg.output_buffer_size, TRUE},
+    {"recurse_for_cover_depth", &cfg.recurse_for_cover_depth, TRUE},
+    {"filepopup_pixelsize", &cfg.filepopup_pixelsize, TRUE},
+    {"filepopup_delay", &cfg.filepopup_delay, TRUE},
+    {"colorize_r", &cfg.colorize_r, TRUE},
+    {"colorize_g", &cfg.colorize_g, TRUE},
+    {"colorize_b", &cfg.colorize_b, TRUE},
+};
+
+static gint ncfgient = G_N_ELEMENTS(bmp_numents);
+
+static bmp_cfg_strent bmp_strents[] = {
+    {"playlist_font", &cfg.playlist_font, TRUE},
+    {"mainwin_font", &cfg.mainwin_font, TRUE},
+    {"eqpreset_default_file", &cfg.eqpreset_default_file, TRUE},
+    {"eqpreset_extension", &cfg.eqpreset_extension, TRUE},
+    {"skin", &cfg.skin, FALSE},
+    {"output_plugin", &cfg.outputplugin, FALSE},
+    {"disabled_iplugins", &cfg.disabled_iplugins, TRUE},
+    {"enabled_gplugins", &cfg.enabled_gplugins, FALSE},
+    {"enabled_vplugins", &cfg.enabled_vplugins, FALSE},
+    {"enabled_eplugins", &cfg.enabled_eplugins, FALSE},
+    {"filesel_path", &cfg.filesel_path, FALSE},
+    {"playlist_path", &cfg.playlist_path, FALSE},
+    {"generic_title_format", &cfg.gentitle_format, TRUE},
+    {"chardet_detector", &cfg.chardet_detector, TRUE},
+    {"chardet_fallback", &cfg.chardet_fallback, TRUE},
+    {"cover_name_include", &cfg.cover_name_include, TRUE},
+    {"cover_name_exclude", &cfg.cover_name_exclude, TRUE},
+    {"session_uri_base", &cfg.session_uri_base, TRUE}
+};
+
+static gint ncfgsent = G_N_ELEMENTS(bmp_strents);
+
+gchar *bmp_paths[BMP_PATH_COUNT] = {};
+
+GList *dock_window_list = NULL;
+
+gboolean pposition_broken = FALSE;
+
+gboolean starting_up = TRUE;
+
+/* XXX: case-sensitivity is bad for lazy nenolods. :( -nenolod */
+static gchar *pl_candidates[] = {
+	PLUGIN_FILENAME("ALSA"),
+	PLUGIN_FILENAME("coreaudio"),
+	PLUGIN_FILENAME("OSS"),
+	PLUGIN_FILENAME("sun"),
+	PLUGIN_FILENAME("ESD"),
+	PLUGIN_FILENAME("pulse_audio"),
+	PLUGIN_FILENAME("disk_writer"),
+	NULL
+};
+
+GCond *cond_scan;
+GMutex *mutex_scan;
+
+static GSList *
+get_feature_list(void)
+{
+    GSList *features = NULL;
+    
+#ifdef HAVE_GCONF
+    features = g_slist_append(features, "GConf");
+#endif
+
+    return features;
+}
+
+static void
+dump_version(void)
+{
+    GSList *features;
+
+    g_printf("%s %s [%s]", _(application_name), VERSION, svn_stamp);
+
+    features = get_feature_list();
+
+    if (features) {
+        GSList *item;
+
+        g_printf(" (");
+
+        for (item = features; g_slist_next(item); item = g_slist_next(item))
+            g_printf("%s, ", (const gchar *) item->data);
+
+        g_printf("%s)", (const gchar *) item->data);
+
+        g_slist_free(features);
+    }
+
+    g_printf("\n");
+}
+
+const gchar *
+xmms_get_gentitle_format(void)
+{
+    guint titlestring_preset = cfg.titlestring_preset;
+
+    if (titlestring_preset < n_titlestring_presets)
+	return bmp_titlestring_presets[titlestring_preset];
+
+    return cfg.gentitle_format;
+}
+
+void
+make_directory(const gchar * path, mode_t mode)
+{
+    if (mkdir(path, mode) == 0)
+        return;
+
+    if (errno == EEXIST)
+        return;
+
+    g_printerr(_("Could not create directory (%s): %s"), path,
+               g_strerror(errno));
+}
+
+static void
+bmp_make_user_dir(void)
+{
+    const mode_t mode755 = S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;
+
+    make_directory(bmp_paths[BMP_PATH_USER_DIR], mode755);
+    make_directory(bmp_paths[BMP_PATH_USER_PLUGIN_DIR], mode755);
+    make_directory(bmp_paths[BMP_PATH_USER_SKIN_DIR], mode755);
+    make_directory(bmp_paths[BMP_PATH_SKIN_THUMB_DIR], mode755);
+}
+
+static void
+bmp_free_paths(void)
+{
+    int i;
+
+    for (i = 0; i < BMP_PATH_COUNT; i++)
+    {
+        g_free(bmp_paths[i]);
+        bmp_paths[i] = 0;
+    }
+}
+
+
+#define USER_PATH(path) \
+    g_build_filename(bmp_paths[BMP_PATH_USER_DIR], path, NULL);
+
+static void
+bmp_init_paths(void)
+{
+    bmp_paths[BMP_PATH_USER_DIR] = g_build_filename(g_get_home_dir(), BMP_RCPATH, NULL);
+
+    bmp_paths[BMP_PATH_USER_PLUGIN_DIR] = USER_PATH(BMP_USER_PLUGIN_DIR_BASENAME);
+    bmp_paths[BMP_PATH_USER_SKIN_DIR] = USER_PATH(BMP_SKIN_DIR_BASENAME);
+    bmp_paths[BMP_PATH_SKIN_THUMB_DIR] = USER_PATH(BMP_SKIN_THUMB_DIR_BASENAME);
+    bmp_paths[BMP_PATH_CONFIG_FILE] = USER_PATH(BMP_CONFIG_BASENAME);
+    bmp_paths[BMP_PATH_PLAYLIST_FILE] = USER_PATH(BMP_PLAYLIST_BASENAME);
+    bmp_paths[BMP_PATH_ACCEL_FILE] = USER_PATH(BMP_ACCEL_BASENAME);
+    bmp_paths[BMP_PATH_LOG_FILE] = USER_PATH(BMP_LOG_BASENAME);
+
+    g_atexit(bmp_free_paths);
+}
+
+
+void
+bmp_config_load(void)
+{
+    ConfigDb *db;
+    gint i, length;
+
+    memcpy(&cfg, &bmp_default_config, sizeof(BmpConfig));
+
+    db = bmp_cfg_db_open();
+    for (i = 0; i < ncfgbent; ++i) {
+        bmp_cfg_db_get_bool(db, NULL,
+                            bmp_boolents[i].be_vname,
+                            bmp_boolents[i].be_vloc);
+    }
+
+    for (i = 0; i < ncfgient; ++i) {
+        bmp_cfg_db_get_int(db, NULL,
+                           bmp_numents[i].ie_vname,
+                           bmp_numents[i].ie_vloc);
+    }
+
+    for (i = 0; i < ncfgsent; ++i) {
+        bmp_cfg_db_get_string(db, NULL,
+                              bmp_strents[i].se_vname,
+                              bmp_strents[i].se_vloc);
+    }
+
+    /* Preset */
+    bmp_cfg_db_get_float(db, NULL, "equalizer_preamp", &cfg.equalizer_preamp);
+    for (i = 0; i < 10; i++) {
+        gchar eqtext[18];
+
+        g_snprintf(eqtext, sizeof(eqtext), "equalizer_band%d", i);
+        bmp_cfg_db_get_float(db, NULL, eqtext, &cfg.equalizer_bands[i]);
+    }
+
+    /* History */
+    if (bmp_cfg_db_get_int(db, NULL, "url_history_length", &length)) {
+        for (i = 1; i <= length; i++) {
+            gchar str[19], *tmp;
+
+            g_snprintf(str, sizeof(str), "url_history%d", i);
+            if (bmp_cfg_db_get_string(db, NULL, str, &tmp))
+                cfg.url_history = g_list_append(cfg.url_history, tmp);
+        }
+    }
+
+    bmp_cfg_db_close(db);
+
+
+    if (cfg.playlist_font && strlen(cfg.playlist_font) == 0) {
+        g_free(cfg.playlist_font);
+        cfg.playlist_font = NULL;
+    }
+
+    if (cfg.mainwin_font && strlen(cfg.mainwin_font) == 0) {
+        g_free(cfg.mainwin_font);
+        cfg.mainwin_font = NULL;
+    }
+
+    if (!cfg.playlist_font)
+        cfg.playlist_font = g_strdup(PLAYLISTWIN_DEFAULT_FONT);
+
+    if (!cfg.mainwin_font)
+        cfg.mainwin_font = g_strdup(MAINWIN_DEFAULT_FONT);
+
+    if (!cfg.gentitle_format)
+        cfg.gentitle_format = g_strdup("%{p:%p - %}%{a:%a - %}%t");
+
+    if (!cfg.outputplugin) {
+	gint iter;
+        gchar *pl_path = g_build_filename(PLUGIN_DIR, plugin_dir_list[0], NULL);
+
+        for (iter = 0; pl_candidates[iter] != NULL && cfg.outputplugin == NULL; iter++)
+	{
+	     cfg.outputplugin = find_file_recursively(pl_path, pl_candidates[iter]);
+	}
+
+        g_free(pl_path);
+    }
+
+    if (!cfg.eqpreset_default_file)
+        cfg.eqpreset_default_file = g_strdup(EQUALIZER_DEFAULT_DIR_PRESET);
+
+    if (!cfg.eqpreset_extension)
+        cfg.eqpreset_extension = g_strdup(EQUALIZER_DEFAULT_PRESET_EXT);
+
+    if (!cfg.chardet_detector)
+        cfg.chardet_detector = g_strdup("");
+
+    if (!cfg.chardet_fallback)
+        cfg.chardet_fallback = g_strdup("");
+
+    if (!cfg.cover_name_include)
+	    cfg.cover_name_include = g_strdup("");
+
+    if (!cfg.cover_name_exclude)
+	    cfg.cover_name_exclude = g_strdup("back");
+
+    if (!cfg.session_uri_base)
+        cfg.session_uri_base = g_strdup("");
+
+    /* at least one of these should be true */
+    if ((!cfg.get_info_on_demand) && (!cfg.get_info_on_load))
+        cfg.get_info_on_demand = TRUE;
+}
+
+
+void
+bmp_config_save(void)
+{
+    GList *node;
+    gchar *str;
+    gint i, cur_pb_time;
+    ConfigDb *db;
+    Playlist *playlist = playlist_get_active();
+
+    cfg.disabled_iplugins = input_stringify_disabled_list();
+
+
+    db = bmp_cfg_db_open();
+
+    for (i = 0; i < ncfgbent; ++i)
+        if (bmp_boolents[i].be_wrt)
+            bmp_cfg_db_set_bool(db, NULL,
+                                bmp_boolents[i].be_vname,
+                                *bmp_boolents[i].be_vloc);
+
+    for (i = 0; i < ncfgient; ++i)
+        if (bmp_numents[i].ie_wrt)
+            bmp_cfg_db_set_int(db, NULL,
+                               bmp_numents[i].ie_vname,
+                               *bmp_numents[i].ie_vloc);
+
+    /* This is a bit lame .. it'll end up being written twice,
+     * could do with being done a bit neater.  -larne   */
+    bmp_cfg_db_set_int(db, NULL, "playlist_position",
+                       playlist_get_position(playlist));
+
+    bmp_cfg_db_set_bool(db, NULL, "mainwin_use_xfont",
+			cfg.mainwin_use_xfont);
+
+    for (i = 0; i < ncfgsent; ++i) {
+        if (bmp_strents[i].se_wrt)
+            bmp_cfg_db_set_string(db, NULL,
+                                  bmp_strents[i].se_vname,
+                                  *bmp_strents[i].se_vloc);
+    }
+
+    bmp_cfg_db_set_float(db, NULL, "equalizer_preamp", cfg.equalizer_preamp);
+
+    for (i = 0; i < 10; i++) {
+        str = g_strdup_printf("equalizer_band%d", i);
+        bmp_cfg_db_set_float(db, NULL, str, cfg.equalizer_bands[i]);
+        g_free(str);
+    }
+
+    if (bmp_active_skin != NULL)
+    {
+        if (bmp_active_skin->path)
+            bmp_cfg_db_set_string(db, NULL, "skin", bmp_active_skin->path);
+        else
+            bmp_cfg_db_unset_key(db, NULL, "skin");
+    }
+
+    if (get_current_output_plugin())
+        bmp_cfg_db_set_string(db, NULL, "output_plugin",
+                              get_current_output_plugin()->filename);
+    else
+        bmp_cfg_db_unset_key(db, NULL, "output_plugin");
+
+    str = general_stringify_enabled_list();
+    if (str) {
+        bmp_cfg_db_set_string(db, NULL, "enabled_gplugins", str);
+        g_free(str);
+    }
+    else
+        bmp_cfg_db_unset_key(db, NULL, "enabled_gplugins");
+
+    str = vis_stringify_enabled_list();
+    if (str) {
+        bmp_cfg_db_set_string(db, NULL, "enabled_vplugins", str);
+        g_free(str);
+    }
+    else
+        bmp_cfg_db_unset_key(db, NULL, "enabled_vplugins");
+
+    str = effect_stringify_enabled_list();
+    if (str) {
+        bmp_cfg_db_set_string(db, NULL, "enabled_eplugins", str);
+        g_free(str);
+    }
+    else
+        bmp_cfg_db_unset_key(db, NULL, "enabled_eplugins");
+
+    if (cfg.filesel_path)
+        bmp_cfg_db_set_string(db, NULL, "filesel_path", cfg.filesel_path);
+
+    if (cfg.playlist_path)
+        bmp_cfg_db_set_string(db, NULL, "playlist_path", cfg.playlist_path);
+
+    bmp_cfg_db_set_int(db, NULL, "url_history_length",
+                       g_list_length(cfg.url_history));
+
+    for (node = cfg.url_history, i = 1; node; node = g_list_next(node), i++) {
+        str = g_strdup_printf("url_history%d", i);
+        bmp_cfg_db_set_string(db, NULL, str, node->data);
+        g_free(str);
+    }
+
+    if (playback_get_playing()) {
+	    cur_pb_time = playback_get_time();
+    } else
+	    cur_pb_time = -1;
+    cfg.resume_playback_on_startup_time = cur_pb_time;
+    bmp_cfg_db_set_int(db, NULL, "resume_playback_on_startup_time",
+		       cfg.resume_playback_on_startup_time);
+
+    bmp_cfg_db_close(db);
+
+    playlist_save(playlist, bmp_paths[BMP_PATH_PLAYLIST_FILE]);
+}
+
+static void
+bmp_set_default_icon(void)
+{
+    GdkPixbuf *icon;
+
+    icon = gdk_pixbuf_new_from_xpm_data((const gchar **) audacious_player_xpm);
+    gtk_window_set_default_icon(icon);
+    g_object_unref(icon);
+}
+
+static void
+register_aud_stock_icons(void)
+{
+  GtkIconFactory *iconfactory = gtk_icon_factory_new();
+  GtkIconSet *iconset;
+  GdkPixbuf *pixbuf;
+
+  /* pick images in icons-csource.h */
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , removedups_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_REMOVEDUPS , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , removeunavail_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_REMOVEUNAVAIL , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , randomizepl_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_RANDOMIZEPL , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , sortbytitle_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_SORTBYTITLE , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , sortbyfilename_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_SORTBYFILENAME , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , sortbyartist_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_SORTBYARTIST , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , sortbypathfile_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_SORTBYPATHFILE , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , selectnone_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_SELECTNONE , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , selectall_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_SELECTALL , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , selectinvert_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_SELECTINVERT , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , invertpl_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_INVERTPL , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , queuetoggle_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_QUEUETOGGLE , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , info_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_INFO , iconset );
+  pixbuf = gdk_pixbuf_new_from_inline( -1 , playlist_pixbuf , FALSE , NULL );
+   iconset = gtk_icon_set_new_from_pixbuf( pixbuf ); g_object_unref( pixbuf );
+   gtk_icon_factory_add( iconfactory , AUD_STOCK_PLAYLIST , iconset );
+
+  gtk_icon_factory_add_default( iconfactory );
+  g_object_unref( iconfactory );
+}
+
+static void
+display_usage(void)
+{
+    g_print(_("Usage: audacious [options] [files] ...\n\n"
+              "Options:\n"
+              "--------\n"));
+    g_print("\n-h, --help             ");
+    /* -h, --help switch */
+    g_print(_("Display this text and exit"));
+    g_print("\n-n, --session          ");
+    /* -n, --session switch */
+    g_print(_("Select Audacious/BMP/XMMS session (Default: 0)"));
+    g_print("\n-r, --rew              ");
+    /* -r, --rew switch */
+    g_print(_("Skip backwards in playlist"));
+    g_print("\n-p, --play             ");
+    /* -p, --play switch */
+    g_print(_("Start playing current playlist"));
+    g_print("\n-u, --pause            ");
+    /* -u, --pause switch */
+    g_print(_("Pause current song"));
+    g_print("\n-s, --stop             ");
+    /* -s, --stop switch */
+    g_print(_("Stop current song"));
+    g_print("\n-t, --play-pause       ");
+    /* -t, --play-pause switch */
+    g_print(_("Pause if playing, play otherwise"));
+    g_print("\n-f, --fwd              ");
+    /* -f, --fwd switch */
+    g_print(_("Skip forward in playlist"));
+    g_print("\n-j, --show-jump-box    ");
+    /* -j, --show-jump-box switch */
+    g_print(_("Display Jump to file dialog"));
+    g_print("\n-e, --enqueue          ");
+    /* -e, --enqueue switch */
+    g_print(_("Don't clear the playlist"));
+    g_print("\n-m, --show-main-window ");
+    /* -m, --show-main-window switch */
+    g_print(_("Show the main window"));
+    g_print("\n-a, --activate         ");
+    /* -a, --activate switch */
+    g_print(_("Activate Audacious"));
+    g_print("\n-i, --sm-client-id     ");
+    /* -i, --sm-client-id switch */
+    g_print(_("Previous session ID"));
+    g_print("\n-H, --headless         ");
+    /* -h, --headless switch */
+    g_print(_("Headless operation [experimental]"));
+    g_print("\n-N, --no-log           ");
+    /* -N, --no-log switch */
+    g_print(_("Disable error/warning interception (logging)"));
+    g_print("\n-v, --version          ");
+    /* -v, --version switch */
+    g_print(_("Print version number and exit\n"));
+
+    exit(EXIT_SUCCESS);
+}
+
+static void
+parse_cmd_line(gint argc,
+               gchar ** argv,
+               BmpCmdLineOpt * options)
+{
+    static struct option long_options[] = {
+        {"help", 0, NULL, 'h'},
+        {"session", 1, NULL, 'n'},
+        {"rew", 0, NULL, 'r'},
+        {"play", 0, NULL, 'p'},
+        {"pause", 0, NULL, 'u'},
+        {"play-pause", 0, NULL, 't'},
+        {"stop", 0, NULL, 's'},
+        {"fwd", 0, NULL, 'f'},
+        {"show-jump-box", 0, NULL, 'j'},
+        {"enqueue", 0, NULL, 'e'},
+	{"enqueue-to-temp", 0, NULL, 'E'},
+        {"show-main-window", 0, NULL, 'm'},
+        {"activate", 0, NULL, 'a'},
+        {"version", 0, NULL, 'v'},
+        {"sm-client-id", 1, NULL, 'i'},
+        {"xmms", 0, NULL, 'x'},
+        {"headless", 0, NULL, 'H'},
+        {"no-log", 0, NULL, 'N'},
+        {0, 0, 0, 0}
+    };
+
+    gchar *filename, *current_dir;
+    gint c, i;
+
+    memset(options, 0, sizeof(BmpCmdLineOpt));
+    options->session = -1;
+
+    while ((c = getopt_long(argc, argv, "chn:HrpusfemavtLSjE", long_options,
+                            NULL)) != -1) {
+        switch (c) {
+        case 'h':
+            display_usage();
+            break;
+        case 'n':
+            options->session = atoi(optarg);
+            break;
+        case 'H':
+            options->headless = TRUE;
+            break;
+        case 'r':
+            options->rew = TRUE;
+            break;
+        case 'p':
+            options->play = TRUE;
+            break;
+        case 'u':
+            options->pause = TRUE;
+            break;
+        case 's':
+            options->stop = TRUE;
+            break;
+        case 'f':
+            options->fwd = TRUE;
+            break;
+        case 't':
+            options->play_pause = TRUE;
+            break;
+        case 'j':
+            options->show_jump_box = TRUE;
+            break;
+        case 'm':
+            options->mainwin = TRUE;
+            break;
+        case 'a':
+            options->activate = TRUE;
+	    break;
+	case 'E':
+	    options->enqueue_to_temp = TRUE;
+	    break;
+        case 'e':
+            options->enqueue = TRUE;
+            break;
+        case 'v':
+            dump_version();
+            exit(EXIT_SUCCESS);
+            break;
+        case 'i':
+            options->previous_session_id = g_strdup(optarg);
+            break;
+        case 'c':
+            options->playcd = TRUE;
+            break;
+        case 'S':
+            options->load_skins = TRUE;
+            break;
+	case 'N':
+	    options->no_log = TRUE;
+            break;
+        }
+    }
+
+    current_dir = g_get_current_dir();
+
+    for (i = optind; i < argc; i++) {
+        if (argv[i][0] == '/' || strstr(argv[i], "://"))
+            filename = g_strdup(argv[i]);
+        else
+            filename = g_build_filename(current_dir, argv[i], NULL);
+
+        options->filenames = g_list_prepend(options->filenames, filename);
+    }
+
+    options->filenames = g_list_reverse(options->filenames);
+
+    g_free(current_dir);
+}
+
+static void
+handle_cmd_line_options(BmpCmdLineOpt * options,
+                        gboolean remote)
+{
+    GList *filenames = options->filenames;
+    gint session = options->session;
+
+    if (session == -1) {
+        if (!remote)
+            session = ctrlsocket_get_session_id();
+        else
+            session = 0;
+    }
+
+    if (filenames) {
+        gint pos = 0;
+
+        if (options->load_skins) {
+            xmms_remote_set_skin(session, filenames->data);
+            skin_install_skin(filenames->data);
+        }
+        else {
+	    if (options->enqueue_to_temp)
+                xmms_remote_playlist_enqueue_to_temp(session, filenames->data);
+
+            if (options->enqueue && options->play)
+                pos = xmms_remote_get_playlist_length(session);
+
+            if (!options->enqueue)
+                xmms_remote_playlist_clear(session);
+
+            xmms_remote_playlist_add(session, filenames);
+
+            if (options->enqueue && options->play &&
+                xmms_remote_get_playlist_length(session) > pos)
+                xmms_remote_set_playlist_pos(session, pos);
+
+            if (!options->enqueue)
+                xmms_remote_play(session);
+        }
+
+        g_list_foreach(filenames, (GFunc) g_free, NULL);
+        g_list_free(filenames);
+    }
+
+    if (options->rew)
+        xmms_remote_playlist_prev(session);
+
+    if (options->play)
+        xmms_remote_play(session);
+
+    if (options->pause)
+        xmms_remote_pause(session);
+
+    if (options->stop)
+        xmms_remote_stop(session);
+
+    if (options->fwd)
+        xmms_remote_playlist_next(session);
+
+    if (options->play_pause)
+        xmms_remote_play_pause(session);
+
+    if (options->show_jump_box)
+        xmms_remote_show_jtf_box(session);
+
+    if (options->mainwin)
+        xmms_remote_main_win_toggle(session, TRUE);
+
+    if (options->activate)
+        xmms_remote_activate(session);
+
+    if (options->playcd)
+        play_medium();
+}
+
+static void
+bmp_setup_logger(void)
+{
+    if (!bmp_logger_start(bmp_paths[BMP_PATH_LOG_FILE]))
+        return;
+
+    g_atexit(bmp_logger_stop);
+}
+
+static void
+run_load_skin_error_dialog(const gchar * skin_path)
+{
+    const gchar *markup =
+        N_("<b><big>Unable to load skin.</big></b>\n"
+           "\n"
+           "Check that skin at '%s' is usable and default skin is properly "
+           "installed at '%s'\n");
+
+    GtkWidget *dialog =
+        gtk_message_dialog_new_with_markup(NULL,
+                                           GTK_DIALOG_MODAL,
+                                           GTK_MESSAGE_ERROR,
+                                           GTK_BUTTONS_CLOSE,
+                                           _(markup),
+                                           skin_path,
+                                           BMP_DEFAULT_SKIN_PATH);
+    gtk_dialog_run(GTK_DIALOG(dialog));
+    gtk_widget_destroy(dialog);
+}
+
+// use a format string?
+void report_error(const gchar *error_text)
+{
+    fprintf(stderr,error_text);
+	if (options.headless!=1) {
+        gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(err),
+                                                 error_text);
+        gtk_dialog_run(GTK_DIALOG(err));
+        gtk_widget_hide(err);
+    }
+}
+
+gint
+main(gint argc, gchar ** argv)
+{
+    gboolean gtk_init_check_ok;
+    Playlist *playlist;
+
+    /* Setup l10n early so we can print localized error messages */
+    gtk_set_locale();
+    bindtextdomain(PACKAGE_NAME, LOCALEDIR);
+    bind_textdomain_codeset(PACKAGE_NAME, "UTF-8");
+    textdomain(PACKAGE_NAME);
+
+    bmp_init_paths();
+    bmp_make_user_dir();
+
+    /* Check GTK version. Really, this is only needed for binary
+     * distribution since configure already checks. */
+    if (!GTK_CHECK_VERSION(2, 6, 0)) {
+        g_printerr(_("Sorry, your GTK+ version (%d.%d.%d) does not work with Audacious.\n"
+                     "Please use GTK+ %s or newer.\n"),
+                   gtk_major_version, gtk_minor_version, gtk_micro_version,
+                   "2.6.0");
+        exit(EXIT_FAILURE);
+    }
+
+    g_set_application_name(_(application_name));
+
+    g_thread_init(NULL);
+    if (!g_thread_supported()) {
+        g_printerr(_("Sorry, threads isn't supported on your platform.\n\n"
+                     "If you're on a libc5 based linux system and installed Glib & GTK+ before you\n"
+                     "installed LinuxThreads you need to recompile Glib & GTK+.\n"));
+        exit(EXIT_FAILURE);
+    }
+
+    gdk_threads_init();
+
+    cond_scan = g_cond_new();
+    mutex_scan = g_mutex_new();
+
+    gtk_init_check_ok = gtk_init_check(&argc, &argv);
+    /* Now let's parse the command line options first. */
+    parse_cmd_line(argc, argv, &options);
+    if (!gtk_init_check_ok) {
+        if (argc < 2) {
+            /* GTK check failed, and no arguments passed to indicate
+               that user is intending to only remote control a running
+               session */
+            g_printerr(_("audacious: Unable to open display, exiting.\n"));
+            exit(EXIT_FAILURE);
+        }
+
+        handle_cmd_line_options(&options, TRUE);
+
+        /* we could be running headless, so GTK probably wont matter */
+        if (options.headless != 1)
+          exit(EXIT_SUCCESS);
+    }
+
+    if (options.no_log == FALSE)
+        bmp_setup_logger();
+
+    signal_handlers_init();
+
+    g_random_set_seed(time(NULL));
+
+    bmp_config_load();
+
+    if (options.session != -1 || !ctrlsocket_setup()) {
+        handle_cmd_line_options(&options, TRUE);
+        exit(EXIT_SUCCESS);
+    }
+
+    plugin_system_init();
+
+    /* Initialize the playlist system. */
+    playlist_init();
+
+    if (options.headless != 1)
+    {
+        /* register icons in stock */
+        register_aud_stock_icons();
+
+        bmp_set_default_icon();
+
+        gtk_accel_map_load(bmp_paths[BMP_PATH_ACCEL_FILE]);
+
+        /* uimanager */
+        ui_manager_init();
+        ui_manager_create_menus();
+
+        if (!init_skins(cfg.skin)) {
+            run_load_skin_error_dialog(cfg.skin);
+            exit(EXIT_FAILURE);
+        }
+
+        GDK_THREADS_ENTER();
+    }
+
+    /* Load the default playlist in. */
+    playlist = playlist_get_active();
+    playlist_load(playlist, bmp_paths[BMP_PATH_PLAYLIST_FILE]);
+    playlist_set_position(playlist, cfg.playlist_position);
+
+    /* this needs to be called after all 3 windows are created and
+     * input plugins are setup'ed 
+     * but not if we're running headless --nenolod
+     */
+    mainwin_setup_menus();
+
+    if (options.headless != 1)
+        GDK_THREADS_LEAVE();
+
+    ctrlsocket_start();
+
+    handle_cmd_line_options(&options, FALSE);
+
+    if (options.headless != 1)
+    {
+        GDK_THREADS_ENTER();
+
+        read_volume(VOLSET_STARTUP);
+        mainwin_set_info_text();
+
+        /* FIXME: delayed, because it deals directly with the plugin
+         * interface to set menu items */
+        create_prefs_window();
+
+	create_fileinfo_window();
+	create_filepopup_window();
+
+        if (cfg.player_visible)
+            mainwin_show(TRUE);
+        else if (!cfg.playlist_visible && !cfg.equalizer_visible)
+            mainwin_show(TRUE);
+
+        if (cfg.equalizer_visible)
+            equalizerwin_show(TRUE);
+
+        if (cfg.playlist_visible)
+            playlistwin_show();
+
+        hint_set_always(cfg.always_on_top);
+
+        playlist_start_get_info_thread();
+        mainwin_attach_idle_func();
+
+	starting_up = FALSE;
+
+	has_x11_connection = TRUE;
+
+	if (cfg.resume_playback_on_startup) {
+		if (cfg.resume_playback_on_startup_time != -1 && playlist_get_length(playlist) > 0) {
+			int i;
+			gint l=0, r=0;
+			while (gtk_events_pending()) gtk_main_iteration();
+			output_get_volume(&l, &r);
+			output_set_volume(0,0);
+			playback_initiate();
+
+			/* Busy wait; loop is fairly tight to minimize duration of "frozen" GUI. Feel free to 
+			 * tune. --chainsaw
+			 */
+			for (i = 0; i < 20; i++) { 
+				g_usleep(1000);
+				if (!ip_data.playing)
+					break;
+			}
+			playback_seek(cfg.resume_playback_on_startup_time /
+					  1000);
+			output_set_volume(l, r);
+		}
+	}
+
+        gtk_main();
+
+        GDK_THREADS_LEAVE();
+
+        g_cond_free(cond_scan);
+        g_mutex_free(mutex_scan);
+
+        return EXIT_SUCCESS;
+    }
+    else
+    {
+        mainwin_set_info_text();
+        playlist_start_get_info_thread();
+
+	starting_up = FALSE;
+
+        for (;;)
+	{
+	    /* headless eventloop */
+	    audcore_generic_events();
+	    free_vis_data();          /* to prevent buffer overflow -- paranoia */
+            xmms_usleep(10000);
+        }
+
+        return EXIT_SUCCESS;
+    }
+}