mirror of
https://bitbucket.org/smil3y/kdelibs.git
synced 2025-02-23 18:32:49 +00:00

tho it can be fixed (by not using queued up message boxes) I am not into supporting such feature Signed-off-by: Ivailo Monev <xakepa10@gmail.com>
573 lines
18 KiB
C++
573 lines
18 KiB
C++
/* This file is part of the KDE libraries
|
|
Copyright (C) 2022 Ivailo Monev <xakepa10@gmail.com>
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public
|
|
License version 2, as published by the Free Software Foundation.
|
|
|
|
This library 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
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public License
|
|
along with this library; see the file COPYING.LIB. If not, write to
|
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "kdebug.h"
|
|
#include "kglobal.h"
|
|
#include "kconfig.h"
|
|
#include "kconfiggroup.h"
|
|
#include "kstandarddirs.h"
|
|
#include "kcomponentdata.h"
|
|
#include "kurl.h"
|
|
|
|
#include <QCoreApplication>
|
|
#include <QFile>
|
|
#include <QMutex>
|
|
#include <QDateTime>
|
|
|
|
#include <unistd.h>
|
|
#include <stdio.h>
|
|
#include <syslog.h>
|
|
|
|
#ifdef HAVE_BACKTRACE
|
|
#include <execinfo.h>
|
|
#endif
|
|
|
|
enum KDebugType {
|
|
TypeFile = 0,
|
|
TypeShell = 1,
|
|
TypeSyslog = 2,
|
|
TypeOff = 3
|
|
};
|
|
|
|
static const QString s_kdebugfilepath = QString::fromLatin1("kdebug.log");
|
|
|
|
static int s_kde_debug_methodname = -1;
|
|
static int s_kde_debug_timestamp = -1;
|
|
static int s_kde_debug_color = -1;
|
|
|
|
static QByteArray kDebugHeader(const QByteArray &areaname, const char* const funcinfo, const int areaoutput)
|
|
{
|
|
if (s_kde_debug_methodname == -1) {
|
|
s_kde_debug_methodname = !qgetenv("KDE_DEBUG_METHODNAME").isEmpty();
|
|
}
|
|
if (s_kde_debug_timestamp == -1) {
|
|
s_kde_debug_timestamp = !qgetenv("KDE_DEBUG_TIMESTAMP").isEmpty();
|
|
}
|
|
|
|
// syslog timestamps messages
|
|
const bool addtimestamp = (s_kde_debug_timestamp && areaoutput != KDebugType::TypeSyslog);
|
|
|
|
if (!s_kde_debug_methodname && !addtimestamp) {
|
|
return areaname;
|
|
}
|
|
|
|
QByteArray result(areaname);
|
|
if (s_kde_debug_methodname) {
|
|
result.append(" from ");
|
|
const QList<QByteArray> funcinfolist = QByteArray(funcinfo).split(' ');
|
|
bool foundfunc = false;
|
|
foreach (const QByteArray &it, funcinfolist) {
|
|
if (it.contains('(') && it.contains(')')) {
|
|
result.append(it);
|
|
foundfunc = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!foundfunc) {
|
|
result.append(funcinfo);
|
|
}
|
|
}
|
|
|
|
if (addtimestamp) {
|
|
static const QString timestamp_format = QString::fromLatin1("hh:mm:ss.zzz");
|
|
const QByteArray timestamp = QTime::currentTime().toString(timestamp_format).toLocal8Bit();
|
|
result.append(" at ");
|
|
result.append(timestamp.constData(), timestamp.size());
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
K_GLOBAL_STATIC(QMutex, globalKDebugMutex)
|
|
|
|
class KDebugNullDevice: public QIODevice
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
KDebugNullDevice() { open(QIODevice::WriteOnly); }
|
|
|
|
protected:
|
|
qint64 readData(char*, qint64) final
|
|
{ return 0; /* eof */ }
|
|
qint64 writeData(const char*, qint64 len)
|
|
{ return len; }
|
|
};
|
|
K_GLOBAL_STATIC(KDebugNullDevice, globalKDebugNullDevie)
|
|
|
|
|
|
class KDebugFileDevice: public KDebugNullDevice
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
KDebugFileDevice()
|
|
: m_type(QtDebugMsg),
|
|
m_abortfatal(true),
|
|
m_filepath(s_kdebugfilepath)
|
|
{ }
|
|
|
|
void setType(const QtMsgType type)
|
|
{ m_type = type; }
|
|
void setAbortFatal(const bool abortfatal)
|
|
{ m_abortfatal = abortfatal; }
|
|
void setHeader(const QByteArray &header)
|
|
{ m_header = header; }
|
|
void setFilepath(const QString &filepath)
|
|
{ m_filepath = filepath; }
|
|
|
|
protected:
|
|
qint64 writeData(const char* data, qint64 len) final
|
|
{
|
|
QFile writefile(m_filepath);
|
|
if (!writefile.open(QIODevice::WriteOnly | QIODevice::Append | QIODevice::Unbuffered)) {
|
|
if (m_abortfatal && m_type == QtFatalMsg) {
|
|
::abort();
|
|
}
|
|
return 0;
|
|
}
|
|
// TODO: insert type somewhere
|
|
QByteArray writedata(m_header);
|
|
writedata.append(": ", 2);
|
|
writedata.append(data, len);
|
|
writedata.append("\n", 1);
|
|
writefile.write(writedata.constData(), writedata.size());
|
|
if (m_abortfatal && m_type == QtFatalMsg) {
|
|
::abort();
|
|
}
|
|
return len;
|
|
}
|
|
|
|
private:
|
|
Q_DISABLE_COPY(KDebugFileDevice);
|
|
QtMsgType m_type;
|
|
bool m_abortfatal;
|
|
QByteArray m_header;
|
|
QString m_filepath;
|
|
};
|
|
|
|
class KDebugShellDevice: public KDebugNullDevice
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
KDebugShellDevice()
|
|
: m_type(QtDebugMsg),
|
|
m_abortfatal(true)
|
|
{ }
|
|
|
|
void setType(const QtMsgType type)
|
|
{ m_type = type; }
|
|
void setAbortFatal(const bool abortfatal)
|
|
{ m_abortfatal = abortfatal; }
|
|
void setHeader(const QByteArray &header)
|
|
{ m_header = header; }
|
|
|
|
protected:
|
|
qint64 writeData(const char* data, qint64 len) final
|
|
{
|
|
if (s_kde_debug_color == -1) {
|
|
s_kde_debug_color = !qgetenv("KDE_DEBUG_COLOR").isEmpty();
|
|
}
|
|
|
|
if (s_kde_debug_color) {
|
|
static const bool isttyoutput = ::isatty(::fileno(stderr));
|
|
if (isttyoutput) {
|
|
switch (m_type) {
|
|
// for reference:
|
|
// https://en.wikipedia.org/wiki/ANSI_escape_code#3-bit_and_4-bit
|
|
case QtDebugMsg: {
|
|
::fprintf(stderr, "\033[0;32m%s: %s\033[0m\n", m_header.constData(), data);
|
|
::fflush(stderr);
|
|
break;
|
|
}
|
|
case QtWarningMsg: {
|
|
::fprintf(stderr, "\033[0;33m%s: %s\033[0m\n", m_header.constData(), data);
|
|
::fflush(stderr);
|
|
break;
|
|
}
|
|
case QtCriticalMsg: {
|
|
::fprintf(stderr, "\033[0;31m%s: %s\033[0m\n", m_header.constData(), data);
|
|
::fflush(stderr);
|
|
break;
|
|
}
|
|
case QtFatalMsg: {
|
|
::fprintf(stderr, "\033[0;5;31m%s: %s\033[0m\n", m_header.constData(), data);
|
|
::fflush(stderr);
|
|
if (m_abortfatal) {
|
|
::abort();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
return len;
|
|
}
|
|
}
|
|
|
|
::fprintf(stderr, "%s: %s\n", m_header.constData(), data);
|
|
::fflush(stderr);
|
|
if (m_abortfatal && m_type == QtFatalMsg) {
|
|
::abort();
|
|
}
|
|
return len;
|
|
}
|
|
|
|
private:
|
|
Q_DISABLE_COPY(KDebugShellDevice);
|
|
QtMsgType m_type;
|
|
bool m_abortfatal;
|
|
QByteArray m_header;
|
|
};
|
|
|
|
class KDebugSyslogDevice: public KDebugNullDevice
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
KDebugSyslogDevice(const QByteArray &areaname)
|
|
: m_type(QtDebugMsg),
|
|
m_abortfatal(true),
|
|
m_areaname(areaname)
|
|
{ }
|
|
|
|
void setType(const QtMsgType type)
|
|
{ m_type = type; }
|
|
void setAbortFatal(const bool abortfatal)
|
|
{ m_abortfatal = abortfatal; }
|
|
void setHeader(const QByteArray &header)
|
|
{ m_header = header; }
|
|
|
|
protected:
|
|
qint64 writeData(const char* data, qint64 len) final
|
|
{
|
|
::openlog(m_areaname.constData(), 0, LOG_USER);
|
|
switch (m_type) {
|
|
case QtDebugMsg: {
|
|
::syslog(LOG_INFO, "%s: %s", m_header.constData(), data);
|
|
break;
|
|
}
|
|
case QtWarningMsg: {
|
|
::syslog(LOG_WARNING, "%s: %s", m_header.constData(), data);
|
|
break;
|
|
}
|
|
case QtCriticalMsg: {
|
|
::syslog(LOG_CRIT, "%s: %s", m_header.constData(), data);
|
|
break;
|
|
}
|
|
case QtFatalMsg: {
|
|
::syslog(LOG_ERR, "%s: %s", m_header.constData(), data);
|
|
break;
|
|
}
|
|
}
|
|
::closelog();
|
|
if (m_abortfatal && m_type == QtFatalMsg) {
|
|
::abort();
|
|
}
|
|
return len;
|
|
}
|
|
|
|
private:
|
|
Q_DISABLE_COPY(KDebugSyslogDevice);
|
|
QtMsgType m_type;
|
|
bool m_abortfatal;
|
|
QByteArray m_header;
|
|
QByteArray m_areaname;
|
|
};
|
|
|
|
class KDebugAreaCache
|
|
{
|
|
public:
|
|
KDebugAreaCache()
|
|
: infooutput(KDebugType::TypeOff),
|
|
warnoutput(KDebugType::TypeShell),
|
|
erroroutput(KDebugType::TypeShell),
|
|
fataloutput(KDebugType::TypeShell),
|
|
infofilename(s_kdebugfilepath),
|
|
warnfilename(s_kdebugfilepath),
|
|
errorfilename(s_kdebugfilepath),
|
|
fatalfilename(s_kdebugfilepath),
|
|
abortfatal(true)
|
|
{ }
|
|
|
|
int infooutput;
|
|
int warnoutput;
|
|
int erroroutput;
|
|
int fataloutput;
|
|
QString infofilename;
|
|
QString warnfilename;
|
|
QString errorfilename;
|
|
QString fatalfilename;
|
|
bool abortfatal;
|
|
};
|
|
|
|
class KDebugConfig : public KConfig
|
|
{
|
|
public:
|
|
KDebugConfig();
|
|
~KDebugConfig();
|
|
|
|
void cacheAreas();
|
|
void destroyDevices();
|
|
|
|
QIODevice* areaDevice(const QtMsgType type, const char* const funcinfo, const int area);
|
|
|
|
private:
|
|
Q_DISABLE_COPY(KDebugConfig);
|
|
|
|
QByteArray areaName(const int area) const;
|
|
KDebugAreaCache areaCache(const int area);
|
|
|
|
bool m_disableall;
|
|
QMap<int,QByteArray> m_areanames;
|
|
QMap<int,KDebugAreaCache> m_areacache;
|
|
QMap<uint,QIODevice*> m_areadevices;
|
|
};
|
|
K_GLOBAL_STATIC(KDebugConfig, globalKDebugConfig)
|
|
|
|
KDebugConfig::KDebugConfig()
|
|
: KConfig(QString::fromLatin1("kdebugrc"), KConfig::NoGlobals),
|
|
m_disableall(false)
|
|
{
|
|
cacheAreas();
|
|
}
|
|
|
|
KDebugConfig::~KDebugConfig()
|
|
{
|
|
destroyDevices();
|
|
}
|
|
|
|
void KDebugConfig::cacheAreas()
|
|
{
|
|
m_areanames.clear();
|
|
m_areacache.clear();
|
|
|
|
KConfigGroup generalgroup = KConfig::group(QString());
|
|
m_disableall = generalgroup.readEntry("DisableAll", false);
|
|
if (m_disableall) {
|
|
return;
|
|
}
|
|
|
|
const QString kdebugareas = KStandardDirs::locate("config", QString::fromLatin1("kdebug.areas"));
|
|
if (kdebugareas.isEmpty()) {
|
|
return;
|
|
}
|
|
QFile kdebugareasfile(kdebugareas);
|
|
if (!kdebugareasfile.open(QFile::ReadOnly)) {
|
|
return;
|
|
}
|
|
while (!kdebugareasfile.atEnd()) {
|
|
const QByteArray kdebugareasline = kdebugareasfile.readLine().trimmed();
|
|
if (kdebugareasline.isEmpty() || kdebugareasline.startsWith('#')) {
|
|
continue;
|
|
}
|
|
const int spaceindex = kdebugareasline.indexOf(' ');
|
|
if (spaceindex < 1) {
|
|
continue;
|
|
}
|
|
const int areanumber = kdebugareasline.mid(0, spaceindex).toLongLong();
|
|
const QByteArray areaname = kdebugareasline.mid(spaceindex + 1, kdebugareasline.size() - spaceindex - 1).trimmed();
|
|
if (areanumber <= 0 || areaname.isEmpty()) {
|
|
continue;
|
|
}
|
|
m_areanames.insert(areanumber, areaname);
|
|
// qDebug() << Q_FUNC_INFO << areanumber << areaname;
|
|
}
|
|
}
|
|
|
|
QByteArray KDebugConfig::areaName(const int area) const
|
|
{
|
|
const QByteArray areaname = m_areanames.value(area);
|
|
if (!areaname.isEmpty()) {
|
|
return areaname;
|
|
}
|
|
|
|
if (KGlobal::hasMainComponent()) {
|
|
return KGlobal::mainComponent().componentName().toUtf8();
|
|
}
|
|
return QCoreApplication::applicationName().toUtf8();
|
|
}
|
|
|
|
KDebugAreaCache KDebugConfig::areaCache(const int area)
|
|
{
|
|
QMap<int,KDebugAreaCache>::const_iterator it = m_areacache.constFind(area);
|
|
if (it == m_areacache.constEnd()) {
|
|
KDebugAreaCache kdebugareacache;
|
|
KConfigGroup areagroup = KConfig::group(QByteArray::number(area));
|
|
kdebugareacache.infooutput = areagroup.readEntry("InfoOutput", int(KDebugType::TypeOff));
|
|
kdebugareacache.infofilename = areagroup.readPathEntry("InfoFilename", s_kdebugfilepath);
|
|
kdebugareacache.warnoutput = areagroup.readEntry("WarnOutput", int(KDebugType::TypeShell));
|
|
kdebugareacache.warnfilename = areagroup.readPathEntry("WarnFilename", s_kdebugfilepath);
|
|
kdebugareacache.erroroutput = areagroup.readEntry("ErrorOutput", int(KDebugType::TypeShell));
|
|
kdebugareacache.errorfilename = areagroup.readPathEntry("ErrorFilename", s_kdebugfilepath);
|
|
kdebugareacache.fataloutput = areagroup.readEntry("FatalOutput", int(KDebugType::TypeShell));
|
|
kdebugareacache.fatalfilename = areagroup.readPathEntry("FatalFilename", s_kdebugfilepath);
|
|
kdebugareacache.abortfatal = areagroup.readEntry("AbortFatal", true);
|
|
m_areacache.insert(area, kdebugareacache);
|
|
return kdebugareacache;
|
|
}
|
|
return it.value();
|
|
}
|
|
|
|
void KDebugConfig::destroyDevices()
|
|
{
|
|
foreach (const uint area, m_areadevices.keys()) {
|
|
QIODevice* qiodevice = m_areadevices.take(area);
|
|
delete qiodevice;
|
|
}
|
|
}
|
|
|
|
QIODevice* KDebugConfig::areaDevice(const QtMsgType type, const char* const funcinfo, const int area)
|
|
{
|
|
if (m_disableall) {
|
|
return globalKDebugNullDevie;
|
|
}
|
|
|
|
const KDebugAreaCache kdebugareacache = KDebugConfig::areaCache(area);
|
|
int areaoutput = int(KDebugType::TypeShell);
|
|
QString areafilename;
|
|
bool areaabort = true;
|
|
switch (type) {
|
|
case QtDebugMsg: {
|
|
areaoutput = kdebugareacache.infooutput;
|
|
areafilename = kdebugareacache.infofilename;
|
|
break;
|
|
}
|
|
case QtWarningMsg: {
|
|
areaoutput = kdebugareacache.warnoutput;
|
|
areafilename = kdebugareacache.warnfilename;
|
|
break;
|
|
}
|
|
case QtCriticalMsg: {
|
|
areaoutput = kdebugareacache.erroroutput;
|
|
areafilename = kdebugareacache.errorfilename;
|
|
break;
|
|
}
|
|
case QtFatalMsg: {
|
|
areaoutput = kdebugareacache.fataloutput;
|
|
areafilename = kdebugareacache.fatalfilename;
|
|
areaabort = kdebugareacache.abortfatal;
|
|
break;
|
|
}
|
|
}
|
|
|
|
const uint areakey = (area << 8 | areaoutput);
|
|
switch (areaoutput) {
|
|
case KDebugType::TypeFile: {
|
|
QIODevice* qiodevice = m_areadevices.value(areakey, nullptr);
|
|
if (!qiodevice) {
|
|
qiodevice = new KDebugFileDevice();
|
|
m_areadevices.insert(areakey, qiodevice);
|
|
}
|
|
KDebugFileDevice* kdebugdevice = qobject_cast<KDebugFileDevice*>(qiodevice);
|
|
kdebugdevice->setType(type);
|
|
kdebugdevice->setAbortFatal(areaabort);
|
|
kdebugdevice->setHeader(kDebugHeader(KDebugConfig::areaName(area), funcinfo, areaoutput));
|
|
kdebugdevice->setFilepath(areafilename);
|
|
return kdebugdevice;
|
|
}
|
|
case KDebugType::TypeShell: {
|
|
QIODevice* qiodevice = m_areadevices.value(areakey, nullptr);
|
|
if (!qiodevice) {
|
|
qiodevice = new KDebugShellDevice();
|
|
m_areadevices.insert(areakey, qiodevice);
|
|
}
|
|
KDebugShellDevice* kdebugdevice = qobject_cast<KDebugShellDevice*>(qiodevice);
|
|
kdebugdevice->setType(type);
|
|
kdebugdevice->setAbortFatal(areaabort);
|
|
kdebugdevice->setHeader(kDebugHeader(KDebugConfig::areaName(area), funcinfo, areaoutput));
|
|
return kdebugdevice;
|
|
}
|
|
case KDebugType::TypeSyslog: {
|
|
QIODevice* qiodevice = m_areadevices.value(areakey, nullptr);
|
|
if (!qiodevice) {
|
|
qiodevice = new KDebugSyslogDevice(KDebugConfig::areaName(area));
|
|
m_areadevices.insert(areakey, qiodevice);
|
|
}
|
|
KDebugSyslogDevice* kdebugdevice = qobject_cast<KDebugSyslogDevice*>(qiodevice);
|
|
kdebugdevice->setType(type);
|
|
kdebugdevice->setAbortFatal(areaabort);
|
|
kdebugdevice->setHeader(kDebugHeader(KDebugConfig::areaName(area), funcinfo, areaoutput));
|
|
return kdebugdevice;
|
|
}
|
|
case KDebugType::TypeOff:
|
|
default: {
|
|
// can't issue a warning about invalid type from KDebug itself
|
|
return globalKDebugNullDevie;
|
|
}
|
|
}
|
|
Q_UNREACHABLE();
|
|
}
|
|
|
|
QByteArray kBacktrace(int levels)
|
|
{
|
|
#ifdef HAVE_BACKTRACE
|
|
void* tracebuffer[256];
|
|
int tracecount = backtrace(tracebuffer, 256);
|
|
if (!tracecount) {
|
|
return QByteArray();
|
|
}
|
|
char** tracestrings = backtrace_symbols(tracebuffer, tracecount);
|
|
|
|
if (levels != -1) {
|
|
tracecount = qMin(tracecount, levels);
|
|
}
|
|
|
|
QByteArray result("[\n");
|
|
for (int i = 0; i < tracecount; i++) {
|
|
result += QByteArray::number(i) + ": " + QByteArray(tracestrings[i]) + '\n';
|
|
}
|
|
result += "]\n";
|
|
|
|
if (tracestrings) {
|
|
::free(tracestrings);
|
|
}
|
|
|
|
return result;
|
|
#else
|
|
return QByteArray();
|
|
#endif // HAVE_BACKTRACE
|
|
}
|
|
|
|
void kClearDebugConfig()
|
|
{
|
|
QMutexLocker locker(globalKDebugMutex);
|
|
|
|
globalKDebugConfig->destroyDevices();
|
|
globalKDebugConfig->reparseConfiguration();
|
|
globalKDebugConfig->cacheAreas();
|
|
|
|
s_kde_debug_methodname = -1;
|
|
s_kde_debug_timestamp = -1;
|
|
s_kde_debug_color = -1;
|
|
}
|
|
|
|
QDebug KDebug(const QtMsgType type, const char* const funcinfo, const int area)
|
|
{
|
|
// e.g. called from late destructor
|
|
if (Q_UNLIKELY(globalKDebugMutex.isDestroyed() || globalKDebugConfig.isDestroyed())) {
|
|
return QDebug(type);
|
|
}
|
|
|
|
QMutexLocker locker(globalKDebugMutex);
|
|
return QDebug(globalKDebugConfig->areaDevice(type, funcinfo, area));
|
|
}
|
|
|
|
QDebug operator<<(QDebug s, const KUrl &url)
|
|
{
|
|
s.nospace() << "KUrl(" << url.prettyUrl() << ")";
|
|
return s.space();
|
|
}
|
|
|
|
#include "kdebug.moc"
|