diff --git a/src/parser/backtraceparser.cpp b/src/parser/backtraceparser.cpp index fb2db64a..81c87b3d 100644 --- a/src/parser/backtraceparser.cpp +++ b/src/parser/backtraceparser.cpp @@ -1,404 +1,406 @@ /* Copyright (C) 2009-2010 George Kiagiadakis 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "backtraceparser_p.h" #include "backtraceparsergdb.h" #include "backtraceparserkdbgwin.h" #include "backtraceparserlldb.h" #include "backtraceparsercdb.h" #include "backtraceparsernull.h" #include "drkonqi_parser_debug.h" #include #include //factory BacktraceParser *BacktraceParser::newParser(const QString & debuggerName, QObject *parent) { if (debuggerName == QLatin1String("gdb")) { return new BacktraceParserGdb(parent); } else if (debuggerName == QLatin1String("kdbgwin")) { return new BacktraceParserKdbgwin(parent); } else if (debuggerName == QLatin1String("lldb")) { return new BacktraceParserLldb(parent); } else if (debuggerName == QLatin1String("cdb")) { return new BacktraceParserCdb(parent); } else { return new BacktraceParserNull(parent); } } BacktraceParser::BacktraceParser(QObject *parent) : QObject(parent), d_ptr(nullptr) {} BacktraceParser::~BacktraceParser() { delete d_ptr; } void BacktraceParser::connectToGenerator(QObject *generator) { connect(generator, SIGNAL(starting()), this, SLOT(resetState())); connect(generator, SIGNAL(newLine(QString)), this, SLOT(newLine(QString))); } QString BacktraceParser::parsedBacktrace() const { Q_D(const BacktraceParser); QString result; if (d) { for (QList::const_iterator i = d->m_linesList.constBegin(), total = d->m_linesList.constEnd(); i != total; ++i) { result += i->toString(); } } return result; } QList BacktraceParser::parsedBacktraceLines() const { Q_D(const BacktraceParser); return d ? d->m_linesList : QList(); } QString BacktraceParser::simplifiedBacktrace() const { Q_D(const BacktraceParser); //if there is no cached usefulness, the data calculation function has not run yet. if (d && d->m_usefulness == InvalidUsefulness) { const_cast(this)->calculateRatingData(); } //if there is no d, the debugger has not run yet, so we have no backtrace. return d ? d->m_simplifiedBacktrace : QString(); } BacktraceParser::Usefulness BacktraceParser::backtraceUsefulness() const { Q_D(const BacktraceParser); //if there is no cached usefulness, the data calculation function has not run yet. if (d && d->m_usefulness == InvalidUsefulness) { const_cast(this)->calculateRatingData(); } //if there is no d, the debugger has not run yet, //so we can say that the (inexistent) backtrace is Useless. return d ? d->m_usefulness : Useless; } QStringList BacktraceParser::firstValidFunctions() const { Q_D(const BacktraceParser); //if there is no cached usefulness, the data calculation function has not run yet. if (d && d->m_usefulness == InvalidUsefulness) { const_cast(this)->calculateRatingData(); } //if there is no d, the debugger has not run yet, so we have no functions to return. return d ? d->m_firstUsefulFunctions : QStringList(); } QSet BacktraceParser::librariesWithMissingDebugSymbols() const { Q_D(const BacktraceParser); //if there is no cached usefulness, the data calculation function has not run yet. if (d && d->m_usefulness == InvalidUsefulness) { const_cast(this)->calculateRatingData(); } //if there is no d, the debugger has not run yet, so we have no libraries. return d ? d->m_librariesWithMissingDebugSymbols : QSet(); } void BacktraceParser::resetState() { //reset the state of the parser by getting a new instance of Private delete d_ptr; d_ptr = constructPrivate(); } BacktraceParserPrivate *BacktraceParser::constructPrivate() const { return new BacktraceParserPrivate; } /* This function returns true if the given stack frame line is the base of the backtrace and thus the parser should not rate any frames below that one. */ static bool lineIsStackBase(const BacktraceLine & line) { //optimization. if there is no function name, do not bother to check it if ( line.rating() == BacktraceLine::MissingEverything || line.rating() == BacktraceLine::MissingFunction ) return false; //this is the base frame for all threads except the main thread //FIXME that probably works only on linux if ( line.functionName() == QLatin1String("start_thread") ) return true; QRegExp regExp; regExp.setPattern(QStringLiteral("(kde)?main")); //main() or kdemain() is the base for the main thread if ( regExp.exactMatch(line.functionName()) ) return true; //HACK for better rating. we ignore all stack frames below any function that matches //the following regular expression. The functions that match this expression are usually //"QApplicationPrivate::notify_helper", "QApplication::notify" and similar, which //are used to send any kind of event to the Qt application. All stack frames below this, //with or without debug symbols, are useless to KDE developers, so we ignore them. regExp.setPattern(QStringLiteral("(Q|K)(Core)?Application(Private)?::notify.*")); if ( regExp.exactMatch(line.functionName()) ) return true; //attempt to recognize crashes that happen after main has returned (bug 200993) if ( line.functionName() == QLatin1String("~KCleanUpGlobalStatic") || line.functionName() == QLatin1String("~QGlobalStatic") || line.functionName() == QLatin1String("exit") || line.functionName() == QLatin1String("*__GI_exit") ) return true; return false; } /* This function returns true if the given stack frame line is the top of the bactrace and thus the parser should not rate any frames above that one. This is used to avoid rating the stack frames of abort(), assert(), Q_ASSERT() and qCCritical(DRKONQI_PARSER_LOG) */ static bool lineIsStackTop(const BacktraceLine & line) { //optimization. if there is no function name, do not bother to check it if ( line.rating() == BacktraceLine::MissingEverything || line.rating() == BacktraceLine::MissingFunction ) return false; if ( line.functionName().startsWith(QLatin1String("qt_assert")) //qt_assert and qt_assert_x || line.functionName() == QLatin1String("qFatal") || line.functionName() == QLatin1String("abort") || line.functionName() == QLatin1String("*__GI_abort") || line.functionName() == QLatin1String("*__GI___assert_fail") ) return true; return false; } /* This function returns true if the given stack frame line should be ignored from rating for some reason. Currently it ignores all libc/libstdc++/libpthread functions. */ static bool lineShouldBeIgnored(const BacktraceLine & line) { if (line.libraryName().contains(QLatin1String("libc.so")) || line.libraryName().contains(QLatin1String("libstdc++.so")) || line.functionName().startsWith(QLatin1String("*__GI_")) //glibc2.9 uses *__GI_ as prefix || line.libraryName().contains(QLatin1String("libpthread.so")) || line.libraryName().contains(QLatin1String("libglib-2.0.so")) + || line.functionName() == QLatin1String("__libc_start_main") // below main on apps without symbols + || line.functionName() == QLatin1String("_start") // below main on apps without symbols #ifdef Q_OS_MACOS || (line.libraryName().startsWith(QLatin1String("libsystem_")) && line.libraryName().endsWith(QLatin1String(".dylib"))) || line.libraryName().contains(QLatin1String("Foundation`")) #endif || line.libraryName().contains(QLatin1String("ntdll.dll")) || line.libraryName().contains(QLatin1String("kernel32.dll")) || line.functionName().contains(QLatin1String("_tmain")) || line.functionName() == QLatin1String("WinMain")) { return true; } return false; } static bool isFunctionUseful(const BacktraceLine & line) { //We need the function name if ( line.rating() == BacktraceLine::MissingEverything || line.rating() == BacktraceLine::MissingFunction ) { return false; } //Misc ignores if ( line.functionName() == QLatin1String("__kernel_vsyscall") || line.functionName() == QLatin1String("raise") || line.functionName() == QLatin1String("abort") || line.functionName() == QLatin1String("__libc_message") || line.functionName() == QLatin1String("thr_kill") /* *BSD */) { return false; } //Ignore core Qt functions //(QObject can be useful in some cases) if ( line.functionName().startsWith(QLatin1String("QBasicAtomicInt::")) || line.functionName().startsWith(QLatin1String("QBasicAtomicPointer::")) || line.functionName().startsWith(QLatin1String("QAtomicInt::")) || line.functionName().startsWith(QLatin1String("QAtomicPointer::")) || line.functionName().startsWith(QLatin1String("QMetaObject::")) || line.functionName().startsWith(QLatin1String("QPointer::")) || line.functionName().startsWith(QLatin1String("QWeakPointer::")) || line.functionName().startsWith(QLatin1String("QSharedPointer::")) || line.functionName().startsWith(QLatin1String("QScopedPointer::")) || line.functionName().startsWith(QLatin1String("QMetaCallEvent::")) ) { return false; } //Ignore core Qt containers misc functions if ( line.functionName().endsWith(QLatin1String("detach")) || line.functionName().endsWith(QLatin1String("detach_helper")) || line.functionName().endsWith(QLatin1String("node_create")) || line.functionName().endsWith(QLatin1String("deref")) || line.functionName().endsWith(QLatin1String("ref")) || line.functionName().endsWith(QLatin1String("node_copy")) || line.functionName().endsWith(QLatin1String("d_func")) ) { return false; } //Misc Qt stuff if ( line.functionName() == QLatin1String("qt_message_output") || line.functionName() == QLatin1String("qt_message") || line.functionName() == QLatin1String("qFatal") || line.functionName().startsWith(QLatin1String("qGetPtrHelper")) || line.functionName().startsWith(QLatin1String("qt_meta_")) ) { return false; } return true; } static bool isFunctionUsefulForSearch(const BacktraceLine & line) { //Ignore Qt containers (and iterators Q*Iterator) if ( line.functionName().startsWith(QLatin1String("QList")) || line.functionName().startsWith(QLatin1String("QLinkedList")) || line.functionName().startsWith(QLatin1String("QVector")) || line.functionName().startsWith(QLatin1String("QStack")) || line.functionName().startsWith(QLatin1String("QQueue")) || line.functionName().startsWith(QLatin1String("QSet")) || line.functionName().startsWith(QLatin1String("QMap")) || line.functionName().startsWith(QLatin1String("QMultiMap")) || line.functionName().startsWith(QLatin1String("QMapData")) || line.functionName().startsWith(QLatin1String("QHash")) || line.functionName().startsWith(QLatin1String("QMultiHash")) || line.functionName().startsWith(QLatin1String("QHashData")) ) { return false; } return true; } void BacktraceParser::calculateRatingData() { Q_D(BacktraceParser); uint rating = 0, bestPossibleRating = 0, counter = 0; bool haveSeenStackBase = false; QListIterator i(d->m_linesToRate); i.toBack(); //start from the end of the list while( i.hasPrevious() ) { const BacktraceLine & line = i.previous(); if ( !i.hasPrevious() && line.rating() == BacktraceLine::MissingEverything ) { //Under some circumstances, the very first stack frame is invalid (ex, calling a function //at an invalid address could result in a stack frame like "0x00000000 in ?? ()"), //which however does not necessarily mean that the backtrace has a missing symbol on //the first line. Here we make sure to ignore this line from rating. (bug 190882) break; //there are no more items anyway, just break the loop } if ( lineIsStackBase(line) ) { rating = bestPossibleRating = counter = 0; //restart rating ignoring any previous frames haveSeenStackBase = true; } else if ( lineIsStackTop(line) ) { break; //we have reached the top, no need to inspect any more frames } if ( lineShouldBeIgnored(line) ) { continue; } if ( line.rating() == BacktraceLine::MissingFunction || line.rating() == BacktraceLine::MissingSourceFile) { d->m_librariesWithMissingDebugSymbols.insert(line.libraryName().trimmed()); } uint multiplier = ++counter; //give weight to the first lines rating += static_cast(line.rating()) * multiplier; bestPossibleRating += static_cast(BacktraceLine::BestRating) * multiplier; qCDebug(DRKONQI_PARSER_LOG) << line.rating() << line.toString(); } //Generate a simplified backtrace //- Starts from the first useful function //- Max of 5 lines //- Replaces garbage with [...] //At the same time, grab the first three useful functions for search queries i.toFront(); //Reuse the list iterator int functionIndex = 0; int usefulFunctionsCount = 0; bool firstUsefulFound = false; while( i.hasNext() && functionIndex < 5 ) { const BacktraceLine & line = i.next(); if ( !lineShouldBeIgnored(line) && isFunctionUseful(line) ) { //Line is not garbage to use if (!firstUsefulFound) { firstUsefulFound = true; } //Save simplified backtrace line d->m_simplifiedBacktrace += line.toString(); //Fetch three useful functions (only functionName) for search queries if (usefulFunctionsCount < 3 && isFunctionUsefulForSearch(line) && !d->m_firstUsefulFunctions.contains(line.functionName())) { d->m_firstUsefulFunctions.append(line.functionName()); usefulFunctionsCount++; } functionIndex++; } else if (firstUsefulFound) { //Add "[...]" if there are invalid functions in the middle if (!d->m_simplifiedBacktrace.endsWith(QLatin1String("[...]\n"))) { d->m_simplifiedBacktrace += QLatin1String("[...]\n"); } } } //calculate rating d->m_usefulness = Useless; if (rating >= (bestPossibleRating*0.90)) { d->m_usefulness = ReallyUseful; } else if (rating >= (bestPossibleRating*0.70)) { d->m_usefulness = MayBeUseful; } else if (rating >= (bestPossibleRating*0.40)) { d->m_usefulness = ProbablyUseless; } //if there is no stack base, the executable is probably stripped, //so we need to be more strict with rating if ( !haveSeenStackBase ) { //less than 4 stack frames is useless if ( counter < 4 ) { d->m_usefulness = Useless; //more than 4 stack frames might have some value, so let's not be so strict, just lower the rating } else if ( d->m_usefulness > Useless ) { d->m_usefulness = (Usefulness) (d->m_usefulness - 1); } } qCDebug(DRKONQI_PARSER_LOG) << "Rating:" << rating << "out of" << bestPossibleRating << "Usefulness:" << staticMetaObject.enumerator(staticMetaObject.indexOfEnumerator("Usefulness")).valueToKey(d->m_usefulness); qCDebug(DRKONQI_PARSER_LOG) << "90%:" << (bestPossibleRating*0.90) << "70%:" << (bestPossibleRating*0.70) << "40%:" << (bestPossibleRating*0.40); qCDebug(DRKONQI_PARSER_LOG) << "Have seen stack base:" << haveSeenStackBase << "Lines counted:" << counter; }