/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "stackframe.h" #include "debuggerengine.h" #include "debuggerprotocol.h" #include "watchutils.h" #include #include #include #include using namespace Utils; namespace Debugger { namespace Internal { //////////////////////////////////////////////////////////////////////// // // StackFrame // //////////////////////////////////////////////////////////////////////// StackFrame::StackFrame() = default; void StackFrame::clear() { line = -1; function.clear(); file.clear(); module.clear(); receiver.clear(); address = 0; } bool StackFrame::isUsable() const { return usable; } QString StackFrame::toString() const { QString res; QTextStream str(&res); str << tr("Address:") << ' '; str.setIntegerBase(16); str << address; str.setIntegerBase(10); str << ' ' << tr("Function:") << ' ' << function << ' ' << tr("File:") << ' ' << file << ' ' << tr("Line:") << ' ' << line << ' ' << tr("From:") << ' ' << module << ' ' << tr("To:") << ' ' << receiver; return res; } QList StackFrame::parseFrames(const GdbMi &data, const DebuggerRunParameters &rp) { StackFrames frames; frames.reserve(data.childCount()); for (const GdbMi &item : data) frames.append(parseFrame(item, rp)); return frames; } StackFrame StackFrame::parseFrame(const GdbMi &frameMi, const DebuggerRunParameters &rp) { StackFrame frame; frame.level = frameMi["level"].data(); frame.function = frameMi["function"].data(); frame.module = frameMi["module"].data(); const FilePath debugger = rp.debugger.command.executable(); frame.file = FilePath::fromString(frameMi["file"].data()).onDevice(debugger); frame.line = frameMi["line"].toInt(); frame.address = frameMi["address"].toAddress(); frame.context = frameMi["context"].data(); if (frameMi["language"].data() == "js" || frame.file.endsWith(".js") || frame.file.endsWith(".qml")) { frame.language = QmlLanguage; frame.fixQrcFrame(rp); } GdbMi usable = frameMi["usable"]; if (usable.isValid()) frame.usable = usable.data().toInt(); else frame.usable = frame.file.isReadableFile(); return frame; } QString StackFrame::toToolTip() const { const QString filePath = file.toUserOutput(); QString res; QTextStream str(&res); str << ""; if (address) str << ""; if (!function.isEmpty()) str << ""; if (!file.isEmpty()) str << ""; if (line != -1) str << ""; if (!module.isEmpty()) str << ""; if (!receiver.isEmpty()) str << ""; str << "
" << tr("Address:") << "" << formatToolTipAddress(address) << "
" << (language == CppLanguage ? tr("Function:") : tr("JS-Function:")) << "" << function << "
" << tr("File:") << "" << filePath << "
" << tr("Line:") << "" << line << "
" << tr("Module:") << "" << module << "
" << tr("Receiver:") << "" << receiver << "
"; str <<"

" << tr("Note:") << " "; bool showDistributionNote = false; if (isUsable()) { str << tr("Sources for this frame are available.
Double-click on " "the file name to open an editor."); } else if (line <= 0) { str << tr("Binary debug information is not accessible for this " "frame. This either means the core was not compiled " "with debug information, or the debug information is not " "accessible."); showDistributionNote = true; } else { str << tr("Binary debug information is accessible for this " "frame. However, matching sources have not been found."); showDistributionNote = true; } if (file.osType() != OsTypeWindows && showDistributionNote) { str << ' ' << tr("Note that most distributions ship debug information " "in separate packages."); } str << ""; return res; } static FilePath findFile(const FilePath &baseDir, const FilePath &relativeFile) { for (FilePath dir(baseDir); !dir.isEmpty(); dir = dir.parentDir()) { const FilePath absolutePath = dir.resolvePath(relativeFile); if (absolutePath.isFile()) return absolutePath; } return {}; } // Try to resolve files coming from resource files. void StackFrame::fixQrcFrame(const DebuggerRunParameters &rp) { if (language != QmlLanguage) return; if (file.isAbsolutePath()) { usable = file.isFile(); return; } if (!file.startsWith("qrc:/")) return; FilePath relativeFile = file; QString relativePath = file.path(); relativePath = relativePath.right(relativePath.size() - 5); while (relativePath.startsWith('/')) relativePath = relativePath.mid(1); relativeFile.setPath(relativePath); FilePath absFile = findFile(rp.projectSourceDirectory, relativeFile); if (absFile.isEmpty()) absFile = findFile(FilePath::fromString(QDir::currentPath()), relativeFile); if (absFile.isEmpty()) return; file = absFile; usable = true; } QDebug operator<<(QDebug d, const StackFrame &f) { QString res; QTextStream str(&res); str << "level=" << f.level << " address=" << f.address; if (!f.function.isEmpty()) str << ' ' << f.function; if (!f.file.isEmpty()) str << ' ' << f.file << ':' << f.line; if (!f.module.isEmpty()) str << " from=" << f.module; if (!f.receiver.isEmpty()) str << " to=" << f.receiver; d.nospace() << res; return d; } } // namespace Internal } // namespace Debugger