#include <QString>
#include <QTimer>
#include <QTranslator>
+#include <QFile>
#include "global.h"
#include "logger.h"
#include "cliparser.h"
#if defined BUILD_CORE
-#include <QCoreApplication>
#include <QDir>
#include "core.h"
#include "message.h"
#elif defined BUILD_QTUI
-#include <QApplication>
#include "client.h"
+#include "qtuiapplication.h"
#include "qtui.h"
#elif defined BUILD_MONO
-#include <QApplication>
#include "client.h"
#include "core.h"
#include "coresession.h"
+#include "qtuiapplication.h"
#include "qtui.h"
#else
#error "Something is wrong - you need to #define a build mode!"
#endif
+
#include <signal.h>
+#ifndef Q_OS_WIN32
+#include <execinfo.h>
+#include <dlfcn.h>
+#include <cxxabi.h>
+#endif
+
//! Signal handler for graceful shutdown.
void handle_signal(int sig) {
qWarning("%s", qPrintable(QString("Caught signal %1 - exiting.").arg(sig)));
QCoreApplication::quit();
}
+#ifndef Q_OS_WIN32
+void handle_crash(int sig) {
+ void* callstack[128];
+ int i, frames = backtrace(callstack, 128);
+
+ QFile dumpFile(QString("Quassel-Crash-%1").arg(QDateTime::currentDateTime().toString("yyyyMMdd-hhmm.log")));
+ dumpFile.open(QIODevice::WriteOnly);
+ QTextStream dumpStream(&dumpFile);
+
+ for (i = 0; i < frames; ++i) {
+ Dl_info info;
+ dladdr (callstack[i], &info);
+ // as a reference:
+ // typedef struct
+ // {
+ // __const char *dli_fname; /* File name of defining object. */
+ // void *dli_fbase; /* Load address of that object. */
+ // __const char *dli_sname; /* Name of nearest symbol. */
+ // void *dli_saddr; /* Exact value of nearest symbol. */
+ // } Dl_info;
+
+#if __LP64__
+ int addrSize = 16;
+#else
+ int addrSize = 8;
+#endif
+
+ QString funcName;
+ if(info.dli_sname) {
+ char *func = abi::__cxa_demangle(info.dli_sname, 0, 0, 0);
+ if(func) {
+ funcName = QString(func);
+ free(func);
+ } else {
+ funcName = QString(info.dli_sname);
+ }
+ } else {
+ funcName = QString("0x%1").arg((long)info.dli_saddr, addrSize, QLatin1Char('0'));
+ }
+
+ // prettificating the filename
+ QString fileName("???");
+ if(info.dli_fname) {
+ fileName = QString(info.dli_fname);
+ int slashPos = fileName.lastIndexOf('/');
+ if(slashPos != -1)
+ fileName = fileName.mid(slashPos + 1);
+ if(fileName.count() < 20)
+ fileName += QString(20 - fileName.count(), ' ');
+ }
+
+ QString debugLine = QString("#%1 %2 0x%3 %4").arg(i, 3, 10)
+ .arg(fileName)
+ .arg((long)(callstack[i]), addrSize, 16, QLatin1Char('0'))
+ .arg(funcName);
+
+ dumpStream << debugLine << "\n";
+ qDebug() << qPrintable(debugLine);
+ }
+ dumpFile.close();
+ exit(27);
+}
+#endif // ifndef Q_OS_WIN32
+
+
int main(int argc, char **argv) {
// We catch SIGTERM and SIGINT (caused by Ctrl+C) to graceful shutdown Quassel.
signal(SIGTERM, handle_signal);
signal(SIGINT, handle_signal);
- // Logger logger;
-
+#ifndef Q_OS_WIN32
+ signal(SIGABRT, handle_crash);
+ signal(SIGBUS, handle_crash);
+ signal(SIGSEGV, handle_crash);
+#endif // ndef Q_OS_WIN32
+
Global::registerMetaTypes();
Global::setupVersion();
+/*
#if defined BUILD_CORE
Global::runMode = Global::CoreOnly;
QCoreApplication app(argc, argv);
Global::runMode = Global::Monolithic;
QApplication app(argc, argv);
#endif
+*/
+#if defined BUILD_CORE
+ Global::runMode = Global::CoreOnly;
+ QCoreApplication app(argc, argv);
+#elif defined BUILD_QTUI
+ Global::runMode = Global::ClientOnly;
+ QtUiApplication app(argc, argv);
+#else
+ Global::runMode = Global::Monolithic;
+ QtUiApplication app(argc, argv);
+#endif
+
+
Global::parser = CliParser(QCoreApplication::arguments());
// put core-only arguments here
Global::parser.addOption("port",'p',"The port quasselcore will listen at",QString("4242"));
Global::parser.addSwitch("norestore", 'n', "Don't restore last core's state");
+ Global::parser.addOption("logfile",'l',"Path to logfile");
+ Global::parser.addOption("loglevel",'L',"Loglevel Debug|Info|Warning|Error","Info");
+ Global::parser.addOption("datadir", 0, "Specify the directory holding datafiles like the Sqlite DB and the SSL Cert");
#endif // BUILD_QTUI
#ifndef BUILD_CORE
// put client-only arguments here
return 1;
}
+ /*
+ This is an initial check if logfile is writable since the warning would spam stdout if done
+ in current Logger implementation. Can be dropped whenever the logfile is only opened once.
+ */
+ if(Global::runMode != Global::ClientOnly) {
+ QFile logFile;
+ if(!Global::parser.value("logfile").isEmpty()) {
+ logFile.setFileName(Global::parser.value("logfile"));
+ if(!logFile.open(QIODevice::Append | QIODevice::Text))
+ qWarning("Warning: Couldn't open logfile '%s' - will log to stdout instead",qPrintable(logFile.fileName()));
+ else logFile.close();
+ }
+ }
+
qsrand(QTime(0,0,0).secsTo(QTime::currentTime()));
// Set up i18n support
QLocale locale = QLocale::system();
- QTranslator qtTranslator;
+ QTranslator qtTranslator(&app);
+ qtTranslator.setObjectName("QtTr");
qtTranslator.load(QString(":i18n/qt_%1").arg(locale.name()));
app.installTranslator(&qtTranslator);
- QTranslator quasselTranslator;
+ QTranslator quasselTranslator(&app);
+ quasselTranslator.setObjectName("QuasselTr");
quasselTranslator.load(QString(":i18n/quassel_%1").arg(locale.name()));
app.installTranslator(&quasselTranslator);
QCoreApplication::setApplicationName("Quassel IRC");
QCoreApplication::setOrganizationName("Quassel Project");
+
#ifndef BUILD_QTUI
Core::instance(); // create and init the core
#endif
//Settings::init();
#ifndef BUILD_CORE
+ // session resume
QtUi *gui = new QtUi();
Client::init(gui);
// init gui only after the event loop has started
}
#endif
+#ifndef BUILD_CORE
+ app.resumeSessionIfPossible();
+#endif
+
int exitCode = app.exec();
#ifndef BUILD_QTUI