forked from qt-creator/qt-creator
		
	
		
			
				
	
	
		
			2796 lines
		
	
	
		
			102 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			2796 lines
		
	
	
		
			102 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**************************************************************************
 | |
| **
 | |
| ** This file is part of Qt Creator
 | |
| **
 | |
| ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).
 | |
| **
 | |
| ** Contact: Nokia Corporation (info@qt.nokia.com)
 | |
| **
 | |
| **
 | |
| ** GNU Lesser General Public License Usage
 | |
| **
 | |
| ** This file may be used under the terms of the GNU Lesser General Public
 | |
| ** License version 2.1 as published by the Free Software Foundation and
 | |
| ** appearing in the file LICENSE.LGPL included in the packaging of this file.
 | |
| ** Please review the following information to ensure the GNU Lesser General
 | |
| ** Public License version 2.1 requirements will be met:
 | |
| ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
 | |
| **
 | |
| ** In addition, as a special exception, Nokia gives you certain additional
 | |
| ** rights. These rights are described in the Nokia Qt LGPL Exception
 | |
| ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
 | |
| **
 | |
| ** Other Usage
 | |
| **
 | |
| ** Alternatively, this file may be used in accordance with the terms and
 | |
| ** conditions contained in a signed written agreement between you and Nokia.
 | |
| **
 | |
| ** If you have questions regarding the use of this file, please contact
 | |
| ** Nokia at qt-info@nokia.com.
 | |
| **
 | |
| **************************************************************************/
 | |
| 
 | |
| #include "cdbengine.h"
 | |
| #include "debuggerstartparameters.h"
 | |
| #include "disassemblerlines.h"
 | |
| #include "cdboptions.h"
 | |
| #include "cdboptionspage.h"
 | |
| #include "bytearrayinputstream.h"
 | |
| #include "breakpoint.h"
 | |
| #include "breakhandler.h"
 | |
| #include "stackframe.h"
 | |
| #include "stackhandler.h"
 | |
| #include "watchhandler.h"
 | |
| #include "threadshandler.h"
 | |
| #include "moduleshandler.h"
 | |
| #include "debuggeractions.h"
 | |
| #include "debuggerinternalconstants.h"
 | |
| #include "debuggercore.h"
 | |
| #include "registerhandler.h"
 | |
| #include "disassembleragent.h"
 | |
| #include "memoryagent.h"
 | |
| #include "debuggerrunner.h"
 | |
| #include "debuggertooltipmanager.h"
 | |
| #include "cdbparsehelpers.h"
 | |
| #include "watchutils.h"
 | |
| #include "gdb/gdbmi.h"
 | |
| #include "shared/cdbsymbolpathlisteditor.h"
 | |
| 
 | |
| #include <coreplugin/icore.h>
 | |
| #include <texteditor/itexteditor.h>
 | |
| #include <projectexplorer/abi.h>
 | |
| #include <projectexplorer/projectexplorerconstants.h>
 | |
| 
 | |
| #include <utils/synchronousprocess.h>
 | |
| #include <utils/winutils.h>
 | |
| #include <utils/qtcassert.h>
 | |
| #include <utils/savedaction.h>
 | |
| #include <utils/consoleprocess.h>
 | |
| 
 | |
| #include <QtCore/QCoreApplication>
 | |
| #include <QtCore/QFileInfo>
 | |
| #include <QtCore/QDir>
 | |
| #include <QtCore/QDebug>
 | |
| #include <QtCore/QTextStream>
 | |
| #include <QtCore/QDateTime>
 | |
| #include <QtGui/QToolTip>
 | |
| #include <QtGui/QMainWindow>
 | |
| #include <QtGui/QMessageBox>
 | |
| 
 | |
| #ifdef Q_OS_WIN
 | |
| #    include <utils/winutils.h>
 | |
| #    include "dbgwinutils.h"
 | |
| #endif
 | |
| 
 | |
| #include <cctype>
 | |
| 
 | |
| Q_DECLARE_METATYPE(Debugger::Internal::DisassemblerAgent*)
 | |
| Q_DECLARE_METATYPE(Debugger::Internal::MemoryAgent*)
 | |
| 
 | |
| enum { debug = 0 };
 | |
| enum { debugLocals = 0 };
 | |
| enum { debugSourceMapping = 0 };
 | |
| enum { debugWatches = 0 };
 | |
| enum { debugBreakpoints = 0 };
 | |
| 
 | |
| #if 0
 | |
| #  define STATE_DEBUG(state, func, line, notifyFunc) qDebug("%s in %s at %s:%d", notifyFunc, stateName(state), func, line);
 | |
| #else
 | |
| #  define STATE_DEBUG(state, func, line, notifyFunc)
 | |
| #endif
 | |
| 
 | |
| /*!
 | |
|     \class Debugger::Internal::CdbEngine
 | |
| 
 | |
|     Cdb engine version 2: Run the CDB process on pipes and parse its output.
 | |
|     The engine relies on a CDB extension Qt Creator provides as an extension
 | |
|     library (32/64bit), which is loaded into cdb.exe. It serves to:
 | |
| 
 | |
|     \list
 | |
|     \o Notify the engine about the state of the debugging session:
 | |
|         \list
 | |
|         \o idle: (hooked up with .idle_cmd) debuggee stopped
 | |
|         \o accessible: Debuggee stopped, cdb.exe accepts commands
 | |
|         \o inaccessible: Debuggee runs, no way to post commands
 | |
|         \o session active/inactive: Lost debuggee, terminating.
 | |
|         \endlist
 | |
|     \o Hook up with output/event callbacks and produce formatted output to be able
 | |
|        to catch application output and exceptions.
 | |
|     \o Provide some extension commands that produce output in a standardized (GDBMI)
 | |
|       format that ends up in handleExtensionMessage(), for example:
 | |
|       \list
 | |
|       \o pid     Return debuggee pid for interrupting.
 | |
|       \o locals  Print locals from SymbolGroup
 | |
|       \o expandLocals Expand locals in symbol group
 | |
|       \o registers, modules, threads
 | |
|       \endlist
 | |
|    \endlist
 | |
| 
 | |
|    Debugger commands can be posted by calling:
 | |
| 
 | |
|    \list
 | |
| 
 | |
|     \o postCommand(): Does not expect a reply
 | |
|     \o postBuiltinCommand(): Run a builtin-command producing free-format, multiline output
 | |
|        that is captured by enclosing it in special tokens using the 'echo' command and
 | |
|        then invokes a callback with a CdbBuiltinCommand structure.
 | |
|     \o postExtensionCommand(): Run a command provided by the extension producing
 | |
|        one-line output and invoke a callback with a CdbExtensionCommand structure
 | |
|        (output is potentially split up in chunks).
 | |
|     \endlist
 | |
| 
 | |
| 
 | |
|     Startup sequence:
 | |
|     [Console: The console stub launches the process. On process startup,
 | |
|               launchCDB() is called with AttachExternal].
 | |
|     setupEngine() calls launchCDB() with the startparameters. The debuggee
 | |
|     runs into the initial breakpoint (session idle). EngineSetupOk is
 | |
|     notified (inferior still stopped). setupInferior() is then called
 | |
|     which does breakpoint synchronization and issues the extension 'pid'
 | |
|     command to obtain the inferior pid (which also hooks up the output callbacks).
 | |
|      handlePid() notifies notifyInferiorSetupOk.
 | |
|     runEngine() is then called which issues 'g' to continue the inferior.
 | |
|     Shutdown mostly uses notifyEngineSpontaneousShutdown() as cdb just quits
 | |
|     when the inferior exits (except attach modes).
 | |
| */
 | |
| 
 | |
| using namespace ProjectExplorer;
 | |
| 
 | |
| namespace Debugger {
 | |
| namespace Internal {
 | |
| 
 | |
| static const char localsPrefixC[] = "local.";
 | |
| 
 | |
| struct MemoryViewCookie
 | |
| {
 | |
|     explicit MemoryViewCookie(MemoryAgent *a = 0, QObject *e = 0,
 | |
|                               quint64 addr = 0, quint64 l = 0) :
 | |
|         agent(a), editorToken(e), address(addr), length(l)
 | |
|     {}
 | |
| 
 | |
|     MemoryAgent *agent;
 | |
|     QObject *editorToken;
 | |
|     quint64 address;
 | |
|     quint64 length;
 | |
| };
 | |
| 
 | |
| struct MemoryChangeCookie
 | |
| {
 | |
|     explicit MemoryChangeCookie(quint64 addr = 0, const QByteArray &d = QByteArray()) :
 | |
|                                address(addr), data(d) {}
 | |
| 
 | |
|     quint64 address;
 | |
|     QByteArray data;
 | |
| };
 | |
| 
 | |
| struct ConditionalBreakPointCookie
 | |
| {
 | |
|     ConditionalBreakPointCookie(BreakpointId i = 0) : id(i) {}
 | |
|     BreakpointId id;
 | |
|     GdbMi stopReason;
 | |
| };
 | |
| 
 | |
| } // namespace Internal
 | |
| } // namespace Debugger
 | |
| 
 | |
| Q_DECLARE_METATYPE(Debugger::Internal::MemoryViewCookie)
 | |
| Q_DECLARE_METATYPE(Debugger::Internal::MemoryChangeCookie)
 | |
| Q_DECLARE_METATYPE(Debugger::Internal::ConditionalBreakPointCookie)
 | |
| 
 | |
| namespace Debugger {
 | |
| namespace Internal {
 | |
| 
 | |
| static inline bool isConsole(const DebuggerStartParameters &sp)
 | |
| {
 | |
|     return (sp.startMode == StartInternal || sp.startMode == StartExternal)
 | |
|         && sp.useTerminal;
 | |
| }
 | |
| 
 | |
| static QMessageBox *
 | |
| nonModalMessageBox(QMessageBox::Icon icon, const QString &title, const QString &text)
 | |
| {
 | |
|     QMessageBox *mb = new QMessageBox(icon, title, text, QMessageBox::Ok,
 | |
|                                       debuggerCore()->mainWindow());
 | |
|     mb->setAttribute(Qt::WA_DeleteOnClose);
 | |
|     mb->show();
 | |
|     return mb;
 | |
| }
 | |
| 
 | |
| // Base data structure for command queue entries with callback
 | |
| struct CdbCommandBase
 | |
| {
 | |
|     typedef CdbEngine::BuiltinCommandHandler CommandHandler;
 | |
| 
 | |
|     CdbCommandBase();
 | |
|     CdbCommandBase(const QByteArray  &cmd, int token, unsigned flags,
 | |
|                    unsigned nc, const QVariant &cookie);
 | |
| 
 | |
|     int token;
 | |
|     unsigned flags;
 | |
|     QByteArray command;
 | |
|     QVariant cookie;
 | |
|     // Continue with another commands as specified in CommandSequenceFlags
 | |
|     unsigned commandSequence;
 | |
| };
 | |
| 
 | |
| CdbCommandBase::CdbCommandBase() :
 | |
|     token(0), flags(0), commandSequence(0)
 | |
| {
 | |
| }
 | |
| 
 | |
| CdbCommandBase::CdbCommandBase(const QByteArray  &cmd, int t, unsigned f,
 | |
|                                unsigned nc, const QVariant &c) :
 | |
|     token(t), flags(f), command(cmd), cookie(c), commandSequence(nc)
 | |
| {
 | |
| }
 | |
| 
 | |
| // Queue entry for builtin commands producing free-format
 | |
| // line-by-line output.
 | |
| struct CdbBuiltinCommand : public CdbCommandBase
 | |
| {
 | |
|     typedef CdbEngine::BuiltinCommandHandler CommandHandler;
 | |
| 
 | |
|     CdbBuiltinCommand() {}
 | |
|     CdbBuiltinCommand(const QByteArray  &cmd, int token, unsigned flags,
 | |
|                       CommandHandler h,
 | |
|                       unsigned nc, const QVariant &cookie) :
 | |
|         CdbCommandBase(cmd, token, flags, nc, cookie), handler(h)
 | |
|     {}
 | |
| 
 | |
| 
 | |
|     QByteArray joinedReply() const;
 | |
| 
 | |
|     CommandHandler handler;
 | |
|     QList<QByteArray> reply;
 | |
| };
 | |
| 
 | |
| QByteArray CdbBuiltinCommand::joinedReply() const
 | |
| {
 | |
|     if (reply.isEmpty())
 | |
|         return QByteArray();
 | |
|     QByteArray answer;
 | |
|     answer.reserve(120  * reply.size());
 | |
|     foreach (const QByteArray &l, reply) {
 | |
|         answer += l;
 | |
|         answer += '\n';
 | |
|     }
 | |
|     return answer;
 | |
| }
 | |
| 
 | |
| // Queue entry for Qt Creator extension commands producing one-line
 | |
| // output with success flag and error message.
 | |
| struct CdbExtensionCommand : public CdbCommandBase
 | |
| {
 | |
|     typedef CdbEngine::ExtensionCommandHandler CommandHandler;
 | |
| 
 | |
|     CdbExtensionCommand() : success(false) {}
 | |
|     CdbExtensionCommand(const QByteArray  &cmd, int token, unsigned flags,
 | |
|                       CommandHandler h,
 | |
|                       unsigned nc, const QVariant &cookie) :
 | |
|         CdbCommandBase(cmd, token, flags, nc, cookie), handler(h),success(false) {}
 | |
| 
 | |
|     CommandHandler handler;
 | |
|     QByteArray reply;
 | |
|     QByteArray errorMessage;
 | |
|     bool success;
 | |
| };
 | |
| 
 | |
| template <class CommandPtrType>
 | |
| int indexOfCommand(const QList<CommandPtrType> &l, int token)
 | |
| {
 | |
|     const int count = l.size();
 | |
|     for (int i = 0; i < count; i++)
 | |
|         if (l.at(i)->token == token)
 | |
|             return i;
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| static inline bool validMode(DebuggerStartMode sm)
 | |
| {
 | |
|     switch (sm) {
 | |
|     case NoStartMode:
 | |
|     case AttachCore:
 | |
|     case StartRemoteGdb:
 | |
|         return false;
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| // Accessed by RunControlFactory
 | |
| DebuggerEngine *createCdbEngine(const DebuggerStartParameters &sp,
 | |
|     DebuggerEngine *masterEngine, QString *errorMessage)
 | |
| {
 | |
| #ifdef Q_OS_WIN
 | |
|     CdbOptionsPage *op = CdbOptionsPage::instance();
 | |
|     if (!op || !op->options()->isValid() || !validMode(sp.startMode)) {
 | |
|         *errorMessage = QLatin1String("Internal error: Invalid start parameters passed for thre CDB engine.");
 | |
|         return 0;
 | |
|     }
 | |
|     return new CdbEngine(sp, masterEngine, op->options());
 | |
| #else
 | |
|     Q_UNUSED(masterEngine)
 | |
|     Q_UNUSED(sp)
 | |
| #endif
 | |
|     *errorMessage = QString::fromLatin1("Unsupported debug mode");
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| bool isCdbEngineEnabled()
 | |
| {
 | |
| #ifdef Q_OS_WIN
 | |
|     return CdbOptionsPage::instance() && CdbOptionsPage::instance()->options()->isValid();
 | |
| #else
 | |
|     return false;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| static inline QString msgNoCdbBinaryForToolChain(const ProjectExplorer::Abi &tc)
 | |
| {
 | |
|     return CdbEngine::tr("There is no CDB binary available for binaries in format '%1'").arg(tc.toString());
 | |
| }
 | |
| 
 | |
| static QString cdbBinary(const DebuggerStartParameters &sp)
 | |
| {
 | |
|     if (!sp.debuggerCommand.isEmpty()) {
 | |
|         // Do not use a GDB binary if we got started for a project with MinGW runtime.
 | |
|         const bool abiMatch = sp.toolChainAbi.os() == ProjectExplorer::Abi::WindowsOS
 | |
|                     && (sp.toolChainAbi.osFlavor() == ProjectExplorer::Abi::WindowsMsvc2005Flavor
 | |
|                         || sp.toolChainAbi.osFlavor() == ProjectExplorer::Abi::WindowsMsvc2008Flavor
 | |
|                         || sp.toolChainAbi.osFlavor() == ProjectExplorer::Abi::WindowsMsvc2010Flavor);
 | |
|         if (abiMatch)
 | |
|             return sp.debuggerCommand;
 | |
|     }
 | |
|     return debuggerCore()->debuggerForAbi(sp.toolChainAbi, CdbEngineType);
 | |
| }
 | |
| 
 | |
| bool checkCdbConfiguration(const DebuggerStartParameters &sp, ConfigurationCheck *check)
 | |
| {
 | |
| #ifdef Q_OS_WIN
 | |
|     if (!isCdbEngineEnabled()) {
 | |
|         check->errorDetails.push_back(CdbEngine::tr("The CDB debug engine required for %1 is currently disabled.").
 | |
|                            arg(sp.toolChainAbi.toString()));
 | |
|         check->settingsCategory = QLatin1String(Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY);
 | |
|         check->settingsPage = CdbOptionsPage::settingsId();
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (!validMode(sp.startMode)) {
 | |
|         check->errorDetails.push_back(CdbEngine::tr("The CDB engine does not support start mode %1.").arg(sp.startMode));
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (sp.toolChainAbi.binaryFormat() != Abi::PEFormat || sp.toolChainAbi.os() != Abi::WindowsOS) {
 | |
|         check->errorDetails.push_back(CdbEngine::tr("The CDB debug engine does not support the %1 ABI.").
 | |
|                                       arg(sp.toolChainAbi.toString()));
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     if (cdbBinary(sp).isEmpty()) {
 | |
|         check->errorDetails.push_back(msgNoCdbBinaryForToolChain(sp.toolChainAbi));
 | |
|         check->settingsCategory = QLatin1String(ProjectExplorer::Constants::TOOLCHAIN_SETTINGS_CATEGORY);
 | |
|         check->settingsPage = QLatin1String(ProjectExplorer::Constants::TOOLCHAIN_SETTINGS_CATEGORY);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| #else
 | |
|     Q_UNUSED(sp);
 | |
|     check->errorDetails.push_back(QString::fromLatin1("Unsupported debug mode"));
 | |
|     return false;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void addCdbOptionPages(QList<Core::IOptionsPage *> *opts)
 | |
| {
 | |
| #ifdef Q_OS_WIN
 | |
|     opts->push_back(new CdbOptionsPage);
 | |
| #else
 | |
|     Q_UNUSED(opts);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| #define QT_CREATOR_CDB_EXT "qtcreatorcdbext"
 | |
| 
 | |
| static inline Utils::SavedAction *theAssemblerAction()
 | |
| {
 | |
|     return debuggerCore()->action(OperateByInstruction);
 | |
| }
 | |
| 
 | |
| CdbEngine::CdbEngine(const DebuggerStartParameters &sp,
 | |
|         DebuggerEngine *masterEngine, const OptionsPtr &options) :
 | |
|     DebuggerEngine(sp, masterEngine),
 | |
|     m_creatorExtPrefix("<qtcreatorcdbext>|"),
 | |
|     m_tokenPrefix("<token>"),
 | |
|     m_options(options),
 | |
|     m_effectiveStartMode(NoStartMode),
 | |
|     m_accessible(false),
 | |
|     m_specialStopMode(NoSpecialStop),
 | |
|     m_nextCommandToken(0),
 | |
|     m_currentBuiltinCommandIndex(-1),
 | |
|     m_extensionCommandPrefixBA("!"QT_CREATOR_CDB_EXT"."),
 | |
|     m_operateByInstructionPending(true),
 | |
|     m_operateByInstruction(true), // Default CDB setting
 | |
|     m_notifyEngineShutdownOnTermination(false),
 | |
|     m_hasDebuggee(false),
 | |
|     m_elapsedLogTime(0),
 | |
|     m_sourceStepInto(false),
 | |
|     m_watchPointX(0),
 | |
|     m_watchPointY(0),
 | |
|     m_ignoreCdbOutput(false)
 | |
| {
 | |
|     connect(theAssemblerAction(), SIGNAL(triggered(bool)), this, SLOT(operateByInstructionTriggered(bool)));
 | |
| 
 | |
|     setObjectName(QLatin1String("CdbEngine"));
 | |
|     connect(&m_process, SIGNAL(finished(int)), this, SLOT(processFinished()));
 | |
|     connect(&m_process, SIGNAL(error(QProcess::ProcessError)), this, SLOT(processError()));
 | |
|     connect(&m_process, SIGNAL(readyReadStandardOutput()), this, SLOT(readyReadStandardOut()));
 | |
|     connect(&m_process, SIGNAL(readyReadStandardError()), this, SLOT(readyReadStandardOut()));
 | |
| }
 | |
| 
 | |
| void CdbEngine::init()
 | |
| {
 | |
|     m_effectiveStartMode = NoStartMode;
 | |
|     notifyInferiorPid(0);
 | |
|     m_accessible = false;
 | |
|     m_specialStopMode = NoSpecialStop;
 | |
|     m_nextCommandToken  = 0;
 | |
|     m_currentBuiltinCommandIndex = -1;
 | |
|     m_operateByInstructionPending = theAssemblerAction()->isChecked();
 | |
|     m_operateByInstruction = true; // Default CDB setting
 | |
|     m_notifyEngineShutdownOnTermination = false;
 | |
|     m_hasDebuggee = false;
 | |
|     m_sourceStepInto = false;
 | |
|     m_watchPointX = m_watchPointY = 0;
 | |
|     m_ignoreCdbOutput = false;
 | |
| 
 | |
|     m_outputBuffer.clear();
 | |
|     m_builtinCommandQueue.clear();
 | |
|     m_extensionCommandQueue.clear();
 | |
|     m_extensionMessageBuffer.clear();
 | |
|     m_pendingBreakpointMap.clear();
 | |
|     m_customSpecialStopData.clear();
 | |
| 
 | |
|     // Create local list of mappings in native separators
 | |
|     m_sourcePathMappings.clear();
 | |
|     const QSharedPointer<GlobalDebuggerOptions> globalOptions = debuggerCore()->globalDebuggerOptions();
 | |
|     if (!globalOptions->sourcePathMap.isEmpty()) {
 | |
|         typedef GlobalDebuggerOptions::SourcePathMap::const_iterator SourcePathMapIterator;
 | |
|         m_sourcePathMappings.reserve(globalOptions->sourcePathMap.size());
 | |
|         const SourcePathMapIterator cend = globalOptions->sourcePathMap.constEnd();
 | |
|         for (SourcePathMapIterator it = globalOptions->sourcePathMap.constBegin(); it != cend; ++it) {
 | |
|             m_sourcePathMappings.push_back(SourcePathMapping(QDir::toNativeSeparators(it.key()),
 | |
|                                                              QDir::toNativeSeparators(it.value())));
 | |
|         }
 | |
|     }
 | |
|     QTC_ASSERT(m_process.state() != QProcess::Running, Utils::SynchronousProcess::stopProcess(m_process); )
 | |
| }
 | |
| 
 | |
| CdbEngine::~CdbEngine()
 | |
| {
 | |
| }
 | |
| 
 | |
| void CdbEngine::operateByInstructionTriggered(bool operateByInstruction)
 | |
| {
 | |
|     // To be set next time session becomes accessible
 | |
|     m_operateByInstructionPending = operateByInstruction;
 | |
|     if (state() == InferiorStopOk)
 | |
|         syncOperateByInstruction(operateByInstruction);
 | |
| }
 | |
| 
 | |
| void CdbEngine::syncOperateByInstruction(bool operateByInstruction)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("syncOperateByInstruction current: %d new %d", m_operateByInstruction, operateByInstruction);
 | |
|     if (m_operateByInstruction == operateByInstruction)
 | |
|         return;
 | |
|     QTC_ASSERT(m_accessible, return; )
 | |
|     m_operateByInstruction = operateByInstruction;
 | |
|     postCommand(m_operateByInstruction ? QByteArray("l-t") : QByteArray("l+t"), 0);
 | |
|     postCommand(m_operateByInstruction ? QByteArray("l-s") : QByteArray("l+s"), 0);
 | |
| }
 | |
| 
 | |
| bool CdbEngine::setToolTipExpression(const QPoint &mousePos,
 | |
|                                      TextEditor::ITextEditor *editor,
 | |
|                                      const DebuggerToolTipContext &contextIn)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug() << Q_FUNC_INFO;
 | |
|     // Need a stopped debuggee and a cpp file in a valid frame
 | |
|     if (state() != InferiorStopOk || !isCppEditor(editor) || stackHandler()->currentIndex() < 0)
 | |
|         return false;
 | |
|     // Determine expression and function
 | |
|     int line;
 | |
|     int column;
 | |
|     DebuggerToolTipContext context = contextIn;
 | |
|     QString exp = cppExpressionAt(editor, context.position, &line, &column, &context.function);
 | |
|     // Are we in the current stack frame
 | |
|     if (context.function.isEmpty() || exp.isEmpty() || context.function != stackHandler()->currentFrame().function)
 | |
|         return false;
 | |
|     // No numerical or any other expressions [yet]
 | |
|     if (!(exp.at(0).isLetter() || exp.at(0) == QLatin1Char('_')))
 | |
|         return false;
 | |
|     // Can this be found as a local variable?
 | |
|     const QByteArray localsPrefix(localsPrefixC);
 | |
|     QByteArray iname = localsPrefix + exp.toAscii();
 | |
|     QModelIndex index = watchHandler()->itemIndex(iname);
 | |
|     if (!index.isValid()) {
 | |
|         // Nope, try a 'local.this.m_foo'.
 | |
|         exp.prepend(QLatin1String("this."));
 | |
|         iname.insert(localsPrefix.size(), "this.");
 | |
|         index = watchHandler()->itemIndex(iname);
 | |
|         if (!index.isValid())
 | |
|             return false;
 | |
|     }
 | |
|     DebuggerTreeViewToolTipWidget *tw = new DebuggerTreeViewToolTipWidget;
 | |
|     tw->setContext(context);
 | |
|     tw->setDebuggerModel(LocalsWatch);
 | |
|     tw->setExpression(exp);
 | |
|     tw->acquireEngine(this);
 | |
|     DebuggerToolTipManager::instance()->showToolTip(mousePos, editor, tw);
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| // Determine full path to the CDB extension library.
 | |
| QString CdbEngine::extensionLibraryName(bool is64Bit)
 | |
| {
 | |
|     // Determine extension lib name and path to use
 | |
|     QString rc;
 | |
|     QTextStream(&rc) << QFileInfo(QCoreApplication::applicationDirPath()).path()
 | |
|                      << "/lib/" << (is64Bit ? QT_CREATOR_CDB_EXT"64" : QT_CREATOR_CDB_EXT"32")
 | |
|                      << '/' << QT_CREATOR_CDB_EXT << ".dll";
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| // Determine environment for CDB.exe, start out with run config and
 | |
| // add CDB extension path merged with system value should there be one.
 | |
| static QStringList mergeEnvironment(QStringList runConfigEnvironment,
 | |
|                                     QString cdbExtensionPath)
 | |
| {
 | |
|     // Determine CDB extension path from Qt Creator
 | |
|     static const char cdbExtensionPathVariableC[] = "_NT_DEBUGGER_EXTENSION_PATH";
 | |
|     const QByteArray oldCdbExtensionPath = qgetenv(cdbExtensionPathVariableC);
 | |
|     if (!oldCdbExtensionPath.isEmpty()) {
 | |
|         cdbExtensionPath.append(QLatin1Char(';'));
 | |
|         cdbExtensionPath.append(QString::fromLocal8Bit(oldCdbExtensionPath));
 | |
|     }
 | |
|     // We do not assume someone sets _NT_DEBUGGER_EXTENSION_PATH in the run
 | |
|     // config, just to make sure, delete any existing entries
 | |
|     const QString cdbExtensionPathVariableAssign =
 | |
|             QLatin1String(cdbExtensionPathVariableC) + QLatin1Char('=');
 | |
|     for (QStringList::iterator it = runConfigEnvironment.begin(); it != runConfigEnvironment.end() ; ) {
 | |
|         if (it->startsWith(cdbExtensionPathVariableAssign)) {
 | |
|             it = runConfigEnvironment.erase(it);
 | |
|             break;
 | |
|         } else {
 | |
|             ++it;
 | |
|         }
 | |
|     }
 | |
|     runConfigEnvironment.append(cdbExtensionPathVariableAssign +
 | |
|                                 QDir::toNativeSeparators(cdbExtensionPath));
 | |
|     return runConfigEnvironment;
 | |
| }
 | |
| 
 | |
| int CdbEngine::elapsedLogTime() const
 | |
| {
 | |
|     const int elapsed = m_logTime.elapsed();
 | |
|     const int delta = elapsed - m_elapsedLogTime;
 | |
|     m_elapsedLogTime = elapsed;
 | |
|     return delta;
 | |
| }
 | |
| 
 | |
| // Start the console stub with the sub process. Continue in consoleStubProcessStarted.
 | |
| bool CdbEngine::startConsole(const DebuggerStartParameters &sp, QString *errorMessage)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("startConsole %s", qPrintable(sp.executable));
 | |
|     m_consoleStub.reset(new Utils::ConsoleProcess);
 | |
|     m_consoleStub->setMode(Utils::ConsoleProcess::Suspend);
 | |
|     connect(m_consoleStub.data(), SIGNAL(processMessage(QString, bool)),
 | |
|             SLOT(consoleStubMessage(QString, bool)));
 | |
|     connect(m_consoleStub.data(), SIGNAL(processStarted()),
 | |
|             SLOT(consoleStubProcessStarted()));
 | |
|     connect(m_consoleStub.data(), SIGNAL(wrapperStopped()),
 | |
|             SLOT(consoleStubExited()));
 | |
|     m_consoleStub->setWorkingDirectory(sp.workingDirectory);
 | |
|     if (sp.environment.size())
 | |
|         m_consoleStub->setEnvironment(sp.environment);
 | |
|     if (!m_consoleStub->start(sp.executable, sp.processArgs)) {
 | |
|         *errorMessage = tr("The console process '%1' could not be started.").arg(sp.executable);
 | |
|         return false;
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void CdbEngine::consoleStubMessage(const QString &msg, bool isError)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("consoleStubProcessMessage() in %s error=%d %s", stateName(state()), isError, qPrintable(msg));
 | |
|     if (isError) {
 | |
|         if (state() == EngineSetupRequested) {
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupFailed")
 | |
|             notifyEngineSetupFailed();
 | |
|         } else {
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineIll")
 | |
|             notifyEngineIll();
 | |
|         }
 | |
|         nonModalMessageBox(QMessageBox::Critical, tr("Debugger Error"), msg);
 | |
|     } else {
 | |
|         showMessage(msg, AppOutput);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::consoleStubProcessStarted()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("consoleStubProcessStarted() PID=%lld", m_consoleStub->applicationPID());
 | |
|     // Attach to console process.
 | |
|     DebuggerStartParameters attachParameters = startParameters();
 | |
|     attachParameters.executable.clear();
 | |
|     attachParameters.processArgs.clear();
 | |
|     attachParameters.attachPID = m_consoleStub->applicationPID();
 | |
|     attachParameters.startMode = AttachExternal;
 | |
|     showMessage(QString::fromLatin1("Attaching to %1...").arg(attachParameters.attachPID), LogMisc);
 | |
|     QString errorMessage;
 | |
|     if (!launchCDB(attachParameters, &errorMessage)) {
 | |
|         showMessage(errorMessage, LogError);
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupFailed")
 | |
|         notifyEngineSetupFailed();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::consoleStubExited()
 | |
| {
 | |
| }
 | |
| 
 | |
| void CdbEngine::setupEngine()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug(">setupEngine");
 | |
|     // Nag to add symbol server
 | |
|     if (CdbSymbolPathListEditor::promptToAddSymbolServer(CdbOptions::settingsGroup(),
 | |
|                                                          &(m_options->symbolPaths)))
 | |
|         m_options->toSettings(Core::ICore::instance()->settings());
 | |
| 
 | |
|     init();
 | |
|     if (!m_logTime.elapsed())
 | |
|         m_logTime.start();
 | |
|     QString errorMessage;
 | |
|     // Console: Launch the stub with the suspended application and attach to it
 | |
|     // CDB in theory has a command line option '-2' that launches a
 | |
|     // console, too, but that immediately closes when the debuggee quits.
 | |
|     // Use the Creator stub instead.
 | |
|     const DebuggerStartParameters &sp = startParameters();
 | |
|     const bool launchConsole = isConsole(sp);
 | |
|     m_effectiveStartMode = launchConsole ? AttachExternal : sp.startMode;
 | |
|     const bool ok = launchConsole ?
 | |
|                 startConsole(startParameters(), &errorMessage) :
 | |
|                 launchCDB(startParameters(), &errorMessage);
 | |
|     if (debug)
 | |
|         qDebug("<setupEngine ok=%d", ok);
 | |
|     if (!ok) {
 | |
|         showMessage(errorMessage, LogError);
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupFailed")
 | |
|         notifyEngineSetupFailed();
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool CdbEngine::launchCDB(const DebuggerStartParameters &sp, QString *errorMessage)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("launchCDB startMode=%d", sp.startMode);
 | |
|     const QChar blank(QLatin1Char(' '));
 | |
|     // Start engine which will run until initial breakpoint:
 | |
|     // Determine binary (force MSVC), extension lib name and path to use
 | |
|     // The extension is passed as relative name with the path variable set
 | |
|     //(does not work with absolute path names)
 | |
|     const QString executable = cdbBinary(sp);
 | |
|     if (executable.isEmpty()) {
 | |
|         *errorMessage = tr("There is no CDB executable specified.");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     const bool is64bit =
 | |
| #ifdef Q_OS_WIN
 | |
|             Utils::winIs64BitBinary(executable);
 | |
| #else
 | |
|             false;
 | |
| #endif
 | |
|     const QFileInfo extensionFi(CdbEngine::extensionLibraryName(is64bit));
 | |
|     if (!extensionFi.isFile()) {
 | |
|         *errorMessage = QString::fromLatin1("Internal error: The extension %1 cannot be found.").
 | |
|                 arg(QDir::toNativeSeparators(extensionFi.absoluteFilePath()));
 | |
|         return false;
 | |
|     }
 | |
|     const QString extensionFileName = extensionFi.fileName();
 | |
|     // Prepare arguments
 | |
|     QStringList arguments;
 | |
|     const bool isRemote = sp.startMode == AttachToRemote;
 | |
|     if (isRemote) { // Must be first
 | |
|         arguments << QLatin1String("-remote") << sp.remoteChannel;
 | |
|     } else {
 | |
|         arguments << (QLatin1String("-a") + extensionFileName);
 | |
|     }
 | |
|     // Source line info/No terminal breakpoint / Pull extension
 | |
|     arguments << QLatin1String("-lines") << QLatin1String("-G")
 | |
|     // register idle (debuggee stop) notification
 | |
|               << QLatin1String("-c")
 | |
|               << QLatin1String(".idle_cmd ") + QString::fromAscii(m_extensionCommandPrefixBA) + QLatin1String("idle");
 | |
|     if (sp.useTerminal) // Separate console
 | |
|         arguments << QLatin1String("-2");
 | |
|     if (!m_options->symbolPaths.isEmpty())
 | |
|         arguments << QLatin1String("-y") << m_options->symbolPaths.join(QString(QLatin1Char(';')));
 | |
|     if (!m_options->sourcePaths.isEmpty())
 | |
|         arguments << QLatin1String("-srcpath") << m_options->sourcePaths.join(QString(QLatin1Char(';')));
 | |
|     // Compile argument string preserving quotes
 | |
|     QString nativeArguments = m_options->additionalArguments;
 | |
|     switch (sp.startMode) {
 | |
|     case StartInternal:
 | |
|     case StartExternal:
 | |
|         if (!nativeArguments.isEmpty())
 | |
|             nativeArguments.push_back(blank);
 | |
|         nativeArguments += QDir::toNativeSeparators(sp.executable);
 | |
|         break;
 | |
|     case AttachToRemote:
 | |
|         break;
 | |
|     case AttachExternal:
 | |
|     case AttachCrashedExternal:
 | |
|         arguments << QLatin1String("-p") << QString::number(sp.attachPID);
 | |
|         if (sp.startMode == AttachCrashedExternal)
 | |
|             arguments << QLatin1String("-e") << sp.crashParameter << QLatin1String("-g");
 | |
|         break;
 | |
|     default:
 | |
|         *errorMessage = QString::fromLatin1("Internal error: Unsupported start mode %1.").arg(sp.startMode);
 | |
|         return false;
 | |
|     }
 | |
|     if (!sp.processArgs.isEmpty()) { // Complete native argument string.
 | |
|         if (!nativeArguments.isEmpty())
 | |
|             nativeArguments.push_back(blank);
 | |
|         nativeArguments += sp.processArgs;
 | |
|     }
 | |
| 
 | |
|     const QString msg = QString::fromLatin1("Launching %1 %2\nusing %3 of %4.").
 | |
|             arg(QDir::toNativeSeparators(executable),
 | |
|                 arguments.join(QString(blank)) + blank + nativeArguments,
 | |
|                 QDir::toNativeSeparators(extensionFi.absoluteFilePath()),
 | |
|                 extensionFi.lastModified().toString(Qt::SystemLocaleShortDate));
 | |
|     showMessage(msg, LogMisc);
 | |
| 
 | |
|     m_outputBuffer.clear();
 | |
|     const QStringList environment = sp.environment.size() == 0 ?
 | |
|                                     QProcessEnvironment::systemEnvironment().toStringList() :
 | |
|                                     sp.environment.toStringList();
 | |
|     m_process.setEnvironment(mergeEnvironment(environment, extensionFi.absolutePath()));
 | |
|     if (!sp.workingDirectory.isEmpty())
 | |
|         m_process.setWorkingDirectory(sp.workingDirectory);
 | |
| 
 | |
| #ifdef Q_OS_WIN
 | |
|     if (!nativeArguments.isEmpty()) // Appends
 | |
|         m_process.setNativeArguments(nativeArguments);
 | |
| #endif
 | |
|     m_process.start(executable, arguments);
 | |
|     if (!m_process.waitForStarted()) {
 | |
|         *errorMessage = QString::fromLatin1("Internal error: Cannot start process %1: %2").
 | |
|                 arg(QDir::toNativeSeparators(executable), m_process.errorString());
 | |
|         return false;
 | |
|     }
 | |
| #ifdef Q_OS_WIN
 | |
|     const unsigned long pid = Utils::winQPidToPid(m_process.pid());
 | |
| #else
 | |
|     const unsigned long pid = 0;
 | |
| #endif
 | |
|     showMessage(QString::fromLatin1("%1 running as %2").
 | |
|                 arg(QDir::toNativeSeparators(executable)).arg(pid), LogMisc);
 | |
|     m_hasDebuggee = true;
 | |
|     if (isRemote) { // We do not get an 'idle' in a remote session, but are accessible
 | |
|         m_accessible = true;
 | |
|         const QByteArray loadCommand = QByteArray(".load ")
 | |
|                 + extensionFileName.toLocal8Bit();
 | |
|         postCommand(loadCommand, 0);
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupOk")
 | |
|         notifyEngineSetupOk();
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void CdbEngine::setupInferior()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("setupInferior");
 | |
|     attemptBreakpointSynchronization();
 | |
|     postCommand("sxn 0x4000001f", 0); // Do not break on WowX86 exceptions.
 | |
|     postCommand(".asm source_line", 0); // Source line in assembly
 | |
|     postExtensionCommand("pid", QByteArray(), 0, &CdbEngine::handlePid);
 | |
| }
 | |
| 
 | |
| void CdbEngine::runEngine()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("runEngine");
 | |
|     // Resume the threads frozen by the console stub.
 | |
|     if (isConsole(startParameters()))
 | |
|         postCommand("~* m", 0);
 | |
|     foreach (const QString &breakEvent, m_options->breakEvents)
 | |
|             postCommand(QByteArray("sxe ") + breakEvent.toAscii(), 0);
 | |
|     postCommand("g", 0);
 | |
| }
 | |
| 
 | |
| bool CdbEngine::commandsPending() const
 | |
| {
 | |
|     return !m_builtinCommandQueue.isEmpty() || !m_extensionCommandQueue.isEmpty();
 | |
| }
 | |
| 
 | |
| void CdbEngine::shutdownInferior()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::shutdownInferior in state '%s', process running %d", stateName(state()),
 | |
|                isCdbProcessRunning());
 | |
| 
 | |
|     if (!isCdbProcessRunning()) { // Direct launch: Terminated with process.
 | |
|         if (debug)
 | |
|             qDebug("notifyInferiorShutdownOk");
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorShutdownOk")
 | |
|         notifyInferiorShutdownOk();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (m_accessible) {
 | |
|         if (m_effectiveStartMode == AttachExternal || m_effectiveStartMode == AttachCrashedExternal)
 | |
|             detachDebugger();
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorShutdownOk")
 | |
|         notifyInferiorShutdownOk();
 | |
|     } else {
 | |
|         // A command got stuck.
 | |
|         if (commandsPending()) {
 | |
|             showMessage(QLatin1String("Cannot shut down inferior due to pending commands."), LogWarning);
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorShutdownFailed")
 | |
|             notifyInferiorShutdownFailed();
 | |
|             return;
 | |
|         }
 | |
|         if (!canInterruptInferior()) {
 | |
|             showMessage(QLatin1String("Cannot interrupt the inferior."), LogWarning);
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorShutdownFailed")
 | |
|             notifyInferiorShutdownFailed();
 | |
|             return;
 | |
|         }
 | |
|         interruptInferior(); // Calls us again
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* shutdownEngine/processFinished:
 | |
|  * Note that in the case of launching a process by the debugger, the debugger
 | |
|  * automatically quits a short time after reporting the session becoming
 | |
|  * inaccessible without debuggee (notifyInferiorExited). In that case,
 | |
|  * processFinished() must not report any arbitrarily notifyEngineShutdownOk()
 | |
|  * as not to confuse the state engine.
 | |
|  */
 | |
| 
 | |
| void CdbEngine::shutdownEngine()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::shutdownEngine in state '%s', process running %d,"
 | |
|                "accessible=%d,commands pending=%d",
 | |
|                stateName(state()), isCdbProcessRunning(), m_accessible,
 | |
|                commandsPending());
 | |
| 
 | |
|     if (!isCdbProcessRunning()) { // Direct launch: Terminated with process.
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineShutdownOk")
 | |
|         notifyEngineShutdownOk();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     // No longer trigger anything from messages
 | |
|     m_ignoreCdbOutput = true;
 | |
|     // Go for kill if there are commands pending.
 | |
|     if (m_accessible && !commandsPending()) {
 | |
|         // detach: Wait for debugger to finish.
 | |
|         if (m_effectiveStartMode == AttachExternal)
 | |
|             detachDebugger();
 | |
|         // Remote requires a bit more force to quit.
 | |
|         if (m_effectiveStartMode == AttachToRemote) {
 | |
|             postCommand(m_extensionCommandPrefixBA + "shutdownex", 0);
 | |
|             postCommand("qq", 0);
 | |
|         } else {
 | |
|             postCommand("q", 0);
 | |
|         }
 | |
|         m_notifyEngineShutdownOnTermination = true;
 | |
|         return;
 | |
|     } else {
 | |
|         // Remote process. No can do, currently
 | |
|         m_notifyEngineShutdownOnTermination = true;
 | |
|         Utils::SynchronousProcess::stopProcess(m_process);
 | |
|         return;
 | |
|     }
 | |
|     // Lost debuggee, debugger should quit anytime now
 | |
|     if (!m_hasDebuggee) {
 | |
|         m_notifyEngineShutdownOnTermination = true;
 | |
|         return;
 | |
|     }
 | |
|     interruptInferior();
 | |
| }
 | |
| 
 | |
| void CdbEngine::processFinished()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::processFinished %dms '%s' notify=%d (exit state=%d, ex=%d)",
 | |
|                elapsedLogTime(), stateName(state()), m_notifyEngineShutdownOnTermination,
 | |
|                m_process.exitStatus(), m_process.exitCode());
 | |
| 
 | |
|     const bool crashed = m_process.exitStatus() == QProcess::CrashExit;
 | |
|     if (crashed) {
 | |
|         showMessage(tr("CDB crashed"), LogError); // not in your life.
 | |
|     } else {
 | |
|         showMessage(tr("CDB exited (%1)").arg(m_process.exitCode()), LogMisc);
 | |
|     }
 | |
| 
 | |
|     if (m_notifyEngineShutdownOnTermination) {
 | |
|         if (crashed) {
 | |
|             if (debug)
 | |
|                 qDebug("notifyEngineIll");
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineIll")
 | |
|             notifyEngineIll();
 | |
|         } else {
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineShutdownOk")
 | |
|             notifyEngineShutdownOk();
 | |
|         }
 | |
|     } else {
 | |
|         // The QML/CPP engine relies on the standard sequence of InferiorShutDown,etc.
 | |
|         // Otherwise, we take a shortcut.
 | |
|         if (isSlaveEngine()) {
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorExited")
 | |
|             notifyInferiorExited();
 | |
|         } else {
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSpontaneousShutdown")
 | |
|             notifyEngineSpontaneousShutdown();
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::detachDebugger()
 | |
| {
 | |
|     postCommand(".detach", 0);
 | |
| }
 | |
| 
 | |
| static inline bool isWatchIName(const QByteArray &iname)
 | |
| {
 | |
|     return iname.startsWith("watch");
 | |
| }
 | |
| 
 | |
| void CdbEngine::updateWatchData(const WatchData &dataIn,
 | |
|                                 const WatchUpdateFlags & flags)
 | |
| {
 | |
|     if (debug || debugLocals || debugWatches)
 | |
|         qDebug("CdbEngine::updateWatchData() %dms accessible=%d %s incr=%d: %s",
 | |
|                elapsedLogTime(), m_accessible, stateName(state()),
 | |
|                flags.tryIncremental,
 | |
|                qPrintable(dataIn.toString()));
 | |
| 
 | |
|     if (!m_accessible) // Add watch data while running?
 | |
|         return;
 | |
| 
 | |
|     // New watch item?
 | |
|     if (isWatchIName(dataIn.iname) && dataIn.isValueNeeded()) {
 | |
|         QByteArray args;
 | |
|         ByteArrayInputStream str(args);
 | |
|         str << dataIn.iname << " \"" << dataIn.exp << '"';
 | |
|         postExtensionCommand("addwatch", args, 0,
 | |
|                              &CdbEngine::handleAddWatch, 0,
 | |
|                              qVariantFromValue(dataIn));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (!dataIn.hasChildren && !dataIn.isValueNeeded()) {
 | |
|         WatchData data = dataIn;
 | |
|         data.setAllUnneeded();
 | |
|         watchHandler()->insertData(data);
 | |
|         return;
 | |
|     }
 | |
|     updateLocalVariable(dataIn.iname);
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleAddWatch(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     WatchData item = qvariant_cast<WatchData>(reply->cookie);
 | |
|     if (debugWatches)
 | |
|         qDebug() << "handleAddWatch ok="  << reply->success << item.iname;
 | |
|     if (reply->success) {
 | |
|         updateLocalVariable(item.iname);
 | |
|     } else {
 | |
|         item.setError(tr("Unable to add expression"));
 | |
|         watchHandler()->insertData(item);
 | |
|         showMessage(QString::fromLatin1("Unable to add watch item '%1'/'%2': %3").
 | |
|                     arg(QString::fromAscii(item.iname), QString::fromAscii(item.exp),
 | |
|                         reply->errorMessage), LogError);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::addLocalsOptions(ByteArrayInputStream &str) const
 | |
| {
 | |
|     if (debuggerCore()->boolSetting(VerboseLog))
 | |
|         str << blankSeparator << "-v";
 | |
|     if (debuggerCore()->boolSetting(UseDebuggingHelpers))
 | |
|         str << blankSeparator << "-c";
 | |
|     const QByteArray typeFormats = watchHandler()->typeFormatRequests();
 | |
|     if (!typeFormats.isEmpty())
 | |
|         str << blankSeparator << "-T " << typeFormats;
 | |
|     const QByteArray individualFormats = watchHandler()->individualFormatRequests();
 | |
|     if (!individualFormats.isEmpty())
 | |
|         str << blankSeparator << "-I " << individualFormats;
 | |
| }
 | |
| 
 | |
| void CdbEngine::updateLocalVariable(const QByteArray &iname)
 | |
| {
 | |
|     const bool isWatch = isWatchIName(iname);
 | |
|     if (debugWatches)
 | |
|         qDebug() << "updateLocalVariable watch=" << isWatch << iname;
 | |
|     QByteArray localsArguments;
 | |
|     ByteArrayInputStream str(localsArguments);
 | |
|     addLocalsOptions(str);
 | |
|     if (!isWatch) {
 | |
|         const int stackFrame = stackHandler()->currentIndex();
 | |
|         if (stackFrame < 0) {
 | |
|             qWarning("Internal error; no stack frame in updateLocalVariable");
 | |
|             return;
 | |
|         }
 | |
|         str << blankSeparator << stackFrame;
 | |
|     }
 | |
|     str << blankSeparator << iname;
 | |
|     postExtensionCommand(isWatch ? "watches" : "locals", localsArguments, 0, &CdbEngine::handleLocals);
 | |
| }
 | |
| 
 | |
| unsigned CdbEngine::debuggerCapabilities() const
 | |
| {
 | |
|     return DisassemblerCapability | RegisterCapability | ShowMemoryCapability
 | |
|            |WatchpointCapability|JumpToLineCapability|AddWatcherCapability
 | |
|            |ReloadModuleCapability
 | |
|            |BreakOnThrowAndCatchCapability // Sort-of: Can break on throw().
 | |
|            |BreakConditionCapability|TracePointCapability
 | |
|            |BreakModuleCapability;
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeStep()
 | |
| {
 | |
|     if (!m_operateByInstruction)
 | |
|         m_sourceStepInto = true; // See explanation at handleStackTrace().
 | |
|     postCommand(QByteArray("t"), 0); // Step into-> t (trace)
 | |
|     STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested")
 | |
|     notifyInferiorRunRequested();
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeStepOut()
 | |
| {
 | |
|     postCommand(QByteArray("gu"), 0); // Step out-> gu (go up)
 | |
|     STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested")
 | |
|     notifyInferiorRunRequested();
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeNext()
 | |
| {
 | |
|     postCommand(QByteArray("p"), 0); // Step over -> p
 | |
|     STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested")
 | |
|     notifyInferiorRunRequested();
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeStepI()
 | |
| {
 | |
|     executeStep();
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeNextI()
 | |
| {
 | |
|     executeNext();
 | |
| }
 | |
| 
 | |
| void CdbEngine::continueInferior()
 | |
| {
 | |
|     STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested")
 | |
|     notifyInferiorRunRequested();
 | |
|     doContinueInferior();
 | |
| }
 | |
| 
 | |
| void CdbEngine::doContinueInferior()
 | |
| {
 | |
|     postCommand(QByteArray("g"), 0);
 | |
| }
 | |
| 
 | |
| bool CdbEngine::canInterruptInferior() const
 | |
| {
 | |
|     return m_effectiveStartMode != AttachToRemote && inferiorPid();
 | |
| }
 | |
| 
 | |
| void CdbEngine::interruptInferior()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug() << "CdbEngine::interruptInferior()" << stateName(state());
 | |
|     if (canInterruptInferior()) {
 | |
|         doInterruptInferior(NoSpecialStop);
 | |
|     } else {
 | |
|         showMessage(tr("Interrupting is not possible in remote sessions."), LogError);
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorStopOk")
 | |
|         notifyInferiorStopOk();
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested")
 | |
|         notifyInferiorRunRequested();
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunOk")
 | |
|         notifyInferiorRunOk();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::doInterruptInferiorCustomSpecialStop(const QVariant &v)
 | |
| {
 | |
|     if (m_specialStopMode == NoSpecialStop)
 | |
|         doInterruptInferior(CustomSpecialStop);
 | |
|     m_customSpecialStopData.push_back(v);
 | |
| }
 | |
| 
 | |
| void CdbEngine::doInterruptInferior(SpecialStopMode sm)
 | |
| {
 | |
| #ifdef Q_OS_WIN
 | |
|     const SpecialStopMode oldSpecialMode = m_specialStopMode;
 | |
|     m_specialStopMode = sm;
 | |
|     QString errorMessage;
 | |
|     showMessage(QString::fromLatin1("Interrupting process %1...").arg(inferiorPid()), LogMisc);
 | |
|     if (!winDebugBreakProcess(inferiorPid(), &errorMessage)) {
 | |
|         m_specialStopMode = oldSpecialMode;
 | |
|         showMessage(QString::fromLatin1("Cannot interrupt process %1: %2").arg(inferiorPid()).arg(errorMessage), LogError);
 | |
|     }
 | |
| #else
 | |
|     Q_UNUSED(sm)
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeRunToLine(const ContextData &data)
 | |
| {
 | |
|     // Add one-shot breakpoint
 | |
|     BreakpointParameters bp;
 | |
|     if (data.address) {
 | |
|         bp.type =BreakpointByAddress;
 | |
|         bp.address = data.address;
 | |
|     } else {
 | |
|         bp.type =BreakpointByFileAndLine;
 | |
|         bp.fileName = data.fileName;
 | |
|         bp.lineNumber = data.lineNumber;
 | |
|     }
 | |
|     postCommand(cdbAddBreakpointCommand(bp, m_sourcePathMappings, BreakpointId(-1), true), 0);
 | |
|     continueInferior();
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeRunToFunction(const QString &functionName)
 | |
| {
 | |
|     // Add one-shot breakpoint
 | |
|     BreakpointParameters bp(BreakpointByFunction);
 | |
|     bp.functionName = functionName;
 | |
| 
 | |
|     postCommand(cdbAddBreakpointCommand(bp, m_sourcePathMappings, BreakpointId(-1), true), 0);
 | |
|     continueInferior();
 | |
| }
 | |
| 
 | |
| void CdbEngine::setRegisterValue(int regnr, const QString &value)
 | |
| {
 | |
|     const Registers registers = registerHandler()->registers();
 | |
|     QTC_ASSERT(regnr < registers.size(), return)
 | |
|     // Value is decimal or 0x-hex-prefixed
 | |
|     QByteArray cmd;
 | |
|     ByteArrayInputStream str(cmd);
 | |
|     str << "r " << registers.at(regnr).name << '=' << value;
 | |
|     postCommand(cmd, 0);
 | |
|     reloadRegisters();
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeJumpToLine(const ContextData &data)
 | |
| {
 | |
|     if (data.address) {
 | |
|         // Goto address directly.
 | |
|         jumpToAddress(data.address);
 | |
|         gotoLocation(Location(data.address));
 | |
|     } else {
 | |
|         // Jump to source line: Resolve source line address and go to that location
 | |
|         QByteArray cmd;
 | |
|         ByteArrayInputStream str(cmd);
 | |
|         str << "? `" << QDir::toNativeSeparators(data.fileName) << ':' << data.lineNumber << '`';
 | |
|         const QVariant cookie = qVariantFromValue(data);
 | |
|         postBuiltinCommand(cmd, 0, &CdbEngine::handleJumpToLineAddressResolution, 0, cookie);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::jumpToAddress(quint64 address)
 | |
| {
 | |
|     // Fake a jump to address by setting the PC register.
 | |
|     QByteArray registerCmd;
 | |
|     ByteArrayInputStream str(registerCmd);
 | |
|     // PC-register depending on 64/32bit.
 | |
|     str << "r " << (startParameters().toolChainAbi.wordWidth() == 64 ? "rip" : "eip") << '=';
 | |
|     str.setHexPrefix(true);
 | |
|     str.setIntegerBase(16);
 | |
|     str << address;
 | |
|     postCommand(registerCmd, 0);
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleJumpToLineAddressResolution(const CdbBuiltinCommandPtr &cmd)
 | |
| {
 | |
|     if (cmd->reply.isEmpty())
 | |
|         return;
 | |
|     // Evaluate expression: 5365511549 = 00000001`3fcf357d
 | |
|     // Set register 'rip' to hex address and goto lcoation
 | |
|     QString answer = QString::fromAscii(cmd->reply.front()).trimmed();
 | |
|     const int equalPos = answer.indexOf(" = ");
 | |
|     if (equalPos == -1)
 | |
|         return;
 | |
|     answer.remove(0, equalPos + 3);
 | |
|     answer.remove(QLatin1Char('`'));
 | |
|     bool ok;
 | |
|     const quint64 address = answer.toLongLong(&ok, 16);
 | |
|     if (ok && address) {
 | |
|         QTC_ASSERT(qVariantCanConvert<ContextData>(cmd->cookie), return);
 | |
|         const ContextData cookie = qvariant_cast<ContextData>(cmd->cookie);
 | |
|         jumpToAddress(address);
 | |
|         gotoLocation(Location(cookie.fileName, cookie.lineNumber));
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::assignValueInDebugger(const WatchData *w, const QString &expr, const QVariant &value)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug() << "CdbEngine::assignValueInDebugger" << w->iname << expr << value;
 | |
| 
 | |
|     if (state() != InferiorStopOk || stackHandler()->currentIndex() < 0) {
 | |
|         qWarning("Internal error: assignValueInDebugger: Invalid state or no stack frame.");
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     QByteArray cmd;
 | |
|     ByteArrayInputStream str(cmd);
 | |
|     str << m_extensionCommandPrefixBA << "assign " << w->iname << '=' << value.toString();
 | |
|     postCommand(cmd, 0);
 | |
|     // Update all locals in case we change a union or something pointed to
 | |
|     // that affects other variables, too.
 | |
|     updateLocals();
 | |
| }
 | |
| 
 | |
| void CdbEngine::parseThreads(const GdbMi &data, int forceCurrentThreadId /* = -1 */)
 | |
| {
 | |
|     int currentThreadId;
 | |
|     Threads threads = ThreadsHandler::parseGdbmiThreads(data, ¤tThreadId);
 | |
|     threadsHandler()->setThreads(threads);
 | |
|     threadsHandler()->setCurrentThreadId(forceCurrentThreadId >= 0 ?
 | |
|                                          forceCurrentThreadId : currentThreadId);
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleThreads(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::handleThreads success=%d", reply->success);
 | |
|     if (reply->success) {
 | |
|         GdbMi data;
 | |
|         data.fromString(reply->reply);
 | |
|         parseThreads(data);
 | |
|         // Continue sequence
 | |
|         postCommandSequence(reply->commandSequence);
 | |
|     } else {
 | |
|         showMessage(QString::fromLatin1(reply->errorMessage), LogError);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::executeDebuggerCommand(const QString &command)
 | |
| {
 | |
|     postCommand(command.toLocal8Bit(), QuietCommand);
 | |
| }
 | |
| 
 | |
| // Post command without callback
 | |
| void CdbEngine::postCommand(const QByteArray &cmd, unsigned flags)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::postCommand %dms '%s' %u %s\n",
 | |
|                elapsedLogTime(), cmd.constData(), flags, stateName(state()));
 | |
|     if (!(flags & QuietCommand))
 | |
|         showMessage(QString::fromLocal8Bit(cmd), LogInput);
 | |
|     m_process.write(cmd + '\n');
 | |
| }
 | |
| 
 | |
| // Post a built-in-command producing free-format output with a callback.
 | |
| // In order to catch the output, it is enclosed in 'echo' commands
 | |
| // printing a specially formatted token to be identifiable in the output.
 | |
| void CdbEngine::postBuiltinCommand(const QByteArray &cmd, unsigned flags,
 | |
|                                    BuiltinCommandHandler handler,
 | |
|                                    unsigned nextCommandFlag,
 | |
|                                    const QVariant &cookie)
 | |
| {
 | |
|     if (!m_accessible) {
 | |
|         const QString msg = QString::fromLatin1("Attempt to issue builtin command '%1' to non-accessible session (%2)")
 | |
|                 .arg(QString::fromLocal8Bit(cmd), QString::fromAscii(stateName(state())));
 | |
|         showMessage(msg, LogError);
 | |
|         return;
 | |
|     }
 | |
|     if (!flags & QuietCommand)
 | |
|         showMessage(QString::fromLocal8Bit(cmd), LogInput);
 | |
| 
 | |
|     const int token = m_nextCommandToken++;
 | |
|     CdbBuiltinCommandPtr pendingCommand(new CdbBuiltinCommand(cmd, token, flags, handler, nextCommandFlag, cookie));
 | |
| 
 | |
|     m_builtinCommandQueue.push_back(pendingCommand);
 | |
|     // Enclose command in echo-commands for token
 | |
|     QByteArray fullCmd;
 | |
|     ByteArrayInputStream str(fullCmd);
 | |
|     str << ".echo \"" << m_tokenPrefix << token << "<\"\n"
 | |
|             << cmd << "\n.echo \"" << m_tokenPrefix << token << ">\"\n";
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::postBuiltinCommand %dms '%s' flags=%u token=%d %s next=%u, cookie='%s', pending=%d, sequence=0x%x",
 | |
|                elapsedLogTime(), cmd.constData(), flags, token, stateName(state()), nextCommandFlag, qPrintable(cookie.toString()),
 | |
|                m_builtinCommandQueue.size(), nextCommandFlag);
 | |
|     if (debug > 1)
 | |
|         qDebug("CdbEngine::postBuiltinCommand: resulting command '%s'\n",
 | |
|                fullCmd.constData());
 | |
|     m_process.write(fullCmd);
 | |
| }
 | |
| 
 | |
| // Post an extension command producing one-line output with a callback,
 | |
| // pass along token for identification in queue.
 | |
| void CdbEngine::postExtensionCommand(const QByteArray &cmd,
 | |
|                                      const QByteArray &arguments,
 | |
|                                      unsigned flags,
 | |
|                                      ExtensionCommandHandler handler,
 | |
|                                      unsigned nextCommandFlag,
 | |
|                                      const QVariant &cookie)
 | |
| {
 | |
|     if (!m_accessible) {
 | |
|         const QString msg = QString::fromLatin1("Attempt to issue extension command '%1' to non-accessible session (%2)")
 | |
|                 .arg(QString::fromLocal8Bit(cmd), QString::fromAscii(stateName(state())));
 | |
|         showMessage(msg, LogError);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     const int token = m_nextCommandToken++;
 | |
| 
 | |
|     // Format full command with token to be recognizeable in the output
 | |
|     QByteArray fullCmd;
 | |
|     ByteArrayInputStream str(fullCmd);
 | |
|     str << m_extensionCommandPrefixBA << cmd << " -t " << token;
 | |
|     if (!arguments.isEmpty())
 | |
|         str <<  ' ' << arguments;
 | |
| 
 | |
|     if (!flags & QuietCommand)
 | |
|         showMessage(QString::fromLocal8Bit(fullCmd), LogInput);
 | |
| 
 | |
|     CdbExtensionCommandPtr pendingCommand(new CdbExtensionCommand(fullCmd, token, flags, handler, nextCommandFlag, cookie));
 | |
| 
 | |
|     m_extensionCommandQueue.push_back(pendingCommand);
 | |
|     // Enclose command in echo-commands for token
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::postExtensionCommand %dms '%s' flags=%u token=%d %s next=%u, cookie='%s', pending=%d, sequence=0x%x",
 | |
|                elapsedLogTime(), fullCmd.constData(), flags, token, stateName(state()), nextCommandFlag, qPrintable(cookie.toString()),
 | |
|                m_extensionCommandQueue.size(), nextCommandFlag);
 | |
|     m_process.write(fullCmd + '\n');
 | |
| }
 | |
| 
 | |
| void CdbEngine::activateFrame(int index)
 | |
| {
 | |
|     // TODO: assembler,etc
 | |
|     if (index < 0)
 | |
|         return;
 | |
|     const StackFrames &frames = stackHandler()->frames();
 | |
|     QTC_ASSERT(index < frames.size(), return; )
 | |
| 
 | |
|     const StackFrame frame = frames.at(index);
 | |
|     if (debug || debugLocals)
 | |
|         qDebug("activateFrame idx=%d '%s' %d", index,
 | |
|                qPrintable(frame.file), frame.line);
 | |
|     stackHandler()->setCurrentIndex(index);
 | |
|     const bool showAssembler = !frames.at(index).isUsable();
 | |
|     if (showAssembler) { // Assembly code: Clean out model and force instruction mode.
 | |
|         watchHandler()->beginCycle();
 | |
|         watchHandler()->endCycle();
 | |
|         QAction *assemblerAction = theAssemblerAction();
 | |
|         if (assemblerAction->isChecked()) {
 | |
|             gotoLocation(frame);
 | |
|         } else {
 | |
|             assemblerAction->trigger(); // Seems to trigger update
 | |
|         }
 | |
|     } else {
 | |
|         gotoLocation(frame);
 | |
|         updateLocals(true);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::updateLocals(bool forNewStackFrame)
 | |
| {
 | |
|     typedef QHash<QByteArray, int> WatcherHash;
 | |
| 
 | |
|     const int frameIndex = stackHandler()->currentIndex();
 | |
|     if (frameIndex < 0) {
 | |
|         watchHandler()->beginCycle();
 | |
|         watchHandler()->endCycle();
 | |
|         return;
 | |
|     }
 | |
|     const StackFrame frame = stackHandler()->currentFrame();
 | |
|     if (!frame.isUsable()) {
 | |
|         watchHandler()->beginCycle();
 | |
|         watchHandler()->endCycle();
 | |
|         return;
 | |
|     }
 | |
|     /* Watchers: Forcibly discard old symbol group as switching from
 | |
|      * thread 0/frame 0 -> thread 1/assembly -> thread 0/frame 0 will otherwise re-use it
 | |
|      * and cause errors as it seems to go 'stale' when switching threads.
 | |
|      * Initial expand, get uninitialized and query */
 | |
|     QByteArray arguments;
 | |
|     ByteArrayInputStream str(arguments);
 | |
|     str << "-D";
 | |
|     // Pre-expand
 | |
|     const QSet<QByteArray> expanded = watchHandler()->expandedINames();
 | |
|     if (!expanded.isEmpty()) {
 | |
|         str << blankSeparator << "-e ";
 | |
|         int i = 0;
 | |
|         foreach(const QByteArray &e, expanded) {
 | |
|             if (i++)
 | |
|                 str << ',';
 | |
|             str << e;
 | |
|         }
 | |
|     }
 | |
|     addLocalsOptions(str);
 | |
|     // Uninitialized variables if desired. Quote as safeguard against shadowed
 | |
|     // variables in case of errors in uninitializedVariables().
 | |
|     if (debuggerCore()->boolSetting(UseCodeModel)) {
 | |
|         QStringList uninitializedVariables;
 | |
|         getUninitializedVariables(debuggerCore()->cppCodeModelSnapshot(),
 | |
|                                   frame.function, frame.file, frame.line, &uninitializedVariables);
 | |
|         if (!uninitializedVariables.isEmpty()) {
 | |
|             str << blankSeparator << "-u \"";
 | |
|             int i = 0;
 | |
|             foreach(const QString &u, uninitializedVariables) {
 | |
|                 if (i++)
 | |
|                     str << ',';
 | |
|                 str << localsPrefixC << u;
 | |
|             }
 | |
|             str << '"';
 | |
|         }
 | |
|     }
 | |
|     // Perform watches synchronization
 | |
|     str << blankSeparator << "-W";
 | |
|     const WatcherHash watcherHash = WatchHandler::watcherNames();
 | |
|     if (!watcherHash.isEmpty()) {
 | |
|         const WatcherHash::const_iterator cend = watcherHash.constEnd();
 | |
|         for (WatcherHash::const_iterator it = watcherHash.constBegin(); it != cend; ++it) {
 | |
|             str << blankSeparator << "-w " << it.value() << " \"" << it.key() << '"';
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Required arguments: frame
 | |
|     str << blankSeparator << frameIndex;
 | |
|     watchHandler()->beginCycle();
 | |
|     postExtensionCommand("locals", arguments, 0, &CdbEngine::handleLocals, 0, QVariant(forNewStackFrame));
 | |
| }
 | |
| 
 | |
| void CdbEngine::selectThread(int index)
 | |
| {
 | |
|     if (index < 0 || index == threadsHandler()->currentThread())
 | |
|         return;
 | |
| 
 | |
|     resetLocation();
 | |
|     const int newThreadId = threadsHandler()->threads().at(index).id;
 | |
|     threadsHandler()->setCurrentThread(index);
 | |
| 
 | |
|     const QByteArray cmd = "~" + QByteArray::number(newThreadId) + " s";
 | |
|     postBuiltinCommand(cmd, 0, &CdbEngine::dummyHandler, CommandListStack);
 | |
| }
 | |
| 
 | |
| void CdbEngine::fetchDisassembler(DisassemblerAgent *agent)
 | |
| {
 | |
|     QTC_ASSERT(m_accessible, return;)
 | |
|     QByteArray cmd;
 | |
|     ByteArrayInputStream str(cmd);
 | |
|     str <<  "u " << hex << hexPrefixOn << agent->address() << " L40";
 | |
|     const QVariant cookie = qVariantFromValue<DisassemblerAgent*>(agent);
 | |
|     postBuiltinCommand(cmd, 0, &CdbEngine::handleDisassembler, 0, cookie);
 | |
| }
 | |
| 
 | |
| // Parse: "00000000`77606060 cc              int     3"
 | |
| void CdbEngine::handleDisassembler(const CdbBuiltinCommandPtr &command)
 | |
| {
 | |
|     QTC_ASSERT(qVariantCanConvert<DisassemblerAgent*>(command->cookie), return;)
 | |
|     DisassemblerAgent *agent = qvariant_cast<DisassemblerAgent*>(command->cookie);
 | |
|     agent->setContents(parseCdbDisassembler(command->reply));
 | |
| }
 | |
| 
 | |
| void CdbEngine::fetchMemory(MemoryAgent *agent, QObject *editor, quint64 addr, quint64 length)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::fetchMemory %llu bytes from 0x%llx", length, addr);
 | |
|     const MemoryViewCookie cookie(agent, editor, addr, length);
 | |
|     if (m_accessible) {
 | |
|         postFetchMemory(cookie);
 | |
|     } else {
 | |
|         doInterruptInferiorCustomSpecialStop(qVariantFromValue(cookie));
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::postFetchMemory(const MemoryViewCookie &cookie)
 | |
| {
 | |
|     QByteArray args;
 | |
|     ByteArrayInputStream str(args);
 | |
|     str << cookie.address << ' ' << cookie.length;
 | |
|     postExtensionCommand("memory", args, 0, &CdbEngine::handleMemory, 0,
 | |
|                          qVariantFromValue(cookie));
 | |
| }
 | |
| 
 | |
| void CdbEngine::changeMemory(Internal::MemoryAgent *, QObject *, quint64 addr, const QByteArray &data)
 | |
| {
 | |
|     QTC_ASSERT(!data.isEmpty(), return; )
 | |
|     if (!m_accessible) {
 | |
|         const MemoryChangeCookie cookie(addr, data);
 | |
|         doInterruptInferiorCustomSpecialStop(qVariantFromValue(cookie));
 | |
|     } else {
 | |
|         postCommand(cdbWriteMemoryCommand(addr, data), 0);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleMemory(const CdbExtensionCommandPtr &command)
 | |
| {
 | |
|     QTC_ASSERT(qVariantCanConvert<MemoryViewCookie>(command->cookie), return;)
 | |
|     const MemoryViewCookie memViewCookie = qvariant_cast<MemoryViewCookie>(command->cookie);
 | |
|     if (command->success) {
 | |
|         const QByteArray data = QByteArray::fromBase64(command->reply);
 | |
|         if (unsigned(data.size()) == memViewCookie.length)
 | |
|             memViewCookie.agent->addLazyData(memViewCookie.editorToken,
 | |
|                                              memViewCookie.address, data);
 | |
|     } else {
 | |
|         showMessage(QString::fromLocal8Bit(command->errorMessage), LogWarning);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::reloadModules()
 | |
| {
 | |
|     postCommandSequence(CommandListModules);
 | |
| }
 | |
| 
 | |
| void CdbEngine::loadSymbols(const QString & /* moduleName */)
 | |
| {
 | |
| }
 | |
| 
 | |
| void CdbEngine::loadAllSymbols()
 | |
| {
 | |
| }
 | |
| 
 | |
| void CdbEngine::requestModuleSymbols(const QString &moduleName)
 | |
| {
 | |
|     Q_UNUSED(moduleName)
 | |
| }
 | |
| 
 | |
| void CdbEngine::reloadRegisters()
 | |
| {
 | |
|     postCommandSequence(CommandListRegisters);
 | |
| }
 | |
| 
 | |
| void CdbEngine::reloadSourceFiles()
 | |
| {
 | |
| }
 | |
| 
 | |
| void CdbEngine::reloadFullStack()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("%s", Q_FUNC_INFO);
 | |
|     postCommandSequence(CommandListStack);
 | |
| }
 | |
| 
 | |
| void CdbEngine::handlePid(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     if (reply->success) {
 | |
|         notifyInferiorPid(reply->reply.toULongLong());
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorSetupOk")
 | |
|         notifyInferiorSetupOk();
 | |
|     }  else {
 | |
|         showMessage(QString::fromLatin1("Failed to determine inferior pid: %1").
 | |
|                     arg(QLatin1String(reply->errorMessage)), LogError);
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorSetupFailed")
 | |
|         notifyInferiorSetupFailed();
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Parse CDB gdbmi register syntax
 | |
| static inline Register parseRegister(const GdbMi &gdbmiReg)
 | |
| {
 | |
|     Register reg;
 | |
|     reg.name = gdbmiReg.findChild("name").data();
 | |
|     const GdbMi description = gdbmiReg.findChild("description");
 | |
|     if (description.type() != GdbMi::Invalid) {
 | |
|         reg.name += " (";
 | |
|         reg.name += description.data();
 | |
|         reg.name += ')';
 | |
|     }
 | |
|     reg.value = QString::fromAscii(gdbmiReg.findChild("value").data());
 | |
|     return reg;
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleModules(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     if (reply->success) {
 | |
|         GdbMi value;
 | |
|         value.fromString(reply->reply);
 | |
|         if (value.type() == GdbMi::List) {
 | |
|             Modules modules;
 | |
|             modules.reserve(value.childCount());
 | |
|             foreach (const GdbMi &gdbmiModule, value.children()) {
 | |
|                 Module module;
 | |
|                 module.moduleName = QString::fromAscii(gdbmiModule.findChild("name").data());
 | |
|                 module.modulePath = QString::fromAscii(gdbmiModule.findChild("image").data());
 | |
|                 module.startAddress = gdbmiModule.findChild("start").data().toULongLong(0, 0);
 | |
|                 module.endAddress = gdbmiModule.findChild("end").data().toULongLong(0, 0);
 | |
|                 if (gdbmiModule.findChild("deferred").type() == GdbMi::Invalid)
 | |
|                     module.symbolsRead = Module::ReadOk;
 | |
|                 modules.push_back(module);
 | |
|             }
 | |
|             modulesHandler()->setModules(modules);
 | |
|         } else {
 | |
|             showMessage(QString::fromLatin1("Parse error in modules response."), LogError);
 | |
|             qWarning("Parse error in modules response:\n%s", reply->reply.constData());
 | |
|         }
 | |
|     }  else {
 | |
|         showMessage(QString::fromLatin1("Failed to determine modules: %1").
 | |
|                     arg(QLatin1String(reply->errorMessage)), LogError);
 | |
|     }
 | |
|     postCommandSequence(reply->commandSequence);
 | |
| 
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleRegisters(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     if (reply->success) {
 | |
|         GdbMi value;
 | |
|         value.fromString(reply->reply);
 | |
|         if (value.type() == GdbMi::List) {
 | |
|             Registers registers;
 | |
|             registers.reserve(value.childCount());
 | |
|             foreach (const GdbMi &gdbmiReg, value.children())
 | |
|                 registers.push_back(parseRegister(gdbmiReg));
 | |
|             registerHandler()->setAndMarkRegisters(registers);
 | |
|         } else {
 | |
|             showMessage(QString::fromLatin1("Parse error in registers response."), LogError);
 | |
|             qWarning("Parse error in registers response:\n%s", reply->reply.constData());
 | |
|         }
 | |
|     }  else {
 | |
|         showMessage(QString::fromLatin1("Failed to determine registers: %1").
 | |
|                     arg(QLatin1String(reply->errorMessage)), LogError);
 | |
|     }
 | |
|     postCommandSequence(reply->commandSequence);
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleLocals(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     if (reply->success) {
 | |
|         QList<WatchData> watchData;
 | |
|         GdbMi root;
 | |
|         root.fromString(reply->reply);
 | |
|         QTC_ASSERT(root.isList(), return ; )
 | |
|         if (debugLocals) {
 | |
|             qDebug() << root.toString(true, 4);
 | |
|         }
 | |
|         // Courtesy of GDB engine
 | |
|         foreach (const GdbMi &child, root.children()) {
 | |
|             WatchData dummy;
 | |
|             dummy.iname = child.findChild("iname").data();
 | |
|             dummy.name = QLatin1String(child.findChild("name").data());
 | |
|             parseWatchData(watchHandler()->expandedINames(), dummy, child, &watchData);
 | |
|         }
 | |
|         watchHandler()->insertBulkData(watchData);
 | |
|         watchHandler()->endCycle();
 | |
|         if (debugLocals) {
 | |
|             QDebug nsp = qDebug().nospace();
 | |
|             nsp << "Obtained " << watchData.size() << " items:\n";
 | |
|             foreach (const WatchData &wd, watchData)
 | |
|                 nsp << wd.toString() <<'\n';
 | |
|         }
 | |
|         const bool forNewStackFrame = reply->cookie.toBool();
 | |
|         if (forNewStackFrame)
 | |
|             emit stackFrameCompleted();
 | |
|     } else {
 | |
|         showMessage(QString::fromLatin1(reply->errorMessage), LogWarning);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleExpandLocals(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     if (!reply->success)
 | |
|         showMessage(QString::fromLatin1(reply->errorMessage), LogError);
 | |
| }
 | |
| 
 | |
| enum CdbExecutionStatus {
 | |
| CDB_STATUS_NO_CHANGE=0, CDB_STATUS_GO = 1, CDB_STATUS_GO_HANDLED = 2,
 | |
| CDB_STATUS_GO_NOT_HANDLED = 3, CDB_STATUS_STEP_OVER = 4,
 | |
| CDB_STATUS_STEP_INTO = 5, CDB_STATUS_BREAK = 6, CDB_STATUS_NO_DEBUGGEE = 7,
 | |
| CDB_STATUS_STEP_BRANCH = 8, CDB_STATUS_IGNORE_EVENT = 9,
 | |
| CDB_STATUS_RESTART_REQUESTED = 10, CDB_STATUS_REVERSE_GO = 11,
 | |
| CDB_STATUS_REVERSE_STEP_BRANCH = 12, CDB_STATUS_REVERSE_STEP_OVER = 13,
 | |
| CDB_STATUS_REVERSE_STEP_INTO = 14 };
 | |
| 
 | |
| static const char *cdbStatusName(unsigned long s)
 | |
| {
 | |
|     switch (s) {
 | |
|     case CDB_STATUS_NO_CHANGE:
 | |
|         return "No change";
 | |
|     case CDB_STATUS_GO:
 | |
|         return "go";
 | |
|     case CDB_STATUS_GO_HANDLED:
 | |
|         return "go_handled";
 | |
|     case CDB_STATUS_GO_NOT_HANDLED:
 | |
|         return "go_not_handled";
 | |
|     case CDB_STATUS_STEP_OVER:
 | |
|         return "step_over";
 | |
|     case CDB_STATUS_STEP_INTO:
 | |
|         return "step_into";
 | |
|     case CDB_STATUS_BREAK:
 | |
|         return "break";
 | |
|     case CDB_STATUS_NO_DEBUGGEE:
 | |
|         return "no_debuggee";
 | |
|     case CDB_STATUS_STEP_BRANCH:
 | |
|         return "step_branch";
 | |
|     case CDB_STATUS_IGNORE_EVENT:
 | |
|         return "ignore_event";
 | |
|     case CDB_STATUS_RESTART_REQUESTED:
 | |
|         return "restart_requested";
 | |
|     case CDB_STATUS_REVERSE_GO:
 | |
|         return "reverse_go";
 | |
|     case CDB_STATUS_REVERSE_STEP_BRANCH:
 | |
|         return "reverse_step_branch";
 | |
|     case CDB_STATUS_REVERSE_STEP_OVER:
 | |
|         return "reverse_step_over";
 | |
|     case CDB_STATUS_REVERSE_STEP_INTO:
 | |
|         return "reverse_step_into";
 | |
|     }
 | |
|     return "unknown";
 | |
| }
 | |
| 
 | |
| /* Examine how to react to a stop. */
 | |
| enum StopActionFlags
 | |
| {
 | |
|     // Report options
 | |
|     StopReportLog = 0x1,
 | |
|     StopReportStatusMessage = 0x2,
 | |
|     StopReportParseError = 0x4,
 | |
|     StopShowExceptionMessageBox = 0x8,
 | |
|     // Notify stop or just continue
 | |
|     StopNotifyStop = 0x10,
 | |
|     StopIgnoreContinue = 0x20,
 | |
|     // Hit on break in artificial stop thread (created by DebugBreak()).
 | |
|     StopInArtificialThread = 0x40,
 | |
|     StopShutdownInProgress = 0x80 // Shutdown in progress
 | |
| };
 | |
| 
 | |
| static inline QString msgTracePointTriggered(BreakpointId id, const int number,
 | |
|                                              const QString &threadId)
 | |
| {
 | |
|     return CdbEngine::tr("Trace point %1 (%2) in thread %3 triggered.")
 | |
|             .arg(id).arg(number).arg(threadId);
 | |
| }
 | |
| 
 | |
| static inline QString msgCheckingConditionalBreakPoint(BreakpointId id, const int number,
 | |
|                                                        const QByteArray &condition,
 | |
|                                                        const QString &threadId)
 | |
| {
 | |
|     return CdbEngine::tr("Conditional breakpoint %1 (%2) in thread %3 triggered, examining expression '%4'.")
 | |
|             .arg(id).arg(number).arg(threadId, QString::fromAscii(condition));
 | |
| }
 | |
| 
 | |
| unsigned CdbEngine::examineStopReason(const GdbMi &stopReason,
 | |
|                                       QString *message,
 | |
|                                       QString *exceptionBoxMessage,
 | |
|                                       bool conditionalBreakPointTriggered)
 | |
| {
 | |
|     // Report stop reason (GDBMI)
 | |
|     unsigned rc  = 0;
 | |
|     if (targetState() == DebuggerFinished)
 | |
|         rc |= StopShutdownInProgress;
 | |
|     if (debug)
 | |
|         qDebug("%s", stopReason.toString(true, 4).constData());
 | |
|     const QByteArray reason = stopReason.findChild("reason").data();
 | |
|     if (reason.isEmpty()) {
 | |
|         *message = tr("Malformed stop response received.");
 | |
|         rc |= StopReportParseError|StopNotifyStop;
 | |
|         return rc;
 | |
|     }
 | |
|     // Additional stop messages occurring for debuggee function calls (widgetAt, etc). Just log.
 | |
|     if (state() == InferiorStopOk) {
 | |
|         *message = QString::fromLatin1("Ignored stop notification from function call (%1).").
 | |
|                     arg(QString::fromAscii(reason));
 | |
|         rc |= StopReportLog;
 | |
|         return rc;
 | |
|     }
 | |
|     const int threadId = stopReason.findChild("threadId").data().toInt();
 | |
|     if (reason == "breakpoint") {
 | |
|         // Note: Internal breakpoints (run to line) are reported with id=0.
 | |
|         // Step out creates temporary breakpoints with id 10000.
 | |
|         BreakpointId id = 0;
 | |
|         int number = 0;
 | |
|         const GdbMi breakpointIdG = stopReason.findChild("breakpointId");
 | |
|         if (breakpointIdG.isValid()) {
 | |
|             id = breakpointIdG.data().toULongLong();
 | |
|             if (id && breakHandler()->engineBreakpointIds(this).contains(id)) {
 | |
|                 const BreakpointResponse parameters =  breakHandler()->response(id);
 | |
|                 // Trace point? Just report.
 | |
|                 number = parameters.number;
 | |
|                 if (parameters.tracepoint) {
 | |
|                     *message = msgTracePointTriggered(id, number, QString::number(threadId));
 | |
|                     return StopReportLog|StopIgnoreContinue;
 | |
|                 }
 | |
|                 // Trigger evaluation of BP expression unless we are already in the response.
 | |
|                 if (!conditionalBreakPointTriggered && !parameters.condition.isEmpty()) {
 | |
|                     *message = msgCheckingConditionalBreakPoint(id, number, parameters.condition,
 | |
|                                                                 QString::number(threadId));
 | |
|                     ConditionalBreakPointCookie cookie(id);
 | |
|                     cookie.stopReason = stopReason;
 | |
|                     evaluateExpression(parameters.condition, qVariantFromValue(cookie));
 | |
|                     return StopReportLog;
 | |
|                 }
 | |
|             } else {
 | |
|                 id = 0;
 | |
|             }
 | |
|         }
 | |
|         if (id && breakHandler()->type(id) == Watchpoint) {
 | |
|             *message = msgWatchpointTriggered(id, number, breakHandler()->address(id), QString::number(threadId));
 | |
|         } else {
 | |
|             *message = msgBreakpointTriggered(id, number, QString::number(threadId));
 | |
|         }
 | |
|         rc |= StopReportStatusMessage|StopNotifyStop;
 | |
|         return rc;
 | |
|     }
 | |
|     if (reason == "exception") {
 | |
|         WinException exception;
 | |
|         exception.fromGdbMI(stopReason);
 | |
|         QString description = exception.toString();
 | |
| #ifdef Q_OS_WIN
 | |
|         // It is possible to hit on a startup trap or WOW86 exception while stepping (if something
 | |
|         // pulls DLLs. Avoid showing a 'stopped' Message box.
 | |
|         if (exception.exceptionCode == winExceptionStartupCompleteTrap
 | |
|             || exception.exceptionCode == winExceptionWX86Breakpoint)
 | |
|             return StopNotifyStop;
 | |
|         if (exception.exceptionCode == winExceptionCtrlPressed) {
 | |
|             // Detect interruption by pressing Ctrl in a console and force a switch to thread 0.
 | |
|             *message = msgInterrupted();
 | |
|             rc |= StopReportStatusMessage|StopNotifyStop|StopInArtificialThread;
 | |
|             return rc;
 | |
|         }
 | |
|         if (isDebuggerWinException(exception.exceptionCode)) {
 | |
|             rc |= StopReportStatusMessage|StopNotifyStop;
 | |
|             // Detect interruption by DebugBreak() and force a switch to thread 0.
 | |
|             if (exception.function == "ntdll!DbgBreakPoint")
 | |
|                 rc |= StopInArtificialThread;
 | |
|             *message = msgInterrupted();
 | |
|             return rc;
 | |
|         }
 | |
| #endif
 | |
|         *exceptionBoxMessage = msgStoppedByException(description, QString::number(threadId));
 | |
|         *message = description;
 | |
|         rc |= StopShowExceptionMessageBox|StopReportStatusMessage|StopNotifyStop;
 | |
|         return rc;
 | |
|     }
 | |
|     *message = msgStopped(QLatin1String(reason));
 | |
|     rc |= StopReportStatusMessage|StopNotifyStop;
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleSessionIdle(const QByteArray &messageBA)
 | |
| {
 | |
|     if (!m_hasDebuggee)
 | |
|         return;
 | |
| 
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::handleSessionIdle %dms '%s' in state '%s', special mode %d",
 | |
|                elapsedLogTime(), messageBA.constData(),
 | |
|                stateName(state()), m_specialStopMode);
 | |
| 
 | |
|     // Switch source level debugging
 | |
|     syncOperateByInstruction(m_operateByInstructionPending);
 | |
| 
 | |
|     // Engine-special stop reasons: Breakpoints and setup
 | |
|     const SpecialStopMode specialStopMode =  m_specialStopMode;
 | |
| 
 | |
|     m_specialStopMode = NoSpecialStop;
 | |
| 
 | |
|     switch(specialStopMode) {
 | |
|     case SpecialStopSynchronizeBreakpoints:
 | |
|         if (debug)
 | |
|             qDebug("attemptBreakpointSynchronization in special stop");
 | |
|         attemptBreakpointSynchronization();
 | |
|         doContinueInferior();
 | |
|         return;
 | |
|     case SpecialStopGetWidgetAt:
 | |
|         postWidgetAtCommand();
 | |
|         return;
 | |
|     case CustomSpecialStop:
 | |
|         foreach (const QVariant &data, m_customSpecialStopData)
 | |
|             handleCustomSpecialStop(data);
 | |
|         m_customSpecialStopData.clear();
 | |
|         doContinueInferior();
 | |
|         return;
 | |
|     case NoSpecialStop:
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     if (state() == EngineSetupRequested) { // Temporary stop at beginning
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupOk")
 | |
|         notifyEngineSetupOk();
 | |
|         return;
 | |
|     }
 | |
|     GdbMi stopReason;
 | |
|     stopReason.fromString(messageBA);
 | |
|     processStop(stopReason, false);
 | |
| }
 | |
| 
 | |
| void CdbEngine::processStop(const GdbMi &stopReason, bool conditionalBreakPointTriggered)
 | |
| {
 | |
|     // Further examine stop and report to user
 | |
|     QString message;
 | |
|     QString exceptionBoxMessage;
 | |
|     int forcedThreadId = -1;
 | |
|     const unsigned stopFlags = examineStopReason(stopReason, &message, &exceptionBoxMessage,
 | |
|                                                  conditionalBreakPointTriggered);
 | |
|     // Do the non-blocking log reporting
 | |
|     if (stopFlags & StopReportLog)
 | |
|         showMessage(message, LogMisc);
 | |
|     if (stopFlags & StopReportStatusMessage)
 | |
|         showStatusMessage(message);
 | |
|     if (stopFlags & StopReportParseError)
 | |
|         showMessage(message, LogError);
 | |
|     // Ignore things like WOW64, report tracepoints.
 | |
|     if (stopFlags & StopIgnoreContinue) {
 | |
|         postCommand("g", 0);
 | |
|         return;
 | |
|     }
 | |
|     // Notify about state and send off command sequence to get stack, etc.
 | |
|     if (stopFlags & StopNotifyStop) {
 | |
|         if (state() == InferiorStopRequested) {
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorStopOk")
 | |
|             notifyInferiorStopOk();
 | |
|         } else {
 | |
|             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorSpontaneousStop")
 | |
|             notifyInferiorSpontaneousStop();
 | |
|         }
 | |
|         // Prevent further commands from being sent if shutdown is in progress
 | |
|         if (stopFlags & StopShutdownInProgress) {
 | |
|             showMessage(QString::fromLatin1("Shutdown request detected..."));
 | |
|             return;
 | |
|         }
 | |
|         const bool sourceStepInto = m_sourceStepInto;
 | |
|         m_sourceStepInto = false;
 | |
|         // Start sequence to get all relevant data.
 | |
|         if (stopFlags & StopInArtificialThread) {
 | |
|             showMessage(tr("Switching to main thread..."), LogMisc);
 | |
|             postCommand("~0 s", 0);
 | |
|             forcedThreadId = 0;
 | |
|             // Re-fetch stack again.
 | |
|             postCommandSequence(CommandListStack);
 | |
|         } else {
 | |
|             const GdbMi stack = stopReason.findChild("stack");
 | |
|             if (stack.isValid()) {
 | |
|                 if (parseStackTrace(stack, sourceStepInto) & ParseStackStepInto) {
 | |
|                     executeStep(); // Hit on a frame while step into, see parseStackTrace().
 | |
|                     return;
 | |
|                 }
 | |
|             } else {
 | |
|                 showMessage(QString::fromAscii(stopReason.findChild("stackerror").data()), LogError);
 | |
|             }
 | |
|         }
 | |
|         const GdbMi threads = stopReason.findChild("threads");
 | |
|         if (threads.isValid()) {
 | |
|             parseThreads(threads, forcedThreadId);
 | |
|         } else {
 | |
|             showMessage(QString::fromAscii(stopReason.findChild("threaderror").data()), LogError);
 | |
|         }
 | |
|         // Fire off remaining commands asynchronously
 | |
|         if (!m_pendingBreakpointMap.isEmpty())
 | |
|             postCommandSequence(CommandListBreakPoints);
 | |
|         if (debuggerCore()->isDockVisible(QLatin1String(Constants::DOCKWIDGET_REGISTER)))
 | |
|             postCommandSequence(CommandListRegisters);
 | |
|         if (debuggerCore()->isDockVisible(QLatin1String(Constants::DOCKWIDGET_MODULES)))
 | |
|             postCommandSequence(CommandListModules);
 | |
|     }
 | |
|     // After the sequence has been sent off and CDB is pondering the commands,
 | |
|     // pop up a message box for exceptions.
 | |
|     if (stopFlags & StopShowExceptionMessageBox)
 | |
|         showStoppedByExceptionMessageBox(exceptionBoxMessage);
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleSessionAccessible(unsigned long cdbExState)
 | |
| {
 | |
|     const DebuggerState s = state();
 | |
|     if (!m_hasDebuggee || s == InferiorRunOk) // suppress reports
 | |
|         return;
 | |
| 
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::handleSessionAccessible %dms in state '%s'/'%s', special mode %d",
 | |
|                elapsedLogTime(), cdbStatusName(cdbExState), stateName(state()), m_specialStopMode);
 | |
| 
 | |
|     switch(s) {
 | |
|     case EngineShutdownRequested:
 | |
|         shutdownEngine();
 | |
|         break;
 | |
|     case InferiorShutdownRequested:
 | |
|         shutdownInferior();
 | |
|         break;
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleSessionInaccessible(unsigned long cdbExState)
 | |
| {
 | |
|     const DebuggerState s = state();
 | |
| 
 | |
|     // suppress reports
 | |
|     if (!m_hasDebuggee || (s == InferiorRunOk && cdbExState != CDB_STATUS_NO_DEBUGGEE))
 | |
|         return;
 | |
| 
 | |
|     if (debug)
 | |
|         qDebug("CdbEngine::handleSessionInaccessible %dms in state '%s', '%s', special mode %d",
 | |
|                elapsedLogTime(), cdbStatusName(cdbExState), stateName(state()), m_specialStopMode);
 | |
| 
 | |
|     switch (state()) {
 | |
|     case EngineSetupRequested:
 | |
|         break;
 | |
|     case EngineRunRequested:
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineRunAndInferiorRunOk")
 | |
|         notifyEngineRunAndInferiorRunOk();
 | |
|         break;
 | |
|     case InferiorRunOk:
 | |
|     case InferiorStopOk:
 | |
|         // Inaccessible without debuggee (exit breakpoint)
 | |
|         // We go for spontaneous engine shutdown instead.
 | |
|         if (cdbExState == CDB_STATUS_NO_DEBUGGEE) {
 | |
|             if (debug)
 | |
|                 qDebug("Lost debuggeee");
 | |
|             m_hasDebuggee = false;
 | |
|         }
 | |
|         break;
 | |
|     case InferiorRunRequested:
 | |
|         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunOk")
 | |
|         notifyInferiorRunOk();
 | |
|         resetLocation();
 | |
|         break;
 | |
|     case EngineShutdownRequested:
 | |
|         break;
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleExtensionMessage(char t, int token, const QByteArray &what, const QByteArray &message)
 | |
| {
 | |
|     if (debug > 1) {
 | |
|         QDebug nospace = qDebug().nospace();
 | |
|         nospace << "handleExtensionMessage " << t << ' ' << token << ' ' << what
 | |
|                 << ' ' << stateName(state());
 | |
|         if (t == 'N' || debug > 1) {
 | |
|             nospace << ' ' << message;
 | |
|         } else {
 | |
|             nospace << ' ' << message.size() << " bytes";
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Is there a reply expected, some command queued?
 | |
|     if (t == 'R' || t == 'N') {
 | |
|         if (token == -1) { // Default token, user typed in extension command
 | |
|             showMessage(QString::fromLatin1(message), LogMisc);
 | |
|             return;
 | |
|         }
 | |
|         const int index = indexOfCommand(m_extensionCommandQueue, token);
 | |
|         if (index != -1) {
 | |
|             // Did the command finish? Take off queue and complete, invoke CB
 | |
|             const CdbExtensionCommandPtr command = m_extensionCommandQueue.takeAt(index);
 | |
|             if (t == 'R') {
 | |
|                 command->success = true;
 | |
|                 command->reply = message;
 | |
|             } else {
 | |
|                 command->success = false;
 | |
|                 command->errorMessage = message;
 | |
|             }
 | |
|             if (debug)
 | |
|                 qDebug("### Completed extension command '%s', token=%d, pending=%d",
 | |
|                        command->command.constData(), command->token, m_extensionCommandQueue.size());
 | |
|             if (command->handler)
 | |
|                 (this->*(command->handler))(command);
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (what == "debuggee_output") {
 | |
|         showMessage(StringFromBase64EncodedUtf16(message), AppOutput);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (what == "event") {
 | |
|         showStatusMessage(QString::fromAscii(message),  5000);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (what == "session_accessible") {
 | |
|         if (!m_accessible) {
 | |
|             m_accessible = true;
 | |
|             handleSessionAccessible(message.toULong());
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (what == "session_inaccessible") {
 | |
|         if (m_accessible) {
 | |
|             m_accessible = false;
 | |
|             handleSessionInaccessible(message.toULong());
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (what == "session_idle") {
 | |
|         handleSessionIdle(message);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (what == "exception") {
 | |
|         WinException exception;
 | |
|         GdbMi gdbmi;
 | |
|         gdbmi.fromString(message);
 | |
|         exception.fromGdbMI(gdbmi);
 | |
|         const QString message = exception.toString(true);
 | |
|         showStatusMessage(message);
 | |
| #ifdef Q_OS_WIN // Report C++ exception in application output as well.
 | |
|         if (exception.exceptionCode == winExceptionCppException)
 | |
|             showMessage(message + QLatin1Char('\n'), AppOutput);
 | |
| #endif
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     return;
 | |
| }
 | |
| 
 | |
| // Check for a CDB prompt '0:000> ' ('process:thread> ')..no regexps for QByteArray...
 | |
| enum { CdbPromptLength = 7 };
 | |
| 
 | |
| static inline bool isCdbPrompt(const QByteArray &c)
 | |
| {
 | |
|     return c.size() >= CdbPromptLength && c.at(6) == ' ' && c.at(5) == '>' && c.at(1) == ':'
 | |
|             && std::isdigit(c.at(0)) &&  std::isdigit(c.at(2)) && std::isdigit(c.at(3))
 | |
|             && std::isdigit(c.at(4));
 | |
| }
 | |
| 
 | |
| // Check for '<token>32>' or '<token>32<'
 | |
| static inline bool checkCommandToken(const QByteArray &tokenPrefix, const QByteArray &c,
 | |
|                                   int *token, bool *isStart)
 | |
| {
 | |
|     *token = 0;
 | |
|     *isStart = false;
 | |
|     const int tokenPrefixSize = tokenPrefix.size();
 | |
|     const int size = c.size();
 | |
|     if (size < tokenPrefixSize + 2 || !std::isdigit(c.at(tokenPrefixSize)))
 | |
|         return false;
 | |
|     switch (c.at(size - 1)) {
 | |
|     case '>':
 | |
|         *isStart = false;
 | |
|         break;
 | |
|     case '<':
 | |
|         *isStart = true;
 | |
|         break;
 | |
|     default:
 | |
|         return false;
 | |
|     }
 | |
|     if (!c.startsWith(tokenPrefix))
 | |
|         return false;
 | |
|     bool ok;
 | |
|     *token = c.mid(tokenPrefixSize, size - tokenPrefixSize - 1).toInt(&ok);
 | |
|     return ok;
 | |
| }
 | |
| 
 | |
| void CdbEngine::parseOutputLine(QByteArray line)
 | |
| {
 | |
|     // The hooked output callback in the extension suppresses prompts,
 | |
|     // it should happen only in initial and exit stages. Note however that
 | |
|     // if the output is not hooked, sequences of prompts are possible which
 | |
|     // can mix things up.
 | |
|     while (isCdbPrompt(line))
 | |
|         line.remove(0, CdbPromptLength);
 | |
|     // An extension notification (potentially consisting of several chunks)
 | |
|     if (line.startsWith(m_creatorExtPrefix)) {
 | |
|         // "<qtcreatorcdbext>|type_char|token|remainingChunks|serviceName|message"
 | |
|         const char type = line.at(m_creatorExtPrefix.size());
 | |
|         // integer token
 | |
|         const int tokenPos = m_creatorExtPrefix.size() + 2;
 | |
|         const int tokenEndPos = line.indexOf('|', tokenPos);
 | |
|         QTC_ASSERT(tokenEndPos != -1, return)
 | |
|         const int token = line.mid(tokenPos, tokenEndPos - tokenPos).toInt();
 | |
|         // remainingChunks
 | |
|         const int remainingChunksPos = tokenEndPos + 1;
 | |
|         const int remainingChunksEndPos = line.indexOf('|', remainingChunksPos);
 | |
|         QTC_ASSERT(remainingChunksEndPos != -1, return)
 | |
|         const int remainingChunks = line.mid(remainingChunksPos, remainingChunksEndPos - remainingChunksPos).toInt();
 | |
|         // const char 'serviceName'
 | |
|         const int whatPos = remainingChunksEndPos + 1;
 | |
|         const int whatEndPos = line.indexOf('|', whatPos);
 | |
|         QTC_ASSERT(whatEndPos != -1, return)
 | |
|         const QByteArray what = line.mid(whatPos, whatEndPos - whatPos);
 | |
|         // Build up buffer, call handler once last chunk was encountered
 | |
|         m_extensionMessageBuffer += line.mid(whatEndPos + 1);
 | |
|         if (remainingChunks == 0) {
 | |
|             handleExtensionMessage(type, token, what, m_extensionMessageBuffer);
 | |
|             m_extensionMessageBuffer.clear();
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
|     // Check for command start/end tokens within which the builtin command
 | |
|     // output is enclosed
 | |
|     int token = 0;
 | |
|     bool isStartToken = false;
 | |
|     const bool isCommandToken = checkCommandToken(m_tokenPrefix, line, &token, &isStartToken);
 | |
|     if (debug > 1)
 | |
|         qDebug("Reading CDB stdout '%s',\n  isCommand=%d, token=%d, isStart=%d, current=%d",
 | |
|                line.constData(), isCommandToken, token, isStartToken, m_currentBuiltinCommandIndex);
 | |
| 
 | |
|     // If there is a current command, wait for end of output indicated by token,
 | |
|     // command, trigger handler and finish, else append to its output.
 | |
|     if (m_currentBuiltinCommandIndex != -1) {
 | |
|         QTC_ASSERT(!isStartToken && m_currentBuiltinCommandIndex < m_builtinCommandQueue.size(), return; );
 | |
|         const CdbBuiltinCommandPtr ¤tCommand = m_builtinCommandQueue.at(m_currentBuiltinCommandIndex);
 | |
|         if (isCommandToken) {
 | |
|             // Did the command finish? Invoke callback and remove from queue.
 | |
|             if (debug)
 | |
|                 qDebug("### Completed builtin command '%s', token=%d, %d lines, pending=%d",
 | |
|                        currentCommand->command.constData(), currentCommand->token,
 | |
|                        currentCommand->reply.size(), m_builtinCommandQueue.size() - 1);
 | |
|             QTC_ASSERT(token == currentCommand->token, return; );
 | |
|             if (currentCommand->handler)
 | |
|                 (this->*(currentCommand->handler))(currentCommand);
 | |
|             m_builtinCommandQueue.removeAt(m_currentBuiltinCommandIndex);
 | |
|             m_currentBuiltinCommandIndex = -1;
 | |
|         } else {
 | |
|             // Record output of current command
 | |
|             currentCommand->reply.push_back(line);
 | |
|         }
 | |
|         return;
 | |
|     } // m_currentCommandIndex
 | |
|     if (isCommandToken) {
 | |
|         // Beginning command token encountered, start to record output.
 | |
|         const int index = indexOfCommand(m_builtinCommandQueue, token);
 | |
|         QTC_ASSERT(isStartToken && index != -1, return; );
 | |
|         m_currentBuiltinCommandIndex = index;
 | |
|         const CdbBuiltinCommandPtr ¤tCommand = m_builtinCommandQueue.at(m_currentBuiltinCommandIndex);
 | |
|         if (debug)
 | |
|             qDebug("### Gathering output for '%s' token %d", currentCommand->command.constData(), currentCommand->token);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     showMessage(QString::fromLocal8Bit(line), LogMisc);
 | |
| }
 | |
| 
 | |
| void CdbEngine::readyReadStandardOut()
 | |
| {
 | |
|     if (m_ignoreCdbOutput)
 | |
|         return;
 | |
|     m_outputBuffer += m_process.readAllStandardOutput();
 | |
|     // Split into lines and parse line by line.
 | |
|     while (true) {
 | |
|         const int endOfLinePos = m_outputBuffer.indexOf('\n');
 | |
|         if (endOfLinePos == -1) {
 | |
|             break;
 | |
|         } else {
 | |
|             // Check for '\r\n'
 | |
|             QByteArray line = m_outputBuffer.left(endOfLinePos);
 | |
|             if (!line.isEmpty() && line.at(line.size() - 1) == '\r')
 | |
|                 line.truncate(line.size() - 1);
 | |
|             parseOutputLine(line);
 | |
|             m_outputBuffer.remove(0, endOfLinePos + 1);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::readyReadStandardError()
 | |
| {
 | |
|     showMessage(QString::fromLocal8Bit(m_process.readAllStandardError()), LogError);
 | |
| }
 | |
| 
 | |
| void CdbEngine::processError()
 | |
| {
 | |
|     showMessage(m_process.errorString(), LogError);
 | |
| }
 | |
| 
 | |
| #if 0
 | |
| // Join breakpoint ids for a multi-breakpoint id commands like 'bc', 'be', 'bd'
 | |
| static QByteArray multiBreakpointCommand(const char *cmdC, const Breakpoints &bps)
 | |
| {
 | |
|     QByteArray cmd(cmdC);
 | |
|     ByteArrayInputStream str(cmd);
 | |
|     foreach(const BreakpointData *bp, bps)
 | |
|         str << ' ' << bp->bpNumber;
 | |
|     return cmd;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| bool CdbEngine::stateAcceptsBreakpointChanges() const
 | |
| {
 | |
|     switch (state()) {
 | |
|     case InferiorRunOk:
 | |
|     case InferiorStopOk:
 | |
|     return true;
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool CdbEngine::acceptsBreakpoint(BreakpointId id) const
 | |
| {
 | |
|     const BreakpointParameters &data = breakHandler()->breakpointData(id);
 | |
|     if (!DebuggerEngine::isCppBreakpoint(data))
 | |
|         return false;
 | |
|     switch (data.type) {
 | |
|     case UnknownType:
 | |
|     case BreakpointAtFork:
 | |
|     //case BreakpointAtVFork:
 | |
|     case BreakpointAtSysCall:
 | |
|         return false;
 | |
|     case Watchpoint:
 | |
|     case BreakpointByFileAndLine:
 | |
|     case BreakpointByFunction:
 | |
|     case BreakpointByAddress:
 | |
|     case BreakpointAtThrow:
 | |
|     case BreakpointAtCatch:
 | |
|     case BreakpointAtMain:
 | |
|     case BreakpointAtExec:
 | |
|         break;
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void CdbEngine::attemptBreakpointSynchronization()
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("attemptBreakpointSynchronization in %s", stateName(state()));
 | |
|     // Check if there is anything to be done at all.
 | |
|     BreakHandler *handler = breakHandler();
 | |
|     // Take ownership of the breakpoint. Requests insertion. TODO: Cpp only?
 | |
|     foreach (BreakpointId id, handler->unclaimedBreakpointIds())
 | |
|         if (acceptsBreakpoint(id))
 | |
|             handler->setEngine(id, this);
 | |
| 
 | |
|     // Quick check: is there a need to change something? - Populate module cache
 | |
|     bool changed = false;
 | |
|     const BreakpointIds ids = handler->engineBreakpointIds(this);
 | |
|     foreach (BreakpointId id, ids) {
 | |
|         switch (handler->state(id)) {
 | |
|         case BreakpointInsertRequested:
 | |
|         case BreakpointRemoveRequested:
 | |
|         case BreakpointChangeRequested:
 | |
|             changed = true;
 | |
|             break;
 | |
|         case BreakpointInserted: {
 | |
|             // Collect the new modules matching the files.
 | |
|             // In the future, that information should be obtained from the build system.
 | |
|             const BreakpointParameters &data = handler->breakpointData(id);
 | |
|             if (data.type == BreakpointByFileAndLine && !data.module.isEmpty())
 | |
|                 m_fileNameModuleHash.insert(data.fileName, data.module);
 | |
|         }
 | |
|         break;
 | |
|         default:
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (debugBreakpoints)
 | |
|         qDebug("attemptBreakpointSynchronizationI %dms accessible=%d, %s %d breakpoints, changed=%d",
 | |
|                elapsedLogTime(), m_accessible, stateName(state()), ids.size(), changed);
 | |
|     if (!changed)
 | |
|         return;
 | |
| 
 | |
|     if (!m_accessible) {
 | |
|         // No nested calls.
 | |
|         if (m_specialStopMode != SpecialStopSynchronizeBreakpoints)
 | |
|             doInterruptInferior(SpecialStopSynchronizeBreakpoints);
 | |
|         return;
 | |
|     }
 | |
|     // Add/Change breakpoints and store pending ones in map, since
 | |
|     // Breakhandler::setResponse() on pending breakpoints clears the pending flag.
 | |
|     // handleBreakPoints will the complete that information and set it on the break handler.
 | |
|     bool addedChanged = false;
 | |
|     foreach (BreakpointId id, ids) {
 | |
|         BreakpointParameters parameters = handler->breakpointData(id);
 | |
|         BreakpointResponse response;
 | |
|         response.fromParameters(parameters);
 | |
|         // If we encountered that file and have a module for it: Add it.
 | |
|         if (parameters.type == BreakpointByFileAndLine && parameters.module.isEmpty()) {
 | |
|             const QHash<QString, QString>::const_iterator it = m_fileNameModuleHash.constFind(parameters.fileName);
 | |
|             if (it != m_fileNameModuleHash.constEnd())
 | |
|                 parameters.module = it.value();
 | |
|         }
 | |
|         switch (handler->state(id)) {
 | |
|         case BreakpointInsertRequested:
 | |
|             postCommand(cdbAddBreakpointCommand(parameters, m_sourcePathMappings, id, false), 0);
 | |
|             if (!parameters.enabled)
 | |
|                 postCommand("bd " + QByteArray::number(id), 0);
 | |
|             handler->notifyBreakpointInsertProceeding(id);
 | |
|             handler->notifyBreakpointInsertOk(id);
 | |
|             m_pendingBreakpointMap.insert(id, response);
 | |
|             addedChanged = true;
 | |
|             // Ensure enabled/disabled is correct in handler and line number is there.
 | |
|             handler->setResponse(id, response);
 | |
|             if (debugBreakpoints)
 | |
|                 qDebug("Adding %llu %s\n", id, qPrintable(response.toString()));
 | |
|             break;
 | |
|         case BreakpointChangeRequested:
 | |
|             handler->notifyBreakpointChangeProceeding(id);
 | |
|             if (debugBreakpoints)
 | |
|                 qDebug("Changing %llu:\n    %s\nTo %s\n", id, qPrintable(handler->response(id).toString()),
 | |
|                        qPrintable(parameters.toString()));
 | |
|             if (parameters.enabled != handler->response(id).enabled) {
 | |
|                 // Change enabled/disabled breakpoints without triggering update.
 | |
|                 postCommand((parameters.enabled ? "be " : "bd ") + QByteArray::number(id), 0);
 | |
|                 response.pending = false;
 | |
|                 response.enabled = parameters.enabled;
 | |
|                 handler->setResponse(id, response);
 | |
|             } else {
 | |
|                 // Delete and re-add, triggering update
 | |
|                 addedChanged = true;
 | |
|                 postCommand("bc " + QByteArray::number(id), 0);
 | |
|                 postCommand(cdbAddBreakpointCommand(parameters, m_sourcePathMappings, id, false), 0);
 | |
|                 m_pendingBreakpointMap.insert(id, response);
 | |
|             }
 | |
|             handler->notifyBreakpointChangeOk(id);
 | |
|             break;
 | |
|         case BreakpointRemoveRequested:
 | |
|             postCommand("bc " + QByteArray::number(id), 0);
 | |
|             handler->notifyBreakpointRemoveProceeding(id);
 | |
|             handler->notifyBreakpointRemoveOk(id);
 | |
|             m_pendingBreakpointMap.remove(id);
 | |
|             break;
 | |
|         default:
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
|     // List breakpoints and send responses
 | |
|     if (addedChanged)
 | |
|         postCommandSequence(CommandListBreakPoints);
 | |
| }
 | |
| 
 | |
| // Pass a file name through source mapping and normalize upper/lower case (for the editor
 | |
| // manager to correctly process it) and convert to clean path.
 | |
| CdbEngine::NormalizedSourceFileName CdbEngine::sourceMapNormalizeFileNameFromDebugger(const QString &f)
 | |
| {
 | |
|     // 1) Check cache.
 | |
|     QMap<QString, NormalizedSourceFileName>::const_iterator it = m_normalizedFileCache.constFind(f);
 | |
|     if (it != m_normalizedFileCache.constEnd())
 | |
|         return it.value();
 | |
|     if (debugSourceMapping)
 | |
|         qDebug(">sourceMapNormalizeFileNameFromDebugger %s", qPrintable(f));
 | |
|     // Do we have source path mappings? ->Apply.
 | |
|     const QString fileName = cdbSourcePathMapping(QDir::toNativeSeparators(f), m_sourcePathMappings,
 | |
|                                                   DebuggerToSource);
 | |
|     // Up/lower case normalization according to Windows.
 | |
| #ifdef Q_OS_WIN
 | |
|     QString normalized = winNormalizeFileName(fileName);
 | |
| #else
 | |
|     QString normalized = fileName;
 | |
| #endif
 | |
|     if (debugSourceMapping)
 | |
|         qDebug(" sourceMapNormalizeFileNameFromDebugger %s->%s", qPrintable(fileName), qPrintable(normalized));
 | |
|     // Check if it really exists, that is normalize worked and QFileInfo confirms it.
 | |
|     const bool exists = !normalized.isEmpty() && QFileInfo(normalized).isFile();
 | |
|     NormalizedSourceFileName result(QDir::cleanPath(normalized.isEmpty() ? fileName : normalized), exists);
 | |
|     if (!exists) {
 | |
|         // At least upper case drive letter if failed.
 | |
|         if (result.fileName.size() > 2 && result.fileName.at(1) == QLatin1Char(':'))
 | |
|             result.fileName[0] = result.fileName.at(0).toUpper();
 | |
|     }
 | |
|     m_normalizedFileCache.insert(f, result);
 | |
|     if (debugSourceMapping)
 | |
|         qDebug("<sourceMapNormalizeFileNameFromDebugger %s %d", qPrintable(result.fileName), result.exists);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| // Parse frame from GDBMI. Duplicate of the gdb code, but that
 | |
| // has more processing.
 | |
| static StackFrames parseFrames(const GdbMi &gdbmi)
 | |
| {
 | |
|     StackFrames rc;
 | |
|     const int count = gdbmi.childCount();
 | |
|     rc.reserve(count);
 | |
|     for (int i = 0; i  < count; i++) {
 | |
|         const GdbMi &frameMi = gdbmi.childAt(i);
 | |
|         StackFrame frame;
 | |
|         frame.level = i;
 | |
|         const GdbMi fullName = frameMi.findChild("fullname");
 | |
|         if (fullName.isValid()) {
 | |
|             frame.file = QFile::decodeName(fullName.data());
 | |
|             frame.line = frameMi.findChild("line").data().toInt();
 | |
|             frame.usable = false; // To be decided after source path mapping.
 | |
|         }
 | |
|         frame.function = QLatin1String(frameMi.findChild("func").data());
 | |
|         frame.from = QLatin1String(frameMi.findChild("from").data());
 | |
|         frame.address = frameMi.findChild("addr").data().toULongLong(0, 16);
 | |
|         rc.push_back(frame);
 | |
|     }
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| unsigned CdbEngine::parseStackTrace(const GdbMi &data, bool sourceStepInto)
 | |
| {
 | |
|     // Parse frames, find current. Special handling for step into:
 | |
|     // When stepping into on an actual function (source mode) by executing 't', an assembler
 | |
|     // frame pointing at the jmp instruction is hit (noticeable by top function being
 | |
|     // 'ILT+'). If that is the case, execute another 't' to step into the actual function.    .
 | |
|     // Note that executing 't 2' does not work since it steps 2 instructions on a non-call code line.
 | |
|     int current = -1;
 | |
|     StackFrames frames = parseFrames(data);
 | |
|     const int count = frames.size();
 | |
|     for (int i = 0; i < count; i++) {
 | |
|         const bool hasFile = !frames.at(i).file.isEmpty();
 | |
|         // jmp-frame hit by step into, do another 't' and abort sequence.
 | |
|         if (!hasFile && i == 0 && sourceStepInto && frames.at(i).function.contains(QLatin1String("ILT+"))) {
 | |
|             showMessage(QString::fromAscii("Step into: Call instruction hit, performing additional step..."), LogMisc);
 | |
|             return ParseStackStepInto;
 | |
|         }
 | |
|         if (hasFile) {
 | |
|             const NormalizedSourceFileName fileName = sourceMapNormalizeFileNameFromDebugger(frames.at(i).file);
 | |
|             frames[i].file = fileName.fileName;
 | |
|             frames[i].usable = fileName.exists;
 | |
|             if (current == -1 && frames[i].usable)
 | |
|                 current = i;
 | |
|         }
 | |
|     }
 | |
|     if (count && current == -1) // No usable frame, use assembly.
 | |
|         current = 0;
 | |
|     // Set
 | |
|     stackHandler()->setFrames(frames);
 | |
|     activateFrame(current);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleStackTrace(const CdbExtensionCommandPtr &command)
 | |
| {
 | |
|     if (command->success) {
 | |
|         GdbMi data;
 | |
|         data.fromString(command->reply);
 | |
|         parseStackTrace(data, false);
 | |
|         postCommandSequence(command->commandSequence);
 | |
|     } else {
 | |
|         showMessage(command->errorMessage, LogError);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleExpression(const CdbExtensionCommandPtr &command)
 | |
| {
 | |
|     int value = 0;
 | |
|     if (command->success) {
 | |
|         value = command->reply.toInt();
 | |
|     } else {
 | |
|         showMessage(command->errorMessage, LogError);
 | |
|     }
 | |
|     // Is this a conditional breakpoint?
 | |
|     if (command->cookie.isValid() && qVariantCanConvert<ConditionalBreakPointCookie>(command->cookie)) {
 | |
|         const ConditionalBreakPointCookie cookie = qvariant_cast<ConditionalBreakPointCookie>(command->cookie);
 | |
|         const QString message = value ?
 | |
|             tr("Value %1 obtained from evaluating the condition of breakpoint %2, stopping.").
 | |
|             arg(value).arg(cookie.id) :
 | |
|             tr("Value 0 obtained from evaluating the condition of breakpoint %1, continuing.").
 | |
|             arg(cookie.id);
 | |
|         showMessage(message, LogMisc);
 | |
|         // Stop if evaluation is true, else continue
 | |
|         if (value) {
 | |
|             processStop(cookie.stopReason, true);
 | |
|         } else {
 | |
|             postCommand("g", 0);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::evaluateExpression(QByteArray exp, const QVariant &cookie)
 | |
| {
 | |
|     if (exp.contains(' ') && !exp.startsWith('"')) {
 | |
|         exp.prepend('"');
 | |
|         exp.append('"');
 | |
|     }
 | |
|     postExtensionCommand("expression", exp, 0, &CdbEngine::handleExpression, 0, cookie);
 | |
| }
 | |
| 
 | |
| void CdbEngine::dummyHandler(const CdbBuiltinCommandPtr &command)
 | |
| {
 | |
|     postCommandSequence(command->commandSequence);
 | |
| }
 | |
| 
 | |
| // Post a sequence of standard commands: Trigger next once one completes successfully
 | |
| void CdbEngine::postCommandSequence(unsigned mask)
 | |
| {
 | |
|     if (debug)
 | |
|         qDebug("postCommandSequence 0x%x\n", mask);
 | |
| 
 | |
|     if (!mask)
 | |
|         return;
 | |
|     if (mask & CommandListThreads) {
 | |
|         postExtensionCommand("threads", QByteArray(), 0, &CdbEngine::handleThreads, mask & ~CommandListThreads);
 | |
|         return;
 | |
|     }
 | |
|     if (mask & CommandListStack) {
 | |
|         postExtensionCommand("stack", QByteArray(), 0, &CdbEngine::handleStackTrace, mask & ~CommandListStack);
 | |
|         return;
 | |
|     }
 | |
|     if (mask & CommandListRegisters) {
 | |
|         QTC_ASSERT(threadsHandler()->currentThread() >= 0,  return; )
 | |
|         postExtensionCommand("registers", QByteArray(), 0, &CdbEngine::handleRegisters, mask & ~CommandListRegisters);
 | |
|         return;
 | |
|     }
 | |
|     if (mask & CommandListModules) {
 | |
|         postExtensionCommand("modules", QByteArray(), 0, &CdbEngine::handleModules, mask & ~CommandListModules);
 | |
|         return;
 | |
|     }
 | |
|     if (mask & CommandListBreakPoints) {
 | |
|         postExtensionCommand("breakpoints", QByteArray("-v"), 0,
 | |
|                              &CdbEngine::handleBreakPoints, mask & ~CommandListBreakPoints);
 | |
|         return;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleWidgetAt(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     bool success = false;
 | |
|     QString message;
 | |
|     do {
 | |
|         if (!reply->success) {
 | |
|             message = QString::fromAscii(reply->errorMessage);
 | |
|             break;
 | |
|         }
 | |
|         // Should be "namespace::QWidget:0x555"
 | |
|         QString watchExp = QString::fromAscii(reply->reply);
 | |
|         const int sepPos = watchExp.lastIndexOf(QLatin1Char(':'));
 | |
|         if (sepPos == -1) {
 | |
|             message = QString::fromAscii("Invalid output: %1").arg(watchExp);
 | |
|             break;
 | |
|         }
 | |
|         // 0x000 -> nothing found
 | |
|         if (!watchExp.mid(sepPos + 1).toULongLong(0, 0)) {
 | |
|             message = QString::fromAscii("No widget could be found at %1, %2.").arg(m_watchPointX).arg(m_watchPointY);
 | |
|             break;
 | |
|         }
 | |
|         // Turn into watch expression: "*(namespace::QWidget*)0x555"
 | |
|         watchExp.replace(sepPos, 1, QLatin1String("*)"));
 | |
|         watchExp.insert(0, QLatin1String("*("));
 | |
|         watchHandler()->watchExpression(watchExp);
 | |
|         success = true;
 | |
|     } while (false);
 | |
|     if (!success)
 | |
|         showMessage(message, LogWarning);
 | |
|     m_watchPointX = m_watchPointY = 0;
 | |
| }
 | |
| 
 | |
| static inline void formatCdbBreakPointResponse(BreakpointId id, const BreakpointResponse &r,
 | |
|                                                   QTextStream &str)
 | |
| {
 | |
|     str << "Obtained breakpoint " << id << " (#" << r.number << ')';
 | |
|     if (r.pending) {
 | |
|         str << ", pending";
 | |
|     } else {
 | |
|         str.setIntegerBase(16);
 | |
|         str << ", at 0x" << r.address;
 | |
|         str.setIntegerBase(10);
 | |
|     }
 | |
|     if (!r.enabled)
 | |
|         str << ", disabled";
 | |
|     if (!r.module.isEmpty())
 | |
|         str << ", module: '" << r.module << '\'';
 | |
|     str << '\n';
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleBreakPoints(const CdbExtensionCommandPtr &reply)
 | |
| {
 | |
|     if (debugBreakpoints)
 | |
|         qDebug("CdbEngine::handleBreakPoints: success=%d: %s", reply->success, reply->reply.constData());
 | |
|     if (!reply->success) {
 | |
|         showMessage(QString::fromAscii(reply->errorMessage), LogError);
 | |
|         return;
 | |
|     }
 | |
|     GdbMi value;
 | |
|     value.fromString(reply->reply);
 | |
|     if (value.type() != GdbMi::List) {
 | |
|         showMessage(QString::fromAscii("Unabled to parse breakpoints reply"), LogError);
 | |
|         return;
 | |
|     }
 | |
|     handleBreakPoints(value);
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleBreakPoints(const GdbMi &value)
 | |
| {
 | |
|     // Report all obtained parameters back. Note that not all parameters are reported
 | |
|     // back, so, match by id and complete
 | |
|     if (debugBreakpoints)
 | |
|         qDebug("\nCdbEngine::handleBreakPoints with %d", value.childCount());
 | |
|     QString message;
 | |
|     QTextStream str(&message);
 | |
|     BreakHandler *handler = breakHandler();
 | |
|     foreach (const GdbMi &breakPointG, value.children()) {
 | |
|         BreakpointResponse reportedResponse;
 | |
|         const BreakpointId id = parseBreakPoint(breakPointG, &reportedResponse);
 | |
|         if (debugBreakpoints)
 | |
|             qDebug("  Parsed %llu: pending=%d %s\n", id, reportedResponse.pending,
 | |
|                    qPrintable(reportedResponse.toString()));
 | |
| 
 | |
|         if (!reportedResponse.pending) {
 | |
|             const PendingBreakPointMap::iterator it = m_pendingBreakpointMap.find(id);
 | |
|             if (it != m_pendingBreakpointMap.end()) {
 | |
|                 // Complete the response and set on handler.
 | |
|                 BreakpointResponse ¤tResponse = it.value();
 | |
|                 currentResponse.number = reportedResponse.number;
 | |
|                 currentResponse.address = reportedResponse.address;
 | |
|                 currentResponse.module = reportedResponse.module;
 | |
|                 currentResponse.pending = reportedResponse.pending;
 | |
|                 currentResponse.enabled = reportedResponse.enabled;
 | |
|                 formatCdbBreakPointResponse(id, currentResponse, str);
 | |
|                 if (debugBreakpoints)
 | |
|                     qDebug("  Setting for %llu: %s\n", id, qPrintable(currentResponse.toString()));
 | |
|                 handler->setResponse(id, currentResponse);
 | |
|                 m_pendingBreakpointMap.erase(it);
 | |
|             }
 | |
|         } // not pending reported
 | |
|     } // foreach
 | |
|     if (m_pendingBreakpointMap.empty()) {
 | |
|         str << QLatin1String("All breakpoints have been resolved.\n");
 | |
|     } else {
 | |
|         str << QString::fromLatin1("%1 breakpoint(s) pending...\n").arg(m_pendingBreakpointMap.size());
 | |
|     }
 | |
|     showMessage(message, LogMisc);
 | |
| }
 | |
| 
 | |
| void CdbEngine::watchPoint(const QPoint &p)
 | |
| {
 | |
|     m_watchPointX = p.x();
 | |
|     m_watchPointY = p.y();
 | |
|     switch (state()) {
 | |
|     case InferiorStopOk:
 | |
|         postWidgetAtCommand();
 | |
|         break;
 | |
|     case InferiorRunOk:
 | |
|         // "Select Widget to Watch" from a running application is currently not
 | |
|         // supported. It could be implemented via SpecialStopGetWidgetAt-mode,
 | |
|         // but requires some work as not to confuse the engine by state-change notifications
 | |
|         // emitted by the debuggee function call.
 | |
|         showMessage(tr("\"Select Widget to Watch\": Please stop the application first."), LogWarning);
 | |
|         break;
 | |
|     default:
 | |
|         showMessage(tr("\"Select Widget to Watch\": Not supported in state '%1'.").
 | |
|                     arg(QString::fromAscii(stateName(state()))), LogWarning);
 | |
|         break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CdbEngine::postWidgetAtCommand()
 | |
| {
 | |
|     QByteArray arguments = QByteArray::number(m_watchPointX);
 | |
|     arguments.append(' ');
 | |
|     arguments.append(QByteArray::number(m_watchPointY));
 | |
|     postExtensionCommand("widgetat", arguments, 0, &CdbEngine::handleWidgetAt, 0);
 | |
| }
 | |
| 
 | |
| void CdbEngine::handleCustomSpecialStop(const QVariant &v)
 | |
| {
 | |
|     if (qVariantCanConvert<MemoryChangeCookie>(v)) {
 | |
|         const MemoryChangeCookie changeData = qVariantValue<MemoryChangeCookie>(v);
 | |
|         postCommand(cdbWriteMemoryCommand(changeData.address, changeData.data), 0);
 | |
|         return;
 | |
|     }
 | |
|     if (qVariantCanConvert<MemoryViewCookie>(v)) {
 | |
|         postFetchMemory(qVariantValue<MemoryViewCookie>(v));
 | |
|         return;
 | |
|     }
 | |
| }
 | |
| 
 | |
| } // namespace Internal
 | |
| } // namespace Debugger
 |