2003-04-27 01:51:04 +00:00
|
|
|
// main.cc for Fluxbox Window manager
|
2006-02-16 06:53:05 +00:00
|
|
|
// Copyright (c) 2001 - 2006 Henrik Kinnunen (fluxgen at fluxbox dot org)
|
2005-01-18 19:06:04 +00:00
|
|
|
// and 2003-2005 Simon Bowden (rathnor at users.sourceforge.net)
|
2001-12-11 20:47:02 +00:00
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
|
|
// to deal in the Software without restriction, including without limitation
|
|
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
2003-05-16 00:46:41 +00:00
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
2001-12-11 20:47:02 +00:00
|
|
|
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
|
|
|
|
#include "fluxbox.hh"
|
2003-05-16 00:46:41 +00:00
|
|
|
#include "version.h"
|
2003-07-18 15:44:36 +00:00
|
|
|
#include "defaults.hh"
|
2014-02-18 18:34:35 +00:00
|
|
|
#include "cli.hh"
|
2004-01-11 13:12:02 +00:00
|
|
|
|
2004-06-07 11:46:05 +00:00
|
|
|
#include "FbTk/I18n.hh"
|
2011-10-31 15:17:28 +00:00
|
|
|
#include "FbTk/StringUtil.hh"
|
2001-12-11 20:47:02 +00:00
|
|
|
|
2002-08-13 23:56:02 +00:00
|
|
|
//use GNU extensions
|
|
|
|
#ifndef _GNU_SOURCE
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#endif // _GNU_SOURCE
|
|
|
|
|
2011-09-10 18:17:57 +00:00
|
|
|
#ifdef HAVE_UNISTD_H
|
2015-01-16 09:52:00 +00:00
|
|
|
#include <unistd.h>
|
2011-09-10 18:17:57 +00:00
|
|
|
#endif
|
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
#ifdef HAVE_SYS_WAIT_H
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#endif // HAVE_SYS_WAIT_H
|
2010-09-18 15:51:30 +00:00
|
|
|
|
2015-01-03 13:07:34 +00:00
|
|
|
#ifdef HAVE_SIGNAL_H
|
|
|
|
#include <signal.h>
|
|
|
|
#endif
|
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
|
2015-01-16 09:52:00 +00:00
|
|
|
#include <cstdlib>
|
|
|
|
#include <cstring>
|
2014-02-18 18:34:35 +00:00
|
|
|
#include <iostream>
|
2002-10-13 21:48:28 +00:00
|
|
|
#include <stdexcept>
|
2003-01-09 18:28:48 +00:00
|
|
|
#include <typeinfo>
|
2002-10-13 21:48:28 +00:00
|
|
|
|
2006-10-27 06:57:43 +00:00
|
|
|
using std::cout;
|
|
|
|
using std::cerr;
|
|
|
|
using std::endl;
|
|
|
|
using std::string;
|
|
|
|
using std::ostream;
|
|
|
|
using std::ofstream;
|
|
|
|
using std::streambuf;
|
|
|
|
using std::auto_ptr;
|
|
|
|
using std::out_of_range;
|
|
|
|
using std::runtime_error;
|
|
|
|
using std::bad_cast;
|
|
|
|
using std::bad_alloc;
|
|
|
|
using std::exception;
|
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
namespace {
|
2006-06-21 14:41:16 +00:00
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
auto_ptr<Fluxbox> fluxbox;
|
2006-06-21 14:41:16 +00:00
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
void handleSignal(int signum) {
|
2013-01-10 11:11:24 +00:00
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
_FB_USES_NLS;
|
|
|
|
static int re_enter = 0;
|
|
|
|
|
|
|
|
switch (signum) {
|
|
|
|
#ifndef _WIN32
|
|
|
|
case SIGCHLD: // we don't want the child process to kill us
|
|
|
|
// more than one process may have terminated
|
|
|
|
while (waitpid(-1, 0, WNOHANG | WUNTRACED) > 0);
|
|
|
|
break;
|
|
|
|
case SIGHUP:
|
|
|
|
// xinit sends HUP when it wants to go down. there is no point in
|
|
|
|
// restoring anything in the screens / workspaces, the connection
|
|
|
|
// to the xserver might drop any moment
|
|
|
|
if (fluxbox.get()) { fluxbox->shutdown(1); }
|
|
|
|
break;
|
|
|
|
case SIGUSR1:
|
|
|
|
if (fluxbox.get()) { fluxbox->restart(); }
|
|
|
|
break;
|
|
|
|
case SIGUSR2:
|
|
|
|
if (fluxbox.get()) { fluxbox->reconfigure(); }
|
|
|
|
break;
|
2012-12-12 09:18:20 +00:00
|
|
|
#endif
|
2014-02-18 18:34:35 +00:00
|
|
|
case SIGSEGV:
|
|
|
|
abort();
|
|
|
|
break;
|
|
|
|
case SIGALRM:
|
|
|
|
// last resort for shutting down fluxbox. the alarm() is set in
|
|
|
|
// Fluxbox::shutdown()
|
|
|
|
if (fluxbox.get() && fluxbox->isShuttingDown()) {
|
|
|
|
cerr << "fluxbox took longer than expected to shutdown\n";
|
|
|
|
exit(13);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case SIGFPE:
|
|
|
|
case SIGINT:
|
|
|
|
#ifndef _WIN32
|
|
|
|
case SIGPIPE:
|
2012-12-12 09:18:20 +00:00
|
|
|
#endif
|
2014-02-18 18:34:35 +00:00
|
|
|
case SIGTERM:
|
|
|
|
if (fluxbox.get()) { fluxbox->shutdown(); }
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
fprintf(stderr,
|
|
|
|
_FB_CONSOLETEXT(BaseDisplay, SignalCaught,
|
|
|
|
"%s: signal %d caught\n",
|
|
|
|
"signal catch debug message. Include %s for Command<void> and %d for signal number").c_str(),
|
|
|
|
"TODO: m_arg[0]", signum);
|
|
|
|
|
|
|
|
if (! fluxbox->isStartup() && ! re_enter) {
|
|
|
|
re_enter = 1;
|
|
|
|
cerr<<_FB_CONSOLETEXT(BaseDisplay, ShuttingDown,
|
|
|
|
"Shutting Down\n",
|
|
|
|
"Quitting because of signal, end with newline");
|
|
|
|
if (fluxbox.get()) { fluxbox->shutdown(); }
|
2010-10-13 13:48:29 +00:00
|
|
|
}
|
2013-06-29 06:39:02 +00:00
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
cerr << _FB_CONSOLETEXT(BaseDisplay, Aborting,
|
|
|
|
"Aborting... dumping core\n",
|
|
|
|
"Aboring and dumping core, end with newline");
|
|
|
|
abort();
|
|
|
|
break;
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
2010-09-17 13:51:16 +00:00
|
|
|
}
|
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
void setupSignalHandling() {
|
|
|
|
signal(SIGSEGV, handleSignal);
|
|
|
|
signal(SIGSEGV, handleSignal);
|
|
|
|
signal(SIGFPE, handleSignal);
|
|
|
|
signal(SIGTERM, handleSignal);
|
|
|
|
signal(SIGINT, handleSignal);
|
|
|
|
#ifdef HAVE_ALARM
|
|
|
|
signal(SIGALRM, handleSignal);
|
2011-10-28 17:16:10 +00:00
|
|
|
#endif
|
2014-02-18 18:34:35 +00:00
|
|
|
#ifndef _WIN32
|
|
|
|
signal(SIGPIPE, handleSignal); // e.g. output sent to grep
|
|
|
|
signal(SIGCHLD, handleSignal);
|
|
|
|
signal(SIGHUP, handleSignal);
|
|
|
|
signal(SIGUSR1, handleSignal);
|
|
|
|
signal(SIGUSR2, handleSignal);
|
2011-10-28 17:15:48 +00:00
|
|
|
#endif
|
2010-09-18 15:51:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-09-17 13:51:16 +00:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
|
2015-01-31 20:37:44 +00:00
|
|
|
FbTk::I18n::init(0);
|
2010-09-17 13:51:16 +00:00
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
FluxboxCli::Options opts;
|
|
|
|
int exitcode = opts.parse(argc, argv);
|
|
|
|
|
|
|
|
if (exitcode != -1) {
|
|
|
|
exit(exitcode);
|
|
|
|
}
|
|
|
|
exitcode = EXIT_FAILURE;
|
2001-12-11 20:47:02 +00:00
|
|
|
|
2003-05-07 22:19:59 +00:00
|
|
|
#ifdef __EMX__
|
2002-12-01 13:42:15 +00:00
|
|
|
_chdir2(getenv("X11ROOT"));
|
2001-12-11 20:47:02 +00:00
|
|
|
#endif // __EMX__
|
2003-05-16 00:46:41 +00:00
|
|
|
|
|
|
|
streambuf *outbuf = 0;
|
|
|
|
streambuf *errbuf = 0;
|
|
|
|
|
2010-09-17 13:51:16 +00:00
|
|
|
ofstream log_file(opts.log_filename.c_str());
|
|
|
|
|
|
|
|
_FB_USES_NLS;
|
2003-05-16 00:46:41 +00:00
|
|
|
|
|
|
|
// setup log file
|
2010-09-17 13:51:16 +00:00
|
|
|
if (log_file.is_open()) {
|
2010-09-18 15:27:28 +00:00
|
|
|
cerr << _FB_CONSOLETEXT(main, LoggingTo, "Logging to", "Logging to a file")
|
|
|
|
<< ": "
|
|
|
|
<< opts.log_filename << endl;
|
|
|
|
|
|
|
|
log_file <<"------------------------------------------" << endl;
|
|
|
|
log_file << _FB_CONSOLETEXT(main, LogFile, "Log File", "")
|
|
|
|
<< ": "
|
|
|
|
<< opts.log_filename <<endl;
|
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
FluxboxCli::showInfo(log_file);
|
2010-09-18 15:27:28 +00:00
|
|
|
log_file << "------------------------------------------" << endl;
|
2003-05-16 00:46:41 +00:00
|
|
|
// setup log to use cout and cerr stream
|
|
|
|
outbuf = cout.rdbuf(log_file.rdbuf());
|
|
|
|
errbuf = cerr.rdbuf(log_file.rdbuf());
|
|
|
|
}
|
|
|
|
|
2014-02-18 18:34:35 +00:00
|
|
|
FluxboxCli::setupConfigFiles(opts.rc_path, opts.rc_file);
|
|
|
|
FluxboxCli::updateConfigFilesIfNeeded(opts.rc_file);
|
2010-09-17 13:51:16 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
try {
|
2004-10-18 01:26:54 +00:00
|
|
|
|
2010-09-17 13:51:16 +00:00
|
|
|
fluxbox.reset(new Fluxbox(argc, argv,
|
2010-09-18 15:51:30 +00:00
|
|
|
opts.session_display,
|
|
|
|
opts.rc_path,
|
|
|
|
opts.rc_file,
|
2010-09-17 13:51:16 +00:00
|
|
|
opts.xsync));
|
2014-02-18 18:34:35 +00:00
|
|
|
setupSignalHandling();
|
2002-12-01 13:42:15 +00:00
|
|
|
fluxbox->eventLoop();
|
2004-10-18 01:26:54 +00:00
|
|
|
exitcode = EXIT_SUCCESS;
|
2003-05-16 00:46:41 +00:00
|
|
|
|
2006-10-27 06:57:43 +00:00
|
|
|
} catch (out_of_range &oor) {
|
2010-09-18 15:27:28 +00:00
|
|
|
cerr <<"Fluxbox: "
|
|
|
|
<< _FB_CONSOLETEXT(main, ErrorOutOfRange, "Out of range", "Error message")
|
|
|
|
<< ": "
|
|
|
|
<< oor.what() << endl;
|
2006-10-27 06:57:43 +00:00
|
|
|
} catch (runtime_error &re) {
|
2010-09-18 15:27:28 +00:00
|
|
|
cerr << "Fluxbox: "
|
|
|
|
<< _FB_CONSOLETEXT(main, ErrorRuntime, "Runtime error", "Error message")
|
|
|
|
<< ": "
|
|
|
|
<< re.what() << endl;
|
2006-10-27 06:57:43 +00:00
|
|
|
} catch (bad_cast &bc) {
|
2010-09-18 15:27:28 +00:00
|
|
|
cerr << "Fluxbox: "
|
|
|
|
<< _FB_CONSOLETEXT(main, ErrorBadCast, "Bad cast", "Error message")
|
|
|
|
<< ": "
|
|
|
|
<< bc.what() << endl;
|
2006-10-27 06:57:43 +00:00
|
|
|
} catch (bad_alloc &ba) {
|
2010-09-18 15:27:28 +00:00
|
|
|
cerr << "Fluxbox: "
|
|
|
|
<< _FB_CONSOLETEXT(main, ErrorBadAlloc, "Bad Alloc", "Error message")
|
|
|
|
<< ": "
|
|
|
|
<< ba.what() << endl;
|
2006-10-27 06:57:43 +00:00
|
|
|
} catch (exception &e) {
|
2010-09-18 15:27:28 +00:00
|
|
|
cerr << "Fluxbox: "
|
|
|
|
<< _FB_CONSOLETEXT(main, ErrorStandardException, "Standard Exception", "Error message")
|
|
|
|
<< ": "
|
|
|
|
<< e.what() << endl;
|
2011-11-02 17:33:37 +00:00
|
|
|
} catch (string & error_str) {
|
2010-09-18 15:27:28 +00:00
|
|
|
cerr << _FB_CONSOLETEXT(Common, Error, "Error", "Error message header")
|
|
|
|
<< ": "
|
|
|
|
<< error_str << endl;
|
2002-12-01 13:42:15 +00:00
|
|
|
} catch (...) {
|
2010-09-18 15:27:28 +00:00
|
|
|
cerr << "Fluxbox: "
|
|
|
|
<< _FB_CONSOLETEXT(main, ErrorUnknown, "Unknown error", "Error message")
|
|
|
|
<< "." << endl;
|
2002-12-02 22:02:35 +00:00
|
|
|
abort();
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
2004-10-18 01:26:54 +00:00
|
|
|
|
2004-11-24 11:46:07 +00:00
|
|
|
bool restarting = false;
|
2006-10-27 06:57:43 +00:00
|
|
|
string restart_argument;
|
|
|
|
|
2004-11-24 11:46:07 +00:00
|
|
|
if (fluxbox.get()) {
|
|
|
|
restarting = fluxbox->isRestarting();
|
2006-10-27 06:57:43 +00:00
|
|
|
restart_argument = fluxbox->getRestartArgument();
|
2004-11-24 11:46:07 +00:00
|
|
|
}
|
2004-10-18 01:26:54 +00:00
|
|
|
|
2004-01-11 13:12:02 +00:00
|
|
|
// destroy fluxbox
|
|
|
|
fluxbox.reset(0);
|
|
|
|
|
2003-05-16 00:46:41 +00:00
|
|
|
// restore cout and cin streams
|
|
|
|
if (outbuf != 0)
|
|
|
|
cout.rdbuf(outbuf);
|
|
|
|
if (errbuf != 0)
|
|
|
|
cerr.rdbuf(errbuf);
|
|
|
|
|
2006-05-22 15:09:21 +00:00
|
|
|
FbTk::FbStringUtil::shutdown();
|
|
|
|
|
2004-10-18 01:26:54 +00:00
|
|
|
if (restarting) {
|
2007-07-31 23:16:05 +00:00
|
|
|
if (!restart_argument.empty()) {
|
|
|
|
const char *shell = getenv("SHELL");
|
|
|
|
if (!shell)
|
|
|
|
shell = "/bin/sh";
|
|
|
|
|
2008-01-25 07:21:35 +00:00
|
|
|
execlp(shell, shell, "-c", restart_argument.c_str(), (const char *) NULL);
|
|
|
|
perror(restart_argument.c_str());
|
2004-10-18 01:26:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// fall back in case the above execlp doesn't work
|
|
|
|
execvp(argv[0], argv);
|
2006-04-25 02:11:19 +00:00
|
|
|
perror(argv[0]);
|
|
|
|
|
2015-01-16 09:52:00 +00:00
|
|
|
const std::string basename = FbTk::StringUtil::basename(argv[0]);
|
|
|
|
execvp(basename.c_str(), argv);
|
|
|
|
perror(basename.c_str());
|
2004-10-18 01:26:54 +00:00
|
|
|
}
|
|
|
|
|
2002-12-03 01:46:28 +00:00
|
|
|
return exitcode;
|
2001-12-11 20:47:02 +00:00
|
|
|
}
|
2007-04-24 19:38:40 +00:00
|
|
|
|