| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2012-01-26 18:33:46 +01:00
										 |  |  | ** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2012-07-19 12:26:56 +02:00
										 |  |  | ** Contact: http://www.qt-project.org/
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** GNU Lesser General Public License Usage | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** 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.
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-12-17 17:14:20 +01:00
										 |  |  | ** In addition, as a special exception, Nokia gives you certain additional | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** rights. These rights are described in the Nokia Qt LGPL Exception | 
					
						
							| 
									
										
										
										
											2010-12-17 17:14:20 +01:00
										 |  |  | ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-04-13 08:42:33 +02:00
										 |  |  | ** 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. | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:22:55 +01:00
										 |  |  | #include "cdbengine.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-03 11:29:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "breakhandler.h"
 | 
					
						
							|  |  |  | #include "breakpoint.h"
 | 
					
						
							|  |  |  | #include "bytearrayinputstream.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:22:55 +01:00
										 |  |  | #include "cdboptions.h"
 | 
					
						
							|  |  |  | #include "cdboptionspage.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-03 11:29:02 +01:00
										 |  |  | #include "cdbparsehelpers.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #include "debuggeractions.h"
 | 
					
						
							|  |  |  | #include "debuggercore.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-03 11:29:02 +01:00
										 |  |  | #include "debuggerinternalconstants.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  | #include "debuggerrunner.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-03 11:29:02 +01:00
										 |  |  | #include "debuggerstartparameters.h"
 | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  | #include "debuggertooltipmanager.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-03 11:29:02 +01:00
										 |  |  | #include "disassembleragent.h"
 | 
					
						
							|  |  |  | #include "disassemblerlines.h"
 | 
					
						
							|  |  |  | #include "memoryagent.h"
 | 
					
						
							|  |  |  | #include "moduleshandler.h"
 | 
					
						
							|  |  |  | #include "registerhandler.h"
 | 
					
						
							|  |  |  | #include "stackframe.h"
 | 
					
						
							|  |  |  | #include "stackhandler.h"
 | 
					
						
							|  |  |  | #include "threadshandler.h"
 | 
					
						
							|  |  |  | #include "watchhandler.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #include "watchutils.h"
 | 
					
						
							|  |  |  | #include "gdb/gdbmi.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 16:22:41 +01:00
										 |  |  | #include "shared/cdbsymbolpathlisteditor.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-03 12:01:50 +01:00
										 |  |  | #include "shared/hostutils.h"
 | 
					
						
							| 
									
										
										
										
											2012-02-02 09:30:22 +01:00
										 |  |  | #include "procinterrupt.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 16:22:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-09 17:02:46 +02:00
										 |  |  | #include <TranslationUnit.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 16:22:41 +01:00
										 |  |  | #include <coreplugin/icore.h>
 | 
					
						
							| 
									
										
										
										
											2010-11-22 17:06:08 +01:00
										 |  |  | #include <texteditor/itexteditor.h>
 | 
					
						
							| 
									
										
										
										
											2011-02-28 16:50:14 +01:00
										 |  |  | #include <projectexplorer/abi.h>
 | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  | #include <projectexplorer/projectexplorerconstants.h>
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  | #include <utils/synchronousprocess.h>
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #include <utils/winutils.h>
 | 
					
						
							|  |  |  | #include <utils/qtcassert.h>
 | 
					
						
							|  |  |  | #include <utils/savedaction.h>
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | #include <utils/consoleprocess.h>
 | 
					
						
							| 
									
										
										
										
											2011-05-09 17:02:46 +02:00
										 |  |  | #include <utils/fileutils.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <cplusplus/findcdbbreakpoint.h>
 | 
					
						
							|  |  |  | #include <cplusplus/CppDocument.h>
 | 
					
						
							| 
									
										
										
										
											2012-03-14 10:25:55 +01:00
										 |  |  | #include <cpptools/ModelManagerInterface.h>
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-15 10:42:41 +01:00
										 |  |  | #include <QCoreApplication>
 | 
					
						
							|  |  |  | #include <QFileInfo>
 | 
					
						
							|  |  |  | #include <QDir>
 | 
					
						
							|  |  |  | #include <QDebug>
 | 
					
						
							|  |  |  | #include <QTextStream>
 | 
					
						
							|  |  |  | #include <QDateTime>
 | 
					
						
							|  |  |  | #include <QToolTip>
 | 
					
						
							|  |  |  | #include <QMainWindow>
 | 
					
						
							|  |  |  | #include <QMessageBox>
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <cctype>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-14 12:29:32 +01:00
										 |  |  | Q_DECLARE_METATYPE(Debugger::Internal::DisassemblerAgent*) | 
					
						
							|  |  |  | Q_DECLARE_METATYPE(Debugger::Internal::MemoryAgent*) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum { debug = 0 }; | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  | enum { debugLocals = 0 }; | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  | enum { debugSourceMapping = 0 }; | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  | enum { debugWatches = 0 }; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | enum { debugBreakpoints = 0 }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  | enum HandleLocalsFlags | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PartialLocalsUpdate = 0x1, | 
					
						
							|  |  |  |     LocalsUpdateForNewFrame = 0x2 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | #  define STATE_DEBUG(state, func, line, notifyFunc) qDebug("%s in %s at %s:%d", notifyFunc, stateName(state), func, line);
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | #  define STATE_DEBUG(state, func, line, notifyFunc)
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-04 15:08:31 +01:00
										 |  |  | /*!
 | 
					
						
							|  |  |  |     \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). | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  | using namespace ProjectExplorer; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | namespace Debugger { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | namespace Internal { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  | static const char localsPrefixC[] = "local."; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-02 12:43:42 +01:00
										 |  |  | struct MemoryViewCookie | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-14 12:29:32 +01:00
										 |  |  |     explicit MemoryViewCookie(MemoryAgent *a = 0, QObject *e = 0, | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                               quint64 addr = 0, quint64 l = 0) : | 
					
						
							| 
									
										
										
										
											2011-02-02 12:43:42 +01:00
										 |  |  |         agent(a), editorToken(e), address(addr), length(l) | 
					
						
							|  |  |  |     {} | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-14 12:29:32 +01:00
										 |  |  |     MemoryAgent *agent; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     QObject *editorToken; | 
					
						
							|  |  |  |     quint64 address; | 
					
						
							|  |  |  |     quint64 length; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  | struct MemoryChangeCookie | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     explicit MemoryChangeCookie(quint64 addr = 0, const QByteArray &d = QByteArray()) : | 
					
						
							|  |  |  |                                address(addr), data(d) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     quint64 address; | 
					
						
							|  |  |  |     QByteArray data; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  | struct ConditionalBreakPointCookie | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |     ConditionalBreakPointCookie(BreakpointModelId i = BreakpointModelId()) : id(i) {} | 
					
						
							|  |  |  |     BreakpointModelId id; | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |     GdbMi stopReason; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | } // namespace Internal
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } // namespace Debugger
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | Q_DECLARE_METATYPE(Debugger::Internal::MemoryViewCookie) | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  | Q_DECLARE_METATYPE(Debugger::Internal::MemoryChangeCookie) | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  | Q_DECLARE_METATYPE(Debugger::Internal::ConditionalBreakPointCookie) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Debugger { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | namespace Internal { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-04 09:40:51 +02:00
										 |  |  | static inline bool isCreatorConsole(const DebuggerStartParameters &sp, const CdbOptions &o) | 
					
						
							| 
									
										
										
										
											2011-02-02 12:43:42 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-05-04 09:40:51 +02:00
										 |  |  |     return !o.cdbConsole && sp.useTerminal | 
					
						
							|  |  |  |            && (sp.startMode == StartInternal || sp.startMode == StartExternal); | 
					
						
							| 
									
										
										
										
											2011-02-02 12:43:42 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-02 12:43:42 +01:00
										 |  |  | static QMessageBox * | 
					
						
							|  |  |  | nonModalMessageBox(QMessageBox::Icon icon, const QString &title, const QString &text) | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     QMessageBox *mb = new QMessageBox(icon, title, text, QMessageBox::Ok, | 
					
						
							|  |  |  |                                       debuggerCore()->mainWindow()); | 
					
						
							|  |  |  |     mb->setAttribute(Qt::WA_DeleteOnClose); | 
					
						
							|  |  |  |     mb->show(); | 
					
						
							|  |  |  |     return mb; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | // 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) : | 
					
						
							| 
									
										
										
										
											2011-02-02 12:43:42 +01:00
										 |  |  |         CdbCommandBase(cmd, token, flags, nc, cookie), handler(h) | 
					
						
							|  |  |  |     {} | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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 StartRemoteGdb: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Accessed by RunControlFactory
 | 
					
						
							| 
									
										
										
										
											2011-01-12 12:10:12 +01:00
										 |  |  | DebuggerEngine *createCdbEngine(const DebuggerStartParameters &sp, | 
					
						
							|  |  |  |     DebuggerEngine *masterEngine, QString *errorMessage) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef Q_OS_WIN
 | 
					
						
							|  |  |  |     CdbOptionsPage *op = CdbOptionsPage::instance(); | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  |     if (!op || !op->options()->isValid() || !validMode(sp.startMode)) { | 
					
						
							|  |  |  |         *errorMessage = QLatin1String("Internal error: Invalid start parameters passed for thre CDB engine."); | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-12 12:10:12 +01:00
										 |  |  |     return new CdbEngine(sp, masterEngine, op->options()); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-01-12 12:10:12 +01:00
										 |  |  |     Q_UNUSED(masterEngine) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     Q_UNUSED(sp) | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-01-12 12:10:12 +01:00
										 |  |  |     *errorMessage = QString::fromLatin1("Unsupported debug mode"); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool isCdbEngineEnabled() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef Q_OS_WIN
 | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  |     return CdbOptionsPage::instance() && CdbOptionsPage::instance()->options()->isValid(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-15 13:32:23 +02:00
										 |  |  | static inline QString msgNoCdbBinaryForToolChain(const Abi &tc) | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  |     return CdbEngine::tr("There is no CDB binary available for binaries in format '%1'").arg(tc.toString()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-15 13:32:23 +02:00
										 |  |  | static inline bool isMsvcFlavor(Abi::OSFlavor osf) | 
					
						
							| 
									
										
										
										
											2011-08-09 10:38:30 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-15 13:32:23 +02:00
										 |  |  |   return osf == Abi::WindowsMsvc2005Flavor | 
					
						
							|  |  |  |       || osf == Abi::WindowsMsvc2008Flavor | 
					
						
							| 
									
										
										
										
											2012-07-27 20:34:46 +02:00
										 |  |  |       || osf == Abi::WindowsMsvc2010Flavor | 
					
						
							|  |  |  |       || osf == Abi::WindowsMsvc2012Flavor; | 
					
						
							| 
									
										
										
										
											2011-08-09 10:38:30 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  | bool checkCdbConfiguration(const DebuggerStartParameters &sp, ConfigurationCheck *check) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef Q_OS_WIN
 | 
					
						
							| 
									
										
										
										
											2012-06-28 16:26:21 +02:00
										 |  |  |     const Abi abi = sp.toolChainAbi; | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  |     if (!isCdbEngineEnabled()) { | 
					
						
							| 
									
										
										
										
											2011-02-25 09:34:31 +01:00
										 |  |  |         check->errorDetails.push_back(CdbEngine::tr("The CDB debug engine required for %1 is currently disabled."). | 
					
						
							| 
									
										
										
										
											2012-06-28 10:00:04 +02:00
										 |  |  |                            arg(abi.toString())); | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  |         check->settingsCategory = QLatin1String(Debugger::Constants::DEBUGGER_SETTINGS_CATEGORY); | 
					
						
							|  |  |  |         check->settingsPage = CdbOptionsPage::settingsId(); | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!validMode(sp.startMode)) { | 
					
						
							| 
									
										
										
										
											2011-02-25 09:34:31 +01:00
										 |  |  |         check->errorDetails.push_back(CdbEngine::tr("The CDB engine does not support start mode %1.").arg(sp.startMode)); | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-28 10:00:04 +02:00
										 |  |  |     if (abi.binaryFormat() != Abi::PEFormat || abi.os() != Abi::WindowsOS) { | 
					
						
							| 
									
										
										
										
											2011-02-25 09:34:31 +01:00
										 |  |  |         check->errorDetails.push_back(CdbEngine::tr("The CDB debug engine does not support the %1 ABI."). | 
					
						
							| 
									
										
										
										
											2012-06-28 10:00:04 +02:00
										 |  |  |                                       arg(abi.toString())); | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-25 13:43:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-28 10:00:04 +02:00
										 |  |  |     if (sp.startMode == AttachCore && !isMsvcFlavor(abi.osFlavor())) { | 
					
						
							| 
									
										
										
										
											2011-08-09 10:38:30 +02:00
										 |  |  |         check->errorDetails.push_back(CdbEngine::tr("The CDB debug engine cannot debug gdb core files.")); | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-28 16:26:21 +02:00
										 |  |  |     if (sp.debuggerCommand.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2012-06-28 10:00:04 +02:00
										 |  |  |         check->errorDetails.push_back(msgNoCdbBinaryForToolChain(abi)); | 
					
						
							| 
									
										
										
										
											2011-09-29 17:50:51 +02:00
										 |  |  |         check->settingsCategory = QLatin1String(ProjectExplorer::Constants::PROJECTEXPLORER_SETTINGS_CATEGORY); | 
					
						
							|  |  |  |         check->settingsPage = QLatin1String(ProjectExplorer::Constants::PROJECTEXPLORER_SETTINGS_CATEGORY); | 
					
						
							| 
									
										
										
										
											2011-02-25 13:43:06 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     Q_UNUSED(sp); | 
					
						
							| 
									
										
										
										
											2011-02-25 09:34:31 +01:00
										 |  |  |     check->errorDetails.push_back(QString::fromLatin1("Unsupported debug mode")); | 
					
						
							| 
									
										
										
										
											2011-02-24 16:50:15 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 12:10:12 +01:00
										 |  |  | void addCdbOptionPages(QList<Core::IOptionsPage *> *opts) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-12 12:10:12 +01:00
										 |  |  | #ifdef Q_OS_WIN
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     opts->push_back(new CdbOptionsPage); | 
					
						
							| 
									
										
										
										
											2011-01-12 12:10:12 +01:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     Q_UNUSED(opts); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define QT_CREATOR_CDB_EXT "qtcreatorcdbext"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline Utils::SavedAction *theAssemblerAction() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |     return debuggerCore()->action(OperateByInstruction); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 12:10:12 +01:00
										 |  |  | CdbEngine::CdbEngine(const DebuggerStartParameters &sp, | 
					
						
							|  |  |  |         DebuggerEngine *masterEngine, const OptionsPtr &options) : | 
					
						
							| 
									
										
										
										
											2011-10-12 16:28:33 +02:00
										 |  |  |     DebuggerEngine(sp, CppLanguage, masterEngine), | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     m_creatorExtPrefix("<qtcreatorcdbext>|"), | 
					
						
							|  |  |  |     m_tokenPrefix("<token>"), | 
					
						
							|  |  |  |     m_options(options), | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     m_effectiveStartMode(NoStartMode), | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     m_accessible(false), | 
					
						
							|  |  |  |     m_specialStopMode(NoSpecialStop), | 
					
						
							|  |  |  |     m_nextCommandToken(0), | 
					
						
							|  |  |  |     m_currentBuiltinCommandIndex(-1), | 
					
						
							| 
									
										
										
										
											2011-11-22 15:50:49 +01:00
										 |  |  |     m_extensionCommandPrefixBA("!" QT_CREATOR_CDB_EXT "."), | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     m_operateByInstructionPending(true), | 
					
						
							|  |  |  |     m_operateByInstruction(true), // Default CDB setting
 | 
					
						
							|  |  |  |     m_notifyEngineShutdownOnTermination(false), | 
					
						
							|  |  |  |     m_hasDebuggee(false), | 
					
						
							| 
									
										
										
										
											2011-01-05 11:34:35 +01:00
										 |  |  |     m_elapsedLogTime(0), | 
					
						
							| 
									
										
										
										
											2011-01-17 15:18:13 +01:00
										 |  |  |     m_sourceStepInto(false), | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |     m_watchPointX(0), | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |     m_watchPointY(0), | 
					
						
							|  |  |  |     m_ignoreCdbOutput(false) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-16 14:13:07 +01:00
										 |  |  |     connect(theAssemblerAction(), SIGNAL(triggered(bool)), this, SLOT(operateByInstructionTriggered(bool))); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  | void CdbEngine::init() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_effectiveStartMode = NoStartMode; | 
					
						
							| 
									
										
										
										
											2011-04-28 10:02:34 +02:00
										 |  |  |     notifyInferiorPid(0); | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |     m_accessible = false; | 
					
						
							|  |  |  |     m_specialStopMode = NoSpecialStop; | 
					
						
							|  |  |  |     m_nextCommandToken  = 0; | 
					
						
							|  |  |  |     m_currentBuiltinCommandIndex = -1; | 
					
						
							| 
									
										
										
										
											2011-02-16 14:13:07 +01:00
										 |  |  |     m_operateByInstructionPending = theAssemblerAction()->isChecked(); | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |     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(); | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  |     m_customSpecialStopData.clear(); | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |     m_symbolAddressCache.clear(); | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  |     m_coreStopReason.reset(); | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // 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()))); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |     QTC_ASSERT(m_process.state() != QProcess::Running, Utils::SynchronousProcess::stopProcess(m_process)); | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | CdbEngine::~CdbEngine() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::operateByInstructionTriggered(bool operateByInstruction) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-13 12:56:42 +02:00
										 |  |  |     // To be set next time session becomes accessible
 | 
					
						
							|  |  |  |     m_operateByInstructionPending = operateByInstruction; | 
					
						
							|  |  |  |     if (state() == InferiorStopOk) | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |         syncOperateByInstruction(operateByInstruction); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::syncOperateByInstruction(bool operateByInstruction) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-16 14:13:07 +01:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("syncOperateByInstruction current: %d new %d", m_operateByInstruction, operateByInstruction); | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |     if (m_operateByInstruction == operateByInstruction) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |     QTC_ASSERT(m_accessible, return); | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |     m_operateByInstruction = operateByInstruction; | 
					
						
							|  |  |  |     postCommand(m_operateByInstruction ? QByteArray("l-t") : QByteArray("l+t"), 0); | 
					
						
							|  |  |  |     postCommand(m_operateByInstruction ? QByteArray("l-s") : QByteArray("l+s"), 0); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-17 10:08:57 +01:00
										 |  |  | bool CdbEngine::setToolTipExpression(const QPoint &mousePos, | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  |                                      TextEditor::ITextEditor *editor, | 
					
						
							|  |  |  |                                      const DebuggerToolTipContext &contextIn) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-22 17:06:08 +01:00
										 |  |  |     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) | 
					
						
							| 
									
										
										
										
											2011-02-17 10:08:57 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-11-22 17:06:08 +01:00
										 |  |  |     // Determine expression and function
 | 
					
						
							|  |  |  |     int line; | 
					
						
							|  |  |  |     int column; | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  |     DebuggerToolTipContext context = contextIn; | 
					
						
							| 
									
										
										
										
											2011-03-29 12:29:10 +02:00
										 |  |  |     QString exp = cppExpressionAt(editor, context.position, &line, &column, &context.function); | 
					
						
							| 
									
										
										
										
											2010-11-22 17:06:08 +01:00
										 |  |  |     // Are we in the current stack frame
 | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  |     if (context.function.isEmpty() || exp.isEmpty() || context.function != stackHandler()->currentFrame().function) | 
					
						
							| 
									
										
										
										
											2011-02-17 10:08:57 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-11-22 17:06:08 +01:00
										 |  |  |     // No numerical or any other expressions [yet]
 | 
					
						
							|  |  |  |     if (!(exp.at(0).isLetter() || exp.at(0) == QLatin1Char('_'))) | 
					
						
							| 
									
										
										
										
											2011-02-17 10:08:57 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2011-03-29 12:29:10 +02:00
										 |  |  |     // Can this be found as a local variable?
 | 
					
						
							|  |  |  |     const QByteArray localsPrefix(localsPrefixC); | 
					
						
							|  |  |  |     QByteArray iname = localsPrefix + exp.toAscii(); | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |     if (!watchHandler()->hasItem(iname)) { | 
					
						
							| 
									
										
										
										
											2011-03-29 12:29:10 +02:00
										 |  |  |         // Nope, try a 'local.this.m_foo'.
 | 
					
						
							|  |  |  |         exp.prepend(QLatin1String("this.")); | 
					
						
							|  |  |  |         iname.insert(localsPrefix.size(), "this."); | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |         if (!watchHandler()->hasItem(iname)) | 
					
						
							| 
									
										
										
										
											2011-03-29 12:29:10 +02:00
										 |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-05-27 11:02:57 +02:00
										 |  |  |     DebuggerToolTipWidget *tw = new DebuggerToolTipWidget; | 
					
						
							| 
									
										
										
										
											2011-02-17 10:08:57 +01:00
										 |  |  |     tw->setContext(context); | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |     tw->setDebuggerModel(LocalsType); | 
					
						
							| 
									
										
										
										
											2011-02-17 10:08:57 +01:00
										 |  |  |     tw->setExpression(exp); | 
					
						
							|  |  |  |     tw->acquireEngine(this); | 
					
						
							| 
									
										
										
										
											2011-02-21 16:45:07 +01:00
										 |  |  |     DebuggerToolTipManager::instance()->showToolTip(mousePos, editor, tw); | 
					
						
							| 
									
										
										
										
											2011-02-17 10:08:57 +01:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Determine full path to the CDB extension library.
 | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  | QString CdbEngine::extensionLibraryName(bool is64Bit) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     // Determine extension lib name and path to use
 | 
					
						
							|  |  |  |     QString rc; | 
					
						
							|  |  |  |     QTextStream(&rc) << QFileInfo(QCoreApplication::applicationDirPath()).path() | 
					
						
							| 
									
										
										
										
											2011-11-22 15:50:49 +01:00
										 |  |  |                      << "/lib/" << (is64Bit ? QT_CREATOR_CDB_EXT "64" : QT_CREATOR_CDB_EXT "32") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                      << '/' << 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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | // 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); | 
					
						
							| 
									
										
										
										
											2011-05-11 16:27:33 +02:00
										 |  |  |     connect(m_consoleStub.data(), SIGNAL(processError(QString)), | 
					
						
							|  |  |  |             SLOT(consoleStubError(QString))); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     connect(m_consoleStub.data(), SIGNAL(processStarted()), | 
					
						
							|  |  |  |             SLOT(consoleStubProcessStarted())); | 
					
						
							|  |  |  |     connect(m_consoleStub.data(), SIGNAL(wrapperStopped()), | 
					
						
							|  |  |  |             SLOT(consoleStubExited())); | 
					
						
							|  |  |  |     m_consoleStub->setWorkingDirectory(sp.workingDirectory); | 
					
						
							| 
									
										
										
										
											2011-02-28 11:48:33 +01:00
										 |  |  |     if (sp.environment.size()) | 
					
						
							|  |  |  |         m_consoleStub->setEnvironment(sp.environment); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-11 16:27:33 +02:00
										 |  |  | void CdbEngine::consoleStubError(const QString &msg) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     if (debug) | 
					
						
							| 
									
										
										
										
											2011-05-11 16:27:33 +02:00
										 |  |  |         qDebug("consoleStubProcessMessage() in %s %s", stateName(state()), qPrintable(msg)); | 
					
						
							|  |  |  |     if (state() == EngineSetupRequested) { | 
					
						
							|  |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupFailed") | 
					
						
							|  |  |  |         notifyEngineSetupFailed(); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-05-11 16:27:33 +02:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineIll") | 
					
						
							|  |  |  |         notifyEngineIll(); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-05-11 16:27:33 +02:00
										 |  |  |     nonModalMessageBox(QMessageBox::Critical, tr("Debugger Error"), msg); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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; | 
					
						
							| 
									
										
										
										
											2011-05-04 09:40:51 +02:00
										 |  |  |     attachParameters.useTerminal = false; | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     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))) | 
					
						
							| 
									
										
										
										
											2012-01-24 15:36:40 +01:00
										 |  |  |         m_options->toSettings(Core::ICore::settings()); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |     init(); | 
					
						
							|  |  |  |     if (!m_logTime.elapsed()) | 
					
						
							|  |  |  |         m_logTime.start(); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     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(); | 
					
						
							| 
									
										
										
										
											2011-05-04 09:40:51 +02:00
										 |  |  |     const bool launchConsole = isCreatorConsole(sp, *m_options); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     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); | 
					
						
							| 
									
										
										
										
											2011-01-19 13:01:50 +01:00
										 |  |  |     const QChar blank(QLatin1Char(' ')); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     // Start engine which will run until initial breakpoint:
 | 
					
						
							| 
									
										
										
										
											2011-02-23 16:56:14 +01:00
										 |  |  |     // Determine binary (force MSVC), extension lib name and path to use
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     // The extension is passed as relative name with the path variable set
 | 
					
						
							|  |  |  |     //(does not work with absolute path names)
 | 
					
						
							| 
									
										
										
										
											2012-06-28 10:00:04 +02:00
										 |  |  |     const QString executable = sp.debuggerCommand; | 
					
						
							| 
									
										
										
										
											2011-02-23 14:57:52 +01:00
										 |  |  |     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)); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     if (!extensionFi.isFile()) { | 
					
						
							|  |  |  |         *errorMessage = QString::fromLatin1("Internal error: The extension %1 cannot be found."). | 
					
						
							|  |  |  |                 arg(QDir::toNativeSeparators(extensionFi.absoluteFilePath())); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     const QString extensionFileName = extensionFi.fileName(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     // Prepare arguments
 | 
					
						
							|  |  |  |     QStringList arguments; | 
					
						
							| 
									
										
										
										
											2011-09-16 11:04:11 +02:00
										 |  |  |     const bool isRemote = sp.startMode == AttachToRemoteServer; | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     if (isRemote) { // Must be first
 | 
					
						
							|  |  |  |         arguments << QLatin1String("-remote") << sp.remoteChannel; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         arguments << (QLatin1String("-a") + extensionFileName); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     // Source line info/No terminal breakpoint / Pull extension
 | 
					
						
							|  |  |  |     arguments << QLatin1String("-lines") << QLatin1String("-G") | 
					
						
							|  |  |  |     // register idle (debuggee stop) notification
 | 
					
						
							|  |  |  |               << QLatin1String("-c") | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |               << QLatin1String(".idle_cmd ") + QString::fromLatin1(m_extensionCommandPrefixBA) + QLatin1String("idle"); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     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(';'))); | 
					
						
							| 
									
										
										
										
											2011-01-19 13:01:50 +01:00
										 |  |  |     // Compile argument string preserving quotes
 | 
					
						
							|  |  |  |     QString nativeArguments = m_options->additionalArguments; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     switch (sp.startMode) { | 
					
						
							|  |  |  |     case StartInternal: | 
					
						
							|  |  |  |     case StartExternal: | 
					
						
							| 
									
										
										
										
											2011-01-19 13:01:50 +01:00
										 |  |  |         if (!nativeArguments.isEmpty()) | 
					
						
							|  |  |  |             nativeArguments.push_back(blank); | 
					
						
							|  |  |  |         nativeArguments += QDir::toNativeSeparators(sp.executable); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2011-09-16 11:04:11 +02:00
										 |  |  |     case AttachToRemoteServer: | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     case AttachExternal: | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     case AttachCrashedExternal: | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         arguments << QLatin1String("-p") << QString::number(sp.attachPID); | 
					
						
							| 
									
										
										
										
											2011-05-12 17:19:42 +02:00
										 |  |  |         if (sp.startMode == AttachCrashedExternal) { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             arguments << QLatin1String("-e") << sp.crashParameter << QLatin1String("-g"); | 
					
						
							| 
									
										
										
										
											2011-05-12 17:19:42 +02:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2011-05-16 17:32:23 +02:00
										 |  |  |             if (isCreatorConsole(startParameters(), *m_options)) | 
					
						
							| 
									
										
										
										
											2011-05-12 17:19:42 +02:00
										 |  |  |                 arguments << QLatin1String("-pr") << QLatin1String("-pb"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  |     case AttachCore: | 
					
						
							|  |  |  |         arguments << QLatin1String("-z") << sp.coreFile; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     default: | 
					
						
							|  |  |  |         *errorMessage = QString::fromLatin1("Internal error: Unsupported start mode %1.").arg(sp.startMode); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-19 13:01:50 +01:00
										 |  |  |     if (!sp.processArgs.isEmpty()) { // Complete native argument string.
 | 
					
						
							|  |  |  |         if (!nativeArguments.isEmpty()) | 
					
						
							|  |  |  |             nativeArguments.push_back(blank); | 
					
						
							|  |  |  |         nativeArguments += sp.processArgs; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     const QString msg = QString::fromLatin1("Launching %1 %2\nusing %3 of %4."). | 
					
						
							|  |  |  |             arg(QDir::toNativeSeparators(executable), | 
					
						
							| 
									
										
										
										
											2011-01-19 13:01:50 +01:00
										 |  |  |                 arguments.join(QString(blank)) + blank + nativeArguments, | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                 QDir::toNativeSeparators(extensionFi.absoluteFilePath()), | 
					
						
							|  |  |  |                 extensionFi.lastModified().toString(Qt::SystemLocaleShortDate)); | 
					
						
							|  |  |  |     showMessage(msg, LogMisc); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_outputBuffer.clear(); | 
					
						
							| 
									
										
										
										
											2011-02-28 11:48:33 +01:00
										 |  |  |     const QStringList environment = sp.environment.size() == 0 ? | 
					
						
							|  |  |  |                                     QProcessEnvironment::systemEnvironment().toStringList() : | 
					
						
							|  |  |  |                                     sp.environment.toStringList(); | 
					
						
							|  |  |  |     m_process.setEnvironment(mergeEnvironment(environment, extensionFi.absolutePath())); | 
					
						
							| 
									
										
										
										
											2011-01-27 11:47:59 +01:00
										 |  |  |     if (!sp.workingDirectory.isEmpty()) | 
					
						
							|  |  |  |         m_process.setWorkingDirectory(sp.workingDirectory); | 
					
						
							| 
									
										
										
										
											2011-01-19 13:01:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 16:06:02 +01:00
										 |  |  | #ifdef Q_OS_WIN
 | 
					
						
							| 
									
										
										
										
											2011-01-19 13:01:50 +01:00
										 |  |  |     if (!nativeArguments.isEmpty()) // Appends
 | 
					
						
							|  |  |  |         m_process.setNativeArguments(nativeArguments); | 
					
						
							| 
									
										
										
										
											2010-11-18 16:06:02 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     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; | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     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); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupOk") | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |         notifyEngineSetupOk(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::setupInferior() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("setupInferior"); | 
					
						
							| 
									
										
										
										
											2011-09-29 15:12:43 +02:00
										 |  |  |     // QmlCppEngine expects the QML engine to be connected before any breakpoints are hit
 | 
					
						
							|  |  |  |     // (attemptBreakpointSynchronization() will be directly called then)
 | 
					
						
							|  |  |  |     attemptBreakpointSynchronization(); | 
					
						
							|  |  |  |     if (startParameters().breakOnMain) { | 
					
						
							|  |  |  |         const BreakpointParameters bp(BreakpointAtMain); | 
					
						
							|  |  |  |         postCommand(cdbAddBreakpointCommand(bp, m_sourcePathMappings, | 
					
						
							| 
									
										
										
										
											2012-05-20 21:15:30 +02:00
										 |  |  |                                             BreakpointModelId(quint16(-1)), true), 0); | 
					
						
							| 
									
										
										
										
											2011-05-04 15:40:15 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-23 08:54:24 +01:00
										 |  |  |     postCommand("sxn 0x4000001f", 0); // Do not break on WowX86 exceptions.
 | 
					
						
							| 
									
										
										
										
											2011-04-13 13:52:03 +02:00
										 |  |  |     postCommand(".asm source_line", 0); // Source line in assembly
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     postExtensionCommand("pid", QByteArray(), 0, &CdbEngine::handlePid); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::runEngine() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("runEngine"); | 
					
						
							| 
									
										
										
										
											2010-12-17 12:00:44 +01:00
										 |  |  |     foreach (const QString &breakEvent, m_options->breakEvents) | 
					
						
							|  |  |  |             postCommand(QByteArray("sxe ") + breakEvent.toAscii(), 0); | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  |     if (startParameters().startMode == AttachCore) { | 
					
						
							|  |  |  |         QTC_ASSERT(!m_coreStopReason.isNull(), return; ); | 
					
						
							|  |  |  |         notifyInferiorUnrunnable(); | 
					
						
							|  |  |  |         processStop(*m_coreStopReason, false); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         postCommand("g", 0); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-16 12:17:54 +01:00
										 |  |  | bool CdbEngine::commandsPending() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return !m_builtinCommandQueue.isEmpty() || !m_extensionCommandQueue.isEmpty(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 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"); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorShutdownOk") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         notifyInferiorShutdownOk(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-23 16:43:30 +02:00
										 |  |  |     if (m_accessible) { // except console.
 | 
					
						
							|  |  |  |         if (startParameters().startMode == AttachExternal || startParameters().startMode == AttachCrashedExternal) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             detachDebugger(); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorShutdownOk") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         notifyInferiorShutdownOk(); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |         // 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; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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) | 
					
						
							| 
									
										
										
										
											2010-12-16 12:17:54 +01:00
										 |  |  |         qDebug("CdbEngine::shutdownEngine in state '%s', process running %d," | 
					
						
							|  |  |  |                "accessible=%d,commands pending=%d", | 
					
						
							|  |  |  |                stateName(state()), isCdbProcessRunning(), m_accessible, | 
					
						
							|  |  |  |                commandsPending()); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!isCdbProcessRunning()) { // Direct launch: Terminated with process.
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineShutdownOk") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         notifyEngineShutdownOk(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-16 12:17:54 +01:00
										 |  |  |     // No longer trigger anything from messages
 | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |     m_ignoreCdbOutput = true; | 
					
						
							| 
									
										
										
										
											2010-12-16 12:17:54 +01:00
										 |  |  |     // Go for kill if there are commands pending.
 | 
					
						
							|  |  |  |     if (m_accessible && !commandsPending()) { | 
					
						
							| 
									
										
										
										
											2011-06-23 16:43:30 +02:00
										 |  |  |         // detach (except console): Wait for debugger to finish.
 | 
					
						
							|  |  |  |         if (startParameters().startMode == AttachExternal || startParameters().startMode == AttachCrashedExternal) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             detachDebugger(); | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |         // Remote requires a bit more force to quit.
 | 
					
						
							| 
									
										
										
										
											2011-09-16 11:04:11 +02:00
										 |  |  |         if (m_effectiveStartMode == AttachToRemoteServer) { | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |             postCommand(m_extensionCommandPrefixBA + "shutdownex", 0); | 
					
						
							|  |  |  |             postCommand("qq", 0); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             postCommand("q", 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         m_notifyEngineShutdownOnTermination = true; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         // Remote process. No can do, currently
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         m_notifyEngineShutdownOnTermination = true; | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |         Utils::SynchronousProcess::stopProcess(m_process); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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"); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineIll") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             notifyEngineIll(); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineShutdownOk") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             notifyEngineShutdownOk(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-03-14 11:43:33 +01:00
										 |  |  |         // 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(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::detachDebugger() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     postCommand(".detach", 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  | static inline bool isWatchIName(const QByteArray &iname) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return iname.startsWith("watch"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | void CdbEngine::updateWatchData(const WatchData &dataIn, | 
					
						
							|  |  |  |                                 const WatchUpdateFlags & flags) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |     if (debug || debugLocals || debugWatches) | 
					
						
							|  |  |  |         qDebug("CdbEngine::updateWatchData() %dms accessible=%d %s incr=%d: %s", | 
					
						
							|  |  |  |                elapsedLogTime(), m_accessible, stateName(state()), | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                flags.tryIncremental, | 
					
						
							|  |  |  |                qPrintable(dataIn.toString())); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |     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()) { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         WatchData data = dataIn; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         data.setAllUnneeded(); | 
					
						
							|  |  |  |         watchHandler()->insertData(data); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     updateLocalVariable(dataIn.iname); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  | 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")); | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |         watchHandler()->insertIncompleteData(item); | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |         showMessage(QString::fromLatin1("Unable to add watch item '%1'/'%2': %3"). | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |                     arg(QString::fromLatin1(item.iname), QString::fromLatin1(item.exp), | 
					
						
							| 
									
										
										
										
											2011-12-21 14:02:52 +01:00
										 |  |  |                         QString::fromLocal8Bit(reply->errorMessage)), LogError); | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-09 17:04:43 +01:00
										 |  |  | void CdbEngine::addLocalsOptions(ByteArrayInputStream &str) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-06 12:53:54 +01:00
										 |  |  |     if (debuggerCore()->boolSetting(VerboseLog)) | 
					
						
							|  |  |  |         str << blankSeparator << "-v"; | 
					
						
							| 
									
										
										
										
											2010-12-09 17:04:43 +01:00
										 |  |  |     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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | void CdbEngine::updateLocalVariable(const QByteArray &iname) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |     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; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |     str << blankSeparator << iname; | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |     postExtensionCommand(isWatch ? "watches" : "locals", | 
					
						
							|  |  |  |                          localsArguments, 0, | 
					
						
							|  |  |  |                          &CdbEngine::handleLocals, | 
					
						
							|  |  |  |                          0, QVariant(int(PartialLocalsUpdate))); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-16 17:33:30 +01:00
										 |  |  | bool CdbEngine::hasCapability(unsigned cap) const | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-01-16 17:33:30 +01:00
										 |  |  |     return cap & (DisassemblerCapability | RegisterCapability | 
					
						
							|  |  |  |            | ShowMemoryCapability | 
					
						
							| 
									
										
										
										
											2011-08-04 14:25:38 +02:00
										 |  |  |            |WatchpointByAddressCapability|JumpToLineCapability|AddWatcherCapability|WatchWidgetsCapability | 
					
						
							| 
									
										
										
										
											2011-04-01 12:50:57 +02:00
										 |  |  |            |ReloadModuleCapability | 
					
						
							| 
									
										
										
										
											2011-02-07 11:04:31 +01:00
										 |  |  |            |BreakOnThrowAndCatchCapability // Sort-of: Can break on throw().
 | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |            |BreakConditionCapability|TracePointCapability | 
					
						
							| 
									
										
										
										
											2011-04-01 19:47:39 +02:00
										 |  |  |            |BreakModuleCapability | 
					
						
							|  |  |  |            |OperateByInstructionCapability | 
					
						
							| 
									
										
										
										
											2011-11-30 11:51:23 +01:00
										 |  |  |            |RunToLineCapability | 
					
						
							| 
									
										
										
										
											2012-01-16 17:33:30 +01:00
										 |  |  |            |MemoryAddressCapability); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeStep() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-05 11:34:35 +01:00
										 |  |  |     if (!m_operateByInstruction) | 
					
						
							|  |  |  |         m_sourceStepInto = true; // See explanation at handleStackTrace().
 | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |     postCommand(QByteArray("t"), 0); // Step into-> t (trace)
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     notifyInferiorRunRequested(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeStepOut() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     postCommand(QByteArray("gu"), 0); // Step out-> gu (go up)
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     notifyInferiorRunRequested(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeNext() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     postCommand(QByteArray("p"), 0); // Step over -> p
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     notifyInferiorRunRequested(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeStepI() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     executeStep(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeNextI() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     executeNext(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::continueInferior() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     notifyInferiorRunRequested(); | 
					
						
							|  |  |  |     doContinueInferior(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::doContinueInferior() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     postCommand(QByteArray("g"), 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  | bool CdbEngine::canInterruptInferior() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-09-16 11:04:11 +02:00
										 |  |  |     return m_effectiveStartMode != AttachToRemoteServer && inferiorPid(); | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | void CdbEngine::interruptInferior() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << "CdbEngine::interruptInferior()" << stateName(state()); | 
					
						
							| 
									
										
										
										
											2012-02-02 09:30:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bool ok = false; | 
					
						
							|  |  |  |     if (!canInterruptInferior()) { | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |         showMessage(tr("Interrupting is not possible in remote sessions."), LogError); | 
					
						
							| 
									
										
										
										
											2012-02-02 09:30:22 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         ok = doInterruptInferior(NoSpecialStop); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // Restore running state if stop failed.
 | 
					
						
							|  |  |  |     if (!ok) { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorStopOk") | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |         notifyInferiorStopOk(); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunRequested") | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |         notifyInferiorRunRequested(); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunOk") | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |         notifyInferiorRunOk(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  | void CdbEngine::doInterruptInferiorCustomSpecialStop(const QVariant &v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_specialStopMode == NoSpecialStop) | 
					
						
							|  |  |  |         doInterruptInferior(CustomSpecialStop); | 
					
						
							|  |  |  |     m_customSpecialStopData.push_back(v); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-02 09:30:22 +01:00
										 |  |  | bool CdbEngine::doInterruptInferior(SpecialStopMode sm) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     const SpecialStopMode oldSpecialMode = m_specialStopMode; | 
					
						
							|  |  |  |     m_specialStopMode = sm; | 
					
						
							| 
									
										
										
										
											2012-02-02 09:30:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-28 10:02:34 +02:00
										 |  |  |     showMessage(QString::fromLatin1("Interrupting process %1...").arg(inferiorPid()), LogMisc); | 
					
						
							| 
									
										
										
										
											2012-02-02 09:30:22 +01:00
										 |  |  |     QString errorMessage; | 
					
						
							|  |  |  |     const bool ok = interruptProcess(inferiorPid(), CdbEngineType, &errorMessage); | 
					
						
							|  |  |  |     if (!ok) { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         m_specialStopMode = oldSpecialMode; | 
					
						
							| 
									
										
										
										
											2012-02-02 09:30:22 +01:00
										 |  |  |         showMessage(errorMessage, LogError); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-02-02 09:30:22 +01:00
										 |  |  |     return ok; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-23 10:16:11 +01:00
										 |  |  | void CdbEngine::executeRunToLine(const ContextData &data) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     // Add one-shot breakpoint
 | 
					
						
							| 
									
										
										
										
											2011-03-01 16:25:11 +01:00
										 |  |  |     BreakpointParameters bp; | 
					
						
							|  |  |  |     if (data.address) { | 
					
						
							|  |  |  |         bp.type =BreakpointByAddress; | 
					
						
							|  |  |  |         bp.address = data.address; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         bp.type =BreakpointByFileAndLine; | 
					
						
							|  |  |  |         bp.fileName = data.fileName; | 
					
						
							|  |  |  |         bp.lineNumber = data.lineNumber; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-05-20 21:15:30 +02:00
										 |  |  |     postCommand(cdbAddBreakpointCommand(bp, m_sourcePathMappings, BreakpointModelId(quint16(-1)), true), 0); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     continueInferior(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeRunToFunction(const QString &functionName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Add one-shot breakpoint
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     BreakpointParameters bp(BreakpointByFunction); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     bp.functionName = functionName; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-20 21:15:30 +02:00
										 |  |  |     postCommand(cdbAddBreakpointCommand(bp, m_sourcePathMappings, BreakpointModelId(quint16(-1)), true), 0); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     continueInferior(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::setRegisterValue(int regnr, const QString &value) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     const Registers registers = registerHandler()->registers(); | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |     QTC_ASSERT(regnr < registers.size(), return); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     // Value is decimal or 0x-hex-prefixed
 | 
					
						
							|  |  |  |     QByteArray cmd; | 
					
						
							|  |  |  |     ByteArrayInputStream str(cmd); | 
					
						
							|  |  |  |     str << "r " << registers.at(regnr).name << '=' << value; | 
					
						
							|  |  |  |     postCommand(cmd, 0); | 
					
						
							|  |  |  |     reloadRegisters(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-23 10:16:11 +01:00
										 |  |  | void CdbEngine::executeJumpToLine(const ContextData &data) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-01 16:25:11 +01:00
										 |  |  |     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); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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
 | 
					
						
							| 
									
										
										
										
											2011-12-21 14:02:52 +01:00
										 |  |  |     QByteArray answer = cmd->reply.front().trimmed(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     const int equalPos = answer.indexOf(" = "); | 
					
						
							|  |  |  |     if (equalPos == -1) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     answer.remove(0, equalPos + 3); | 
					
						
							| 
									
										
										
										
											2011-12-21 14:02:52 +01:00
										 |  |  |     const int apPos = answer.indexOf('`'); | 
					
						
							|  |  |  |     if (apPos != -1) | 
					
						
							|  |  |  |         answer.remove(apPos, 1); | 
					
						
							| 
									
										
										
										
											2011-03-01 16:25:11 +01:00
										 |  |  |     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)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-17 12:19:24 +02:00
										 |  |  | static inline bool isAsciiWord(const QString &s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     foreach (const QChar &c, s) { | 
					
						
							|  |  |  |         if (!c.isLetterOrNumber() || c.toAscii() == 0) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | void CdbEngine::assignValueInDebugger(const WatchData *w, const QString &expr, const QVariant &value) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     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); | 
					
						
							| 
									
										
										
										
											2011-05-17 12:19:24 +02:00
										 |  |  |     switch (value.type()) { | 
					
						
							|  |  |  |     case QVariant::String: { | 
					
						
							|  |  |  |         // Convert qstring to Utf16 data not considering endianness for Windows.
 | 
					
						
							|  |  |  |         const QString s = value.toString(); | 
					
						
							|  |  |  |         if (isAsciiWord(s)) { | 
					
						
							|  |  |  |             str << m_extensionCommandPrefixBA << "assign \"" << w->iname << '=' | 
					
						
							|  |  |  |                 << s.toLatin1() << '"'; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             const QByteArray utf16(reinterpret_cast<const char *>(s.utf16()), 2 * s.size()); | 
					
						
							|  |  |  |             str << m_extensionCommandPrefixBA << "assign -u " << w->iname << '=' | 
					
						
							|  |  |  |                 << utf16.toHex(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         str << m_extensionCommandPrefixBA << "assign " << w->iname << '=' | 
					
						
							|  |  |  |             << value.toString(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     postCommand(cmd, 0); | 
					
						
							| 
									
										
										
										
											2011-01-11 14:58:32 +01:00
										 |  |  |     // Update all locals in case we change a union or something pointed to
 | 
					
						
							|  |  |  |     // that affects other variables, too.
 | 
					
						
							|  |  |  |     updateLocals(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  | 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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | void CdbEngine::handleThreads(const CdbExtensionCommandPtr &reply) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("CdbEngine::handleThreads success=%d", reply->success); | 
					
						
							|  |  |  |     if (reply->success) { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         GdbMi data; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         data.fromString(reply->reply); | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |         parseThreads(data); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         // Continue sequence
 | 
					
						
							|  |  |  |         postCommandSequence(reply->commandSequence); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         showMessage(QString::fromLatin1(reply->errorMessage), LogError); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-10 09:36:15 +02:00
										 |  |  | void CdbEngine::executeDebuggerCommand(const QString &command, DebuggerLanguages languages) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-10 09:36:15 +02:00
										 |  |  |     if (languages & CppLanguage) | 
					
						
							|  |  |  |         postCommand(command.toLocal8Bit(), QuietCommand); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // 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)") | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |                 .arg(QString::fromLocal8Bit(cmd), QString::fromLatin1(stateName(state()))); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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)") | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |                 .arg(QString::fromLocal8Bit(cmd), QString::fromLatin1(stateName(state()))); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |     if (!flags & QuietCommand) | 
					
						
							|  |  |  |         showMessage(QString::fromLocal8Bit(fullCmd), LogInput); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     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; | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     const StackFrames &frames = stackHandler()->frames(); | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |     QTC_ASSERT(index < frames.size(), return); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |     const StackFrame frame = frames.at(index); | 
					
						
							|  |  |  |     if (debug || debugLocals) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         qDebug("activateFrame idx=%d '%s' %d", index, | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |                qPrintable(frame.file), frame.line); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     stackHandler()->setCurrentIndex(index); | 
					
						
							|  |  |  |     const bool showAssembler = !frames.at(index).isUsable(); | 
					
						
							|  |  |  |     if (showAssembler) { // Assembly code: Clean out model and force instruction mode.
 | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |         watchHandler()->removeAllData(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         QAction *assemblerAction = theAssemblerAction(); | 
					
						
							|  |  |  |         if (assemblerAction->isChecked()) { | 
					
						
							| 
									
										
										
										
											2010-12-16 19:06:33 +01:00
										 |  |  |             gotoLocation(frame); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             assemblerAction->trigger(); // Seems to trigger update
 | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-11 14:58:32 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         gotoLocation(frame); | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  |         updateLocals(true); | 
					
						
							| 
									
										
										
										
											2011-01-11 14:58:32 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  | void CdbEngine::updateLocals(bool forNewStackFrame) | 
					
						
							| 
									
										
										
										
											2011-01-11 14:58:32 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |     typedef QHash<QByteArray, int> WatcherHash; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 14:58:32 +01:00
										 |  |  |     const int frameIndex = stackHandler()->currentIndex(); | 
					
						
							|  |  |  |     if (frameIndex < 0) { | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |         watchHandler()->removeAllData(); | 
					
						
							| 
									
										
										
										
											2011-01-11 14:58:32 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const StackFrame frame = stackHandler()->currentFrame(); | 
					
						
							|  |  |  |     if (!frame.isUsable()) { | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |         watchHandler()->removeAllData(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-18 15:28:55 +01:00
										 |  |  |     /* 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 */ | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |     QByteArray arguments; | 
					
						
							|  |  |  |     ByteArrayInputStream str(arguments); | 
					
						
							| 
									
										
										
										
											2011-01-18 15:28:55 +01:00
										 |  |  |     str << "-D"; | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |     // Pre-expand
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     const QSet<QByteArray> expanded = watchHandler()->expandedINames(); | 
					
						
							|  |  |  |     if (!expanded.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |         str << blankSeparator << "-e "; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         int i = 0; | 
					
						
							|  |  |  |         foreach(const QByteArray &e, expanded) { | 
					
						
							|  |  |  |             if (i++) | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |                 str << ','; | 
					
						
							|  |  |  |             str << e; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-12-09 17:04:43 +01:00
										 |  |  |     addLocalsOptions(str); | 
					
						
							| 
									
										
										
										
											2011-04-12 11:57:57 +02:00
										 |  |  |     // Uninitialized variables if desired. Quote as safeguard against shadowed
 | 
					
						
							|  |  |  |     // variables in case of errors in uninitializedVariables().
 | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |     if (debuggerCore()->boolSetting(UseCodeModel)) { | 
					
						
							|  |  |  |         QStringList uninitializedVariables; | 
					
						
							|  |  |  |         getUninitializedVariables(debuggerCore()->cppCodeModelSnapshot(), | 
					
						
							|  |  |  |                                   frame.function, frame.file, frame.line, &uninitializedVariables); | 
					
						
							|  |  |  |         if (!uninitializedVariables.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2011-04-12 11:57:57 +02:00
										 |  |  |             str << blankSeparator << "-u \""; | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |             int i = 0; | 
					
						
							|  |  |  |             foreach(const QString &u, uninitializedVariables) { | 
					
						
							|  |  |  |                 if (i++) | 
					
						
							|  |  |  |                     str << ','; | 
					
						
							|  |  |  |                 str << localsPrefixC << u; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-04-12 11:57:57 +02:00
										 |  |  |             str << '"'; | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |     // 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() << '"'; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |     // Required arguments: frame
 | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |     const int flags = forNewStackFrame ? LocalsUpdateForNewFrame : 0; | 
					
						
							| 
									
										
										
										
											2011-01-11 14:58:32 +01:00
										 |  |  |     str << blankSeparator << frameIndex; | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |     postExtensionCommand("locals", arguments, 0, | 
					
						
							|  |  |  |                          &CdbEngine::handleLocals, 0, | 
					
						
							|  |  |  |                          QVariant(flags)); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::selectThread(int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (index < 0 || index == threadsHandler()->currentThread()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const int newThreadId = threadsHandler()->threads().at(index).id; | 
					
						
							|  |  |  |     threadsHandler()->setCurrentThread(index); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-06 19:27:49 +01:00
										 |  |  |     const QByteArray cmd = '~' + QByteArray::number(newThreadId) + " s"; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     postBuiltinCommand(cmd, 0, &CdbEngine::dummyHandler, CommandListStack); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  | // Default address range for showing disassembly.
 | 
					
						
							|  |  |  | enum { DisassemblerRange = 512 }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  | /* Called with a stack frame (address and function) or just a function
 | 
					
						
							|  |  |  |  * name from the context menu. When address and function are | 
					
						
							|  |  |  |  * passed, try to emulate gdb's behaviour to display the whole function. | 
					
						
							|  |  |  |  * CDB's 'u' (disassemble) command takes a symbol, | 
					
						
							|  |  |  |  * but displays only 10 lines per default. So, to ensure the agent's | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |  * address is in that range, resolve the function symbol, cache it and | 
					
						
							|  |  |  |  * request the disassembly for a range that contains the agent's address. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | void CdbEngine::fetchDisassembler(DisassemblerAgent *agent) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |     QTC_ASSERT(m_accessible, return); | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |     const QVariant cookie = qVariantFromValue<DisassemblerAgent*>(agent); | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |     const Location location = agent->location(); | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug() << "CdbEngine::fetchDisassembler 0x" | 
					
						
							|  |  |  |                  << QString::number(location.address(), 16) | 
					
						
							|  |  |  |                  << location.from() << '!' << location.functionName(); | 
					
						
							|  |  |  |     if (!location.functionName().isEmpty()) { | 
					
						
							|  |  |  |         // Resolve function (from stack frame with function and address
 | 
					
						
							|  |  |  |         // or just function from editor).
 | 
					
						
							|  |  |  |         postResolveSymbol(location.from(), location.functionName(), cookie); | 
					
						
							|  |  |  |     } else if (location.address()) { | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |         // No function, display a default range.
 | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |         postDisassemblerCommand(location.address(), cookie); | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |         QTC_ASSERT(false, return); | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::postDisassemblerCommand(quint64 address, const QVariant &cookie) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     postDisassemblerCommand(address - DisassemblerRange / 2, | 
					
						
							|  |  |  |                             address + DisassemblerRange / 2, cookie); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::postDisassemblerCommand(quint64 address, quint64 endAddress, | 
					
						
							|  |  |  |                                         const QVariant &cookie) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     QByteArray cmd; | 
					
						
							|  |  |  |     ByteArrayInputStream str(cmd); | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |     str <<  "u " << hex <<hexPrefixOn << address << ' ' << endAddress; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     postBuiltinCommand(cmd, 0, &CdbEngine::handleDisassembler, 0, cookie); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  | void CdbEngine::postResolveSymbol(const QString &module, const QString &function, | 
					
						
							|  |  |  |                                   const QVariant &cookie) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |     QString symbol = module.isEmpty() ? QString(QLatin1Char('*')) : module; | 
					
						
							|  |  |  |     symbol += QLatin1Char('!'); | 
					
						
							|  |  |  |     symbol += function; | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |     const QList<quint64> addresses = m_symbolAddressCache.values(symbol); | 
					
						
							|  |  |  |     if (addresses.isEmpty()) { | 
					
						
							|  |  |  |         QVariantList cookieList; | 
					
						
							|  |  |  |         cookieList << QVariant(symbol) << cookie; | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |         showMessage(QLatin1String("Resolving symbol: ") + symbol + QLatin1String("..."), LogMisc); | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |         postBuiltinCommand(QByteArray("x ") + symbol.toLatin1(), 0, | 
					
						
							|  |  |  |                            &CdbEngine::handleResolveSymbol, 0, | 
					
						
							|  |  |  |                            QVariant(cookieList)); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         showMessage(QString::fromLatin1("Using cached addresses for %1."). | 
					
						
							|  |  |  |                     arg(symbol), LogMisc); | 
					
						
							|  |  |  |         handleResolveSymbol(addresses, cookie); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parse address from 'x' response.
 | 
					
						
							|  |  |  | // "00000001`3f7ebe80 module!foo (void)"
 | 
					
						
							|  |  |  | static inline quint64 resolvedAddress(const QByteArray &line) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const int blankPos = line.indexOf(' '); | 
					
						
							|  |  |  |     if (blankPos >= 0) { | 
					
						
							|  |  |  |         QByteArray addressBA = line.left(blankPos); | 
					
						
							|  |  |  |         if (addressBA.size() > 9 && addressBA.at(8) == '`') | 
					
						
							|  |  |  |             addressBA.remove(8, 1); | 
					
						
							|  |  |  |         bool ok; | 
					
						
							|  |  |  |         const quint64 address = addressBA.toULongLong(&ok, 16); | 
					
						
							|  |  |  |         if (ok) | 
					
						
							|  |  |  |             return address; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::handleResolveSymbol(const CdbBuiltinCommandPtr &command) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QTC_ASSERT(command->cookie.type() == QVariant::List, return; ); | 
					
						
							|  |  |  |     const QVariantList cookieList = command->cookie.toList(); | 
					
						
							|  |  |  |     const QString symbol = cookieList.front().toString(); | 
					
						
							|  |  |  |     // Insert all matches of (potentially) ambiguous symbols
 | 
					
						
							|  |  |  |     if (const int size = command->reply.size()) { | 
					
						
							|  |  |  |         for (int i = 0; i < size; i++) { | 
					
						
							|  |  |  |             if (const quint64 address = resolvedAddress(command->reply.at(i))) { | 
					
						
							|  |  |  |                 m_symbolAddressCache.insert(symbol, address); | 
					
						
							|  |  |  |                 showMessage(QString::fromLatin1("Obtained 0x%1 for %2 (#%3)"). | 
					
						
							|  |  |  |                             arg(address, 0, 16).arg(symbol).arg(i + 1), LogMisc); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         showMessage(QLatin1String("Symbol resolution failed: ") | 
					
						
							|  |  |  |                     + QString::fromLatin1(command->joinedReply()), | 
					
						
							|  |  |  |                     LogError); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     handleResolveSymbol(m_symbolAddressCache.values(symbol), cookieList.back()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Find the function address matching needle in a list of function
 | 
					
						
							|  |  |  | // addresses obtained from the 'x' command. Check for the
 | 
					
						
							|  |  |  | // mimimum POSITIVE offset (needle >= function address.)
 | 
					
						
							|  |  |  | static inline quint64 findClosestFunctionAddress(const QList<quint64> &addresses, | 
					
						
							|  |  |  |                                                  quint64 needle) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const int size = addresses.size(); | 
					
						
							|  |  |  |     if (!size) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (size == 1) | 
					
						
							|  |  |  |        return addresses.front(); | 
					
						
							|  |  |  |     int closestIndex = 0; | 
					
						
							|  |  |  |     quint64 closestOffset = 0xFFFFFFFF; | 
					
						
							|  |  |  |     for (int i = 0; i < size; i++) { | 
					
						
							|  |  |  |         if (addresses.at(i) <= needle) { | 
					
						
							|  |  |  |             const quint64 offset = needle - addresses.at(i); | 
					
						
							| 
									
										
										
										
											2011-07-11 16:30:58 +02:00
										 |  |  |             if (offset < closestOffset) { | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |                 closestOffset = offset; | 
					
						
							|  |  |  |                 closestIndex = i; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return addresses.at(closestIndex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  | static inline QString msgAmbiguousFunction(const QString &functionName, | 
					
						
							|  |  |  |                                            quint64 address, | 
					
						
							|  |  |  |                                            const QList<quint64> &addresses) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QString result; | 
					
						
							|  |  |  |     QTextStream str(&result); | 
					
						
							|  |  |  |     str.setIntegerBase(16); | 
					
						
							|  |  |  |     str.setNumberFlags(str.numberFlags() | QTextStream::ShowBase); | 
					
						
							|  |  |  |     str << "Several overloads of function '" << functionName | 
					
						
							|  |  |  |         << "()' were found ("; | 
					
						
							|  |  |  |     for (int i = 0; i < addresses.size(); ++i) { | 
					
						
							|  |  |  |         if (i) | 
					
						
							|  |  |  |             str << ", "; | 
					
						
							|  |  |  |         str << addresses.at(i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     str << "), using " << address << '.'; | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  | void CdbEngine::handleResolveSymbol(const QList<quint64> &addresses, const QVariant &cookie) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Disassembly mode: Determine suitable range containing the
 | 
					
						
							|  |  |  |     // agent's address within the function to display.
 | 
					
						
							|  |  |  |     if (qVariantCanConvert<DisassemblerAgent*>(cookie)) { | 
					
						
							|  |  |  |         DisassemblerAgent *agent = cookie.value<DisassemblerAgent *>(); | 
					
						
							|  |  |  |         const quint64 agentAddress = agent->address(); | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |         quint64 functionAddress = 0; | 
					
						
							|  |  |  |         quint64 endAddress = 0; | 
					
						
							|  |  |  |         if (agentAddress) { | 
					
						
							|  |  |  |             // We have an address from the agent, find closest.
 | 
					
						
							|  |  |  |             if (const quint64 closest = findClosestFunctionAddress(addresses, agentAddress)) { | 
					
						
							|  |  |  |                 if (closest <= agentAddress) { | 
					
						
							|  |  |  |                     functionAddress = closest; | 
					
						
							|  |  |  |                     endAddress = agentAddress + DisassemblerRange / 2; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             // No agent address, disassembly was started with a function name only.
 | 
					
						
							|  |  |  |             if (!addresses.isEmpty()) { | 
					
						
							|  |  |  |                 functionAddress = addresses.first(); | 
					
						
							|  |  |  |                 endAddress = functionAddress + DisassemblerRange / 2; | 
					
						
							|  |  |  |                 if (addresses.size() > 1) | 
					
						
							|  |  |  |                     showMessage(msgAmbiguousFunction(agent->location().functionName(), functionAddress, addresses), LogMisc); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Disassemble a function, else use default range around agent address
 | 
					
						
							|  |  |  |         if (functionAddress) { | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |             if (const quint64 remainder = endAddress % 8) | 
					
						
							|  |  |  |                 endAddress += 8 - remainder; | 
					
						
							|  |  |  |             postDisassemblerCommand(functionAddress, endAddress, cookie); | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |         } else if (agentAddress) { | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |             postDisassemblerCommand(agentAddress, cookie); | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             QTC_ASSERT(false, return); | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2012-02-01 17:44:07 +01:00
										 |  |  |     } // DisassemblerAgent
 | 
					
						
							| 
									
										
										
										
											2011-07-04 09:29:28 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | // Parse: "00000000`77606060 cc              int     3"
 | 
					
						
							|  |  |  | void CdbEngine::handleDisassembler(const CdbBuiltinCommandPtr &command) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |     QTC_ASSERT(qVariantCanConvert<DisassemblerAgent*>(command->cookie), return); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     DisassemblerAgent *agent = qvariant_cast<DisassemblerAgent*>(command->cookie); | 
					
						
							| 
									
										
										
										
											2011-04-13 13:52:03 +02:00
										 |  |  |     agent->setContents(parseCdbDisassembler(command->reply)); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | void CdbEngine::fetchMemory(MemoryAgent *agent, QObject *editor, quint64 addr, quint64 length) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("CdbEngine::fetchMemory %llu bytes from 0x%llx", length, addr); | 
					
						
							| 
									
										
										
										
											2011-04-19 13:57:25 +02:00
										 |  |  |     const MemoryViewCookie cookie(agent, editor, addr, length); | 
					
						
							|  |  |  |     if (m_accessible) { | 
					
						
							|  |  |  |         postFetchMemory(cookie); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         doInterruptInferiorCustomSpecialStop(qVariantFromValue(cookie)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-19 13:57:25 +02:00
										 |  |  | void CdbEngine::postFetchMemory(const MemoryViewCookie &cookie) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     QByteArray args; | 
					
						
							|  |  |  |     ByteArrayInputStream str(args); | 
					
						
							| 
									
										
										
										
											2011-04-19 13:57:25 +02:00
										 |  |  |     str << cookie.address << ' ' << cookie.length; | 
					
						
							|  |  |  |     postExtensionCommand("memory", args, 0, &CdbEngine::handleMemory, 0, | 
					
						
							|  |  |  |                          qVariantFromValue(cookie)); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  | void CdbEngine::changeMemory(Internal::MemoryAgent *, QObject *, quint64 addr, const QByteArray &data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |     QTC_ASSERT(!data.isEmpty(), return); | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  |     if (!m_accessible) { | 
					
						
							|  |  |  |         const MemoryChangeCookie cookie(addr, data); | 
					
						
							|  |  |  |         doInterruptInferiorCustomSpecialStop(qVariantFromValue(cookie)); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         postCommand(cdbWriteMemoryCommand(addr, data), 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | void CdbEngine::handleMemory(const CdbExtensionCommandPtr &command) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |     QTC_ASSERT(qVariantCanConvert<MemoryViewCookie>(command->cookie), return); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     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 { | 
					
						
							| 
									
										
										
										
											2011-04-19 13:57:25 +02:00
										 |  |  |         showMessage(QString::fromLocal8Bit(command->errorMessage), LogWarning); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  |     // Fails for core dumps.
 | 
					
						
							|  |  |  |     if (reply->success) | 
					
						
							| 
									
										
										
										
											2011-04-28 10:02:34 +02:00
										 |  |  |         notifyInferiorPid(reply->reply.toULongLong()); | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  |     if (reply->success || startParameters().startMode == AttachCore) { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorSetupOk") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         notifyInferiorSetupOk(); | 
					
						
							|  |  |  |     }  else { | 
					
						
							|  |  |  |         showMessage(QString::fromLatin1("Failed to determine inferior pid: %1"). | 
					
						
							|  |  |  |                     arg(QLatin1String(reply->errorMessage)), LogError); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorSetupFailed") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         notifyInferiorSetupFailed(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-14 12:53:17 +02:00
										 |  |  | // Parse CDB gdbmi register syntax.
 | 
					
						
							|  |  |  | static Register parseRegister(const GdbMi &gdbmiReg) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     Register reg; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     reg.name = gdbmiReg.findChild("name").data(); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     const GdbMi description = gdbmiReg.findChild("description"); | 
					
						
							|  |  |  |     if (description.type() != GdbMi::Invalid) { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         reg.name += " ("; | 
					
						
							|  |  |  |         reg.name += description.data(); | 
					
						
							|  |  |  |         reg.name += ')'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-09-14 12:53:17 +02:00
										 |  |  |     reg.value = gdbmiReg.findChild("value").data(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     return reg; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::handleModules(const CdbExtensionCommandPtr &reply) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (reply->success) { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         GdbMi value; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         value.fromString(reply->reply); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         if (value.type() == GdbMi::List) { | 
					
						
							|  |  |  |             Modules modules; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             modules.reserve(value.childCount()); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |             foreach (const GdbMi &gdbmiModule, value.children()) { | 
					
						
							|  |  |  |                 Module module; | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |                 module.moduleName = QString::fromLatin1(gdbmiModule.findChild("name").data()); | 
					
						
							|  |  |  |                 module.modulePath = QString::fromLatin1(gdbmiModule.findChild("image").data()); | 
					
						
							| 
									
										
										
										
											2010-12-21 13:34:59 +01:00
										 |  |  |                 module.startAddress = gdbmiModule.findChild("start").data().toULongLong(0, 0); | 
					
						
							|  |  |  |                 module.endAddress = gdbmiModule.findChild("end").data().toULongLong(0, 0); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |                 if (gdbmiModule.findChild("deferred").type() == GdbMi::Invalid) | 
					
						
							|  |  |  |                     module.symbolsRead = Module::ReadOk; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                 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) { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         GdbMi value; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         value.fromString(reply->reply); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         if (value.type() == GdbMi::List) { | 
					
						
							|  |  |  |             Registers registers; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             registers.reserve(value.childCount()); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |             foreach (const GdbMi &gdbmiReg, value.children()) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                 registers.push_back(parseRegister(gdbmiReg)); | 
					
						
							| 
									
										
										
										
											2011-04-06 14:07:09 +02:00
										 |  |  |             registerHandler()->setAndMarkRegisters(registers); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         } 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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |     const int flags = reply->cookie.toInt(); | 
					
						
							|  |  |  |     if (!(flags & PartialLocalsUpdate)) | 
					
						
							|  |  |  |         watchHandler()->removeAllData(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     if (reply->success) { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         QList<WatchData> watchData; | 
					
						
							| 
									
										
										
										
											2010-12-15 14:07:48 +01:00
										 |  |  |         GdbMi root; | 
					
						
							|  |  |  |         root.fromString(reply->reply); | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |         QTC_ASSERT(root.isList(), return); | 
					
						
							| 
									
										
										
										
											2010-12-15 14:07:48 +01:00
										 |  |  |         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); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |         watchHandler()->insertData(watchData); | 
					
						
							| 
									
										
										
										
											2010-12-15 14:07:48 +01:00
										 |  |  |         if (debugLocals) { | 
					
						
							|  |  |  |             QDebug nsp = qDebug().nospace(); | 
					
						
							|  |  |  |             nsp << "Obtained " << watchData.size() << " items:\n"; | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |             foreach (const WatchData &wd, watchData) | 
					
						
							| 
									
										
										
										
											2010-12-15 14:07:48 +01:00
										 |  |  |                 nsp << wd.toString() <<'\n'; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2012-05-18 02:28:41 +02:00
										 |  |  |         if (flags & LocalsUpdateForNewFrame) | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  |             emit stackFrameCompleted(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-04-18 11:51:44 +02:00
										 |  |  |         showMessage(QString::fromLatin1(reply->errorMessage), LogWarning); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  | /* Examine how to react to a stop. */ | 
					
						
							|  |  |  | enum StopActionFlags | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Report options
 | 
					
						
							|  |  |  |     StopReportLog = 0x1, | 
					
						
							|  |  |  |     StopReportStatusMessage = 0x2, | 
					
						
							| 
									
										
										
										
											2011-01-25 11:52:50 +01:00
										 |  |  |     StopReportParseError = 0x4, | 
					
						
							|  |  |  |     StopShowExceptionMessageBox = 0x8, | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     // Notify stop or just continue
 | 
					
						
							| 
									
										
										
										
											2011-01-25 11:52:50 +01:00
										 |  |  |     StopNotifyStop = 0x10, | 
					
						
							|  |  |  |     StopIgnoreContinue = 0x20, | 
					
						
							| 
									
										
										
										
											2011-01-18 15:28:55 +01:00
										 |  |  |     // Hit on break in artificial stop thread (created by DebugBreak()).
 | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |     StopInArtificialThread = 0x40, | 
					
						
							|  |  |  |     StopShutdownInProgress = 0x80 // Shutdown in progress
 | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  | static inline QString msgTracePointTriggered(BreakpointModelId id, const int number, | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |                                              const QString &threadId) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return CdbEngine::tr("Trace point %1 (%2) in thread %3 triggered.") | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |             .arg(id.toString()).arg(number).arg(threadId); | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  | static inline QString msgCheckingConditionalBreakPoint(BreakpointModelId id, const int number, | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |                                                        const QByteArray &condition, | 
					
						
							|  |  |  |                                                        const QString &threadId) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return CdbEngine::tr("Conditional breakpoint %1 (%2) in thread %3 triggered, examining expression '%4'.") | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |             .arg(id.toString()).arg(number).arg(threadId, QString::fromLatin1(condition)); | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  | unsigned CdbEngine::examineStopReason(const GdbMi &stopReason, | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |                                       QString *message, | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |                                       QString *exceptionBoxMessage, | 
					
						
							|  |  |  |                                       bool conditionalBreakPointTriggered) | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     // Report stop reason (GDBMI)
 | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |     unsigned rc  = 0; | 
					
						
							|  |  |  |     if (targetState() == DebuggerFinished) | 
					
						
							|  |  |  |         rc |= StopShutdownInProgress; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     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."); | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |         rc |= StopReportParseError|StopNotifyStop; | 
					
						
							|  |  |  |         return rc; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |     // 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)."). | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |                     arg(QString::fromLatin1(reason)); | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |         rc |= StopReportLog; | 
					
						
							|  |  |  |         return rc; | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     const int threadId = stopReason.findChild("threadId").data().toInt(); | 
					
						
							|  |  |  |     if (reason == "breakpoint") { | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         // Note: Internal breakpoints (run to line) are reported with id=0.
 | 
					
						
							| 
									
										
										
										
											2011-03-16 11:55:24 +01:00
										 |  |  |         // Step out creates temporary breakpoints with id 10000.
 | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |         BreakpointModelId id; | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         int number = 0; | 
					
						
							|  |  |  |         const GdbMi breakpointIdG = stopReason.findChild("breakpointId"); | 
					
						
							|  |  |  |         if (breakpointIdG.isValid()) { | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |             id = BreakpointModelId(breakpointIdG.data().toInt()); | 
					
						
							| 
									
										
										
										
											2011-03-16 11:55:24 +01:00
										 |  |  |             if (id && breakHandler()->engineBreakpointIds(this).contains(id)) { | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |                 const BreakpointResponse parameters =  breakHandler()->response(id); | 
					
						
							| 
									
										
										
										
											2011-06-27 10:37:57 +02:00
										 |  |  |                 if (!parameters.message.isEmpty()) { | 
					
						
							|  |  |  |                     showMessage(parameters.message + QLatin1Char('\n'), AppOutput); | 
					
						
							|  |  |  |                     showMessage(parameters.message, LogMisc); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |                 // Trace point? Just report.
 | 
					
						
							| 
									
										
										
										
											2011-06-21 16:45:23 +02:00
										 |  |  |                 number = parameters.id.majorPart(); | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |                 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; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2011-03-16 11:55:24 +01:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |                 id = BreakpointModelId(); | 
					
						
							| 
									
										
										
										
											2011-03-16 11:55:24 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-05-09 08:35:58 +02:00
										 |  |  |         QString tid = QString::number(threadId); | 
					
						
							|  |  |  |         if (id && breakHandler()->type(id) == WatchpointAtAddress) { | 
					
						
							|  |  |  |             *message = msgWatchpointByAddressTriggered(id, number, breakHandler()->address(id), tid); | 
					
						
							|  |  |  |         } else if (id && breakHandler()->type(id) == WatchpointAtExpression) { | 
					
						
							|  |  |  |             *message = msgWatchpointByExpressionTriggered(id, number, breakHandler()->expression(id), tid); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2011-05-09 08:35:58 +02:00
										 |  |  |             *message = msgBreakpointTriggered(id, number, tid); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |         rc |= StopReportStatusMessage|StopNotifyStop; | 
					
						
							|  |  |  |         return rc; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (reason == "exception") { | 
					
						
							|  |  |  |         WinException exception; | 
					
						
							|  |  |  |         exception.fromGdbMI(stopReason); | 
					
						
							| 
									
										
										
										
											2011-01-12 14:21:44 +01:00
										 |  |  |         QString description = exception.toString(); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  | #ifdef Q_OS_WIN
 | 
					
						
							| 
									
										
										
										
											2011-03-02 13:42:45 +01:00
										 |  |  |         // It is possible to hit on a startup trap or WOW86 exception while stepping (if something
 | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         // pulls DLLs. Avoid showing a 'stopped' Message box.
 | 
					
						
							| 
									
										
										
										
											2011-03-02 13:42:45 +01:00
										 |  |  |         if (exception.exceptionCode == winExceptionStartupCompleteTrap | 
					
						
							|  |  |  |             || exception.exceptionCode == winExceptionWX86Breakpoint) | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |             return StopNotifyStop; | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         if (exception.exceptionCode == winExceptionCtrlPressed) { | 
					
						
							|  |  |  |             // Detect interruption by pressing Ctrl in a console and force a switch to thread 0.
 | 
					
						
							|  |  |  |             *message = msgInterrupted(); | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |             rc |= StopReportStatusMessage|StopNotifyStop|StopInArtificialThread; | 
					
						
							|  |  |  |             return rc; | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         if (isDebuggerWinException(exception.exceptionCode)) { | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |             rc |= StopReportStatusMessage|StopNotifyStop; | 
					
						
							| 
									
										
										
										
											2011-01-18 15:28:55 +01:00
										 |  |  |             // Detect interruption by DebugBreak() and force a switch to thread 0.
 | 
					
						
							|  |  |  |             if (exception.function == "ntdll!DbgBreakPoint") | 
					
						
							|  |  |  |                 rc |= StopInArtificialThread; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |             *message = msgInterrupted(); | 
					
						
							| 
									
										
										
										
											2011-01-18 15:28:55 +01:00
										 |  |  |             return rc; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |         *exceptionBoxMessage = msgStoppedByException(description, QString::number(threadId)); | 
					
						
							|  |  |  |         *message = description; | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |         rc |= StopShowExceptionMessageBox|StopReportStatusMessage|StopNotifyStop; | 
					
						
							|  |  |  |         return rc; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     *message = msgStopped(QLatin1String(reason)); | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |     rc |= StopReportStatusMessage|StopNotifyStop; | 
					
						
							|  |  |  |     return rc; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::handleSessionIdle(const QByteArray &messageBA) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!m_hasDebuggee) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("CdbEngine::handleSessionIdle %dms '%s' in state '%s', special mode %d", | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |                elapsedLogTime(), messageBA.constData(), | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                stateName(state()), m_specialStopMode); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Switch source level debugging
 | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |     syncOperateByInstruction(m_operateByInstructionPending); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     // Engine-special stop reasons: Breakpoints and setup
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     const SpecialStopMode specialStopMode =  m_specialStopMode; | 
					
						
							| 
									
										
										
										
											2011-02-16 14:13:07 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     m_specialStopMode = NoSpecialStop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch(specialStopMode) { | 
					
						
							|  |  |  |     case SpecialStopSynchronizeBreakpoints: | 
					
						
							|  |  |  |         if (debug) | 
					
						
							|  |  |  |             qDebug("attemptBreakpointSynchronization in special stop"); | 
					
						
							|  |  |  |         attemptBreakpointSynchronization(); | 
					
						
							|  |  |  |         doContinueInferior(); | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |     case SpecialStopGetWidgetAt: | 
					
						
							|  |  |  |         postWidgetAtCommand(); | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  |     case CustomSpecialStop: | 
					
						
							|  |  |  |         foreach (const QVariant &data, m_customSpecialStopData) | 
					
						
							|  |  |  |             handleCustomSpecialStop(data); | 
					
						
							|  |  |  |         m_customSpecialStopData.clear(); | 
					
						
							|  |  |  |         doContinueInferior(); | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     case NoSpecialStop: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (state() == EngineSetupRequested) { // Temporary stop at beginning
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupOk") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         notifyEngineSetupOk(); | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  |         // Store stop reason to be handled in runEngine().
 | 
					
						
							|  |  |  |         if (startParameters().startMode == AttachCore) { | 
					
						
							|  |  |  |             m_coreStopReason.reset(new GdbMi); | 
					
						
							|  |  |  |             m_coreStopReason->fromString(messageBA); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |     GdbMi stopReason; | 
					
						
							|  |  |  |     stopReason.fromString(messageBA); | 
					
						
							|  |  |  |     processStop(stopReason, false); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  | void CdbEngine::processStop(const GdbMi &stopReason, bool conditionalBreakPointTriggered) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     // Further examine stop and report to user
 | 
					
						
							|  |  |  |     QString message; | 
					
						
							|  |  |  |     QString exceptionBoxMessage; | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |     int forcedThreadId = -1; | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |     const unsigned stopFlags = examineStopReason(stopReason, &message, &exceptionBoxMessage, | 
					
						
							|  |  |  |                                                  conditionalBreakPointTriggered); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     // Do the non-blocking log reporting
 | 
					
						
							|  |  |  |     if (stopFlags & StopReportLog) | 
					
						
							|  |  |  |         showMessage(message, LogMisc); | 
					
						
							|  |  |  |     if (stopFlags & StopReportStatusMessage) | 
					
						
							|  |  |  |         showStatusMessage(message); | 
					
						
							|  |  |  |     if (stopFlags & StopReportParseError) | 
					
						
							|  |  |  |         showMessage(message, LogError); | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |     // Ignore things like WOW64, report tracepoints.
 | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     if (stopFlags & StopIgnoreContinue) { | 
					
						
							|  |  |  |         postCommand("g", 0); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     // Notify about state and send off command sequence to get stack, etc.
 | 
					
						
							|  |  |  |     if (stopFlags & StopNotifyStop) { | 
					
						
							| 
									
										
										
										
											2011-08-05 17:42:32 +02:00
										 |  |  |         if (startParameters().startMode != AttachCore) { | 
					
						
							|  |  |  |             if (state() == InferiorStopRequested) { | 
					
						
							|  |  |  |                 STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorStopOk") | 
					
						
							|  |  |  |                         notifyInferiorStopOk(); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorSpontaneousStop") | 
					
						
							|  |  |  |                         notifyInferiorSpontaneousStop(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |         // Prevent further commands from being sent if shutdown is in progress
 | 
					
						
							|  |  |  |         if (stopFlags & StopShutdownInProgress) { | 
					
						
							|  |  |  |             showMessage(QString::fromLatin1("Shutdown request detected...")); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |         const bool sourceStepInto = m_sourceStepInto; | 
					
						
							|  |  |  |         m_sourceStepInto = false; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         // Start sequence to get all relevant data.
 | 
					
						
							| 
									
										
										
										
											2011-01-18 15:28:55 +01:00
										 |  |  |         if (stopFlags & StopInArtificialThread) { | 
					
						
							|  |  |  |             showMessage(tr("Switching to main thread..."), LogMisc); | 
					
						
							|  |  |  |             postCommand("~0 s", 0); | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |             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 { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |                 showMessage(QString::fromLatin1(stopReason.findChild("stackerror").data()), LogError); | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-01-18 15:28:55 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |         const GdbMi threads = stopReason.findChild("threads"); | 
					
						
							|  |  |  |         if (threads.isValid()) { | 
					
						
							|  |  |  |             parseThreads(threads, forcedThreadId); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |             showMessage(QString::fromLatin1(stopReason.findChild("threaderror").data()), LogError); | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         // Fire off remaining commands asynchronously
 | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         if (!m_pendingBreakpointMap.isEmpty()) | 
					
						
							|  |  |  |             postCommandSequence(CommandListBreakPoints); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         if (debuggerCore()->isDockVisible(QLatin1String(Constants::DOCKWIDGET_REGISTER))) | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |             postCommandSequence(CommandListRegisters); | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         if (debuggerCore()->isDockVisible(QLatin1String(Constants::DOCKWIDGET_MODULES))) | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |             postCommandSequence(CommandListModules); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |     // 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); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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: | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineRunAndInferiorRunOk") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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: | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorRunOk") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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') { | 
					
						
							| 
									
										
										
										
											2010-11-25 10:01:40 +01:00
										 |  |  |         if (token == -1) { // Default token, user typed in extension command
 | 
					
						
							|  |  |  |             showMessage(QString::fromLatin1(message), LogMisc); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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") { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |         showStatusMessage(QString::fromLatin1(message),  5000); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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; | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         GdbMi gdbmi; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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.
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |         if (exception.exceptionCode == winExceptionCppException) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             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); | 
					
						
							| 
									
										
										
										
											2011-01-04 12:40:52 +01:00
										 |  |  |     // An extension notification (potentially consisting of several chunks)
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     if (line.startsWith(m_creatorExtPrefix)) { | 
					
						
							| 
									
										
										
										
											2011-01-04 12:40:52 +01:00
										 |  |  |         // "<qtcreatorcdbext>|type_char|token|remainingChunks|serviceName|message"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         const char type = line.at(m_creatorExtPrefix.size()); | 
					
						
							|  |  |  |         // integer token
 | 
					
						
							|  |  |  |         const int tokenPos = m_creatorExtPrefix.size() + 2; | 
					
						
							|  |  |  |         const int tokenEndPos = line.indexOf('|', tokenPos); | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |         QTC_ASSERT(tokenEndPos != -1, return); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         const int token = line.mid(tokenPos, tokenEndPos - tokenPos).toInt(); | 
					
						
							| 
									
										
										
										
											2011-01-04 12:40:52 +01:00
										 |  |  |         // remainingChunks
 | 
					
						
							|  |  |  |         const int remainingChunksPos = tokenEndPos + 1; | 
					
						
							|  |  |  |         const int remainingChunksEndPos = line.indexOf('|', remainingChunksPos); | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |         QTC_ASSERT(remainingChunksEndPos != -1, return); | 
					
						
							| 
									
										
										
										
											2011-01-04 12:40:52 +01:00
										 |  |  |         const int remainingChunks = line.mid(remainingChunksPos, remainingChunksEndPos - remainingChunksPos).toInt(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         // const char 'serviceName'
 | 
					
						
							| 
									
										
										
										
											2011-01-04 12:40:52 +01:00
										 |  |  |         const int whatPos = remainingChunksEndPos + 1; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         const int whatEndPos = line.indexOf('|', whatPos); | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |         QTC_ASSERT(whatEndPos != -1, return); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         const QByteArray what = line.mid(whatPos, whatEndPos - whatPos); | 
					
						
							| 
									
										
										
										
											2011-01-04 12:40:52 +01:00
										 |  |  |         // 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(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         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() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |     if (m_ignoreCdbOutput) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     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'
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | static QByteArray multiBreakpointCommand(const char *cmdC, const Breakpoints &bps) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     QByteArray cmd(cmdC); | 
					
						
							|  |  |  |     ByteArrayInputStream str(cmd); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     foreach(const BreakpointData *bp, bps) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         str << ' ' << bp->bpNumber; | 
					
						
							|  |  |  |     return cmd; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-26 13:06:03 +01:00
										 |  |  | bool CdbEngine::stateAcceptsBreakpointChanges() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (state()) { | 
					
						
							|  |  |  |     case InferiorRunOk: | 
					
						
							|  |  |  |     case InferiorStopOk: | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  | bool CdbEngine::acceptsBreakpoint(BreakpointModelId id) const | 
					
						
							| 
									
										
										
										
											2010-11-26 13:06:03 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-07 11:02:07 +01:00
										 |  |  |     const BreakpointParameters &data = breakHandler()->breakpointData(id); | 
					
						
							| 
									
										
										
										
											2012-01-24 16:41:30 +01:00
										 |  |  |     if (!data.isCppBreakpoint()) | 
					
						
							| 
									
										
										
										
											2011-03-07 11:02:07 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     switch (data.type) { | 
					
						
							|  |  |  |     case UnknownType: | 
					
						
							|  |  |  |     case BreakpointAtFork: | 
					
						
							| 
									
										
										
										
											2011-05-09 09:49:42 +02:00
										 |  |  |     case WatchpointAtExpression: | 
					
						
							| 
									
										
										
										
											2011-03-07 11:02:07 +01:00
										 |  |  |     case BreakpointAtSysCall: | 
					
						
							| 
									
										
										
										
											2012-04-16 12:00:23 +02:00
										 |  |  |     case BreakpointOnQmlSignalEmit: | 
					
						
							| 
									
										
										
										
											2011-10-10 15:25:18 +02:00
										 |  |  |     case BreakpointAtJavaScriptThrow: | 
					
						
							| 
									
										
										
										
											2011-03-07 11:02:07 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2011-05-09 08:35:58 +02:00
										 |  |  |     case WatchpointAtAddress: | 
					
						
							| 
									
										
										
										
											2011-03-07 11:02:07 +01:00
										 |  |  |     case BreakpointByFileAndLine: | 
					
						
							|  |  |  |     case BreakpointByFunction: | 
					
						
							|  |  |  |     case BreakpointByAddress: | 
					
						
							|  |  |  |     case BreakpointAtThrow: | 
					
						
							|  |  |  |     case BreakpointAtCatch: | 
					
						
							|  |  |  |     case BreakpointAtMain: | 
					
						
							|  |  |  |     case BreakpointAtExec: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2010-11-26 13:06:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-09 17:02:46 +02:00
										 |  |  | // Context for fixing file/line-type breakpoints, for delayed creation.
 | 
					
						
							|  |  |  | class BreakpointCorrectionContext | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit BreakpointCorrectionContext(const CPlusPlus::Snapshot &s, | 
					
						
							|  |  |  |                                          const CPlusPlus::CppModelManagerInterface::WorkingCopy &workingCopy) : | 
					
						
							|  |  |  |         m_snapshot(s), m_workingCopy(workingCopy) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned fixLineNumber(const QString &fileName, unsigned lineNumber) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     const CPlusPlus::Snapshot m_snapshot; | 
					
						
							|  |  |  |     CPlusPlus::CppModelManagerInterface::WorkingCopy m_workingCopy; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static CPlusPlus::Document::Ptr getParsedDocument(const QString &fileName, | 
					
						
							|  |  |  |                                                   const CPlusPlus::CppModelManagerInterface::WorkingCopy &workingCopy, | 
					
						
							|  |  |  |                                                   const CPlusPlus::Snapshot &snapshot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QString src; | 
					
						
							|  |  |  |     if (workingCopy.contains(fileName)) { | 
					
						
							|  |  |  |         src = workingCopy.source(fileName); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         Utils::FileReader reader; | 
					
						
							|  |  |  |         if (reader.fetch(fileName)) // ### FIXME error reporting
 | 
					
						
							|  |  |  |             src = QString::fromLocal8Bit(reader.data()); // ### FIXME encoding
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QByteArray source = snapshot.preprocessedCode(src, fileName); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CPlusPlus::Document::Ptr doc = snapshot.documentFromSource(source, fileName); | 
					
						
							|  |  |  |     doc->parse(); | 
					
						
							|  |  |  |     return doc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned BreakpointCorrectionContext::fixLineNumber(const QString &fileName, | 
					
						
							|  |  |  |                                                     unsigned lineNumber) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CPlusPlus::Document::Ptr doc = m_snapshot.document(fileName); | 
					
						
							|  |  |  |     if (!doc || !doc->translationUnit()->ast()) | 
					
						
							|  |  |  |         doc = getParsedDocument(fileName, m_workingCopy, m_snapshot); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CPlusPlus::FindCdbBreakpoint findVisitor(doc->translationUnit()); | 
					
						
							|  |  |  |     const unsigned correctedLine = findVisitor(lineNumber); | 
					
						
							|  |  |  |     if (!correctedLine) { | 
					
						
							|  |  |  |         qWarning("Unable to find breakpoint location for %s:%d", | 
					
						
							|  |  |  |                  qPrintable(QDir::toNativeSeparators(fileName)), lineNumber); | 
					
						
							|  |  |  |         return lineNumber; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("Code model: Breakpoint line %u -> %u in %s", | 
					
						
							|  |  |  |                lineNumber, correctedLine, qPrintable(fileName)); | 
					
						
							|  |  |  |     return correctedLine; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | void CdbEngine::attemptBreakpointSynchronization() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-05-09 17:02:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("attemptBreakpointSynchronization in %s", stateName(state())); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     // Check if there is anything to be done at all.
 | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |     BreakHandler *handler = breakHandler(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     // Take ownership of the breakpoint. Requests insertion. TODO: Cpp only?
 | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |     foreach (BreakpointModelId id, handler->unclaimedBreakpointIds()) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         if (acceptsBreakpoint(id)) | 
					
						
							|  |  |  |             handler->setEngine(id, this); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     // Quick check: is there a need to change something? - Populate module cache
 | 
					
						
							|  |  |  |     bool changed = false; | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |     const BreakpointModelIds ids = handler->engineBreakpointIds(this); | 
					
						
							|  |  |  |     foreach (BreakpointModelId id, ids) { | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         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; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     if (debugBreakpoints) | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         qDebug("attemptBreakpointSynchronizationI %dms accessible=%d, %s %d breakpoints, changed=%d", | 
					
						
							|  |  |  |                elapsedLogTime(), m_accessible, stateName(state()), ids.size(), changed); | 
					
						
							|  |  |  |     if (!changed) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_accessible) { | 
					
						
							|  |  |  |         // No nested calls.
 | 
					
						
							|  |  |  |         if (m_specialStopMode != SpecialStopSynchronizeBreakpoints) | 
					
						
							|  |  |  |             doInterruptInferior(SpecialStopSynchronizeBreakpoints); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     // 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; | 
					
						
							| 
									
										
										
										
											2011-05-09 17:02:46 +02:00
										 |  |  |     QScopedPointer<BreakpointCorrectionContext> lineCorrection; | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |     foreach (BreakpointModelId id, ids) { | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         BreakpointParameters parameters = handler->breakpointData(id); | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |         BreakpointResponse response; | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         response.fromParameters(parameters); | 
					
						
							| 
									
										
										
										
											2011-06-30 11:01:10 +02:00
										 |  |  |         response.id = BreakpointResponseId(id.majorPart(), id.minorPart()); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         // 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(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         switch (handler->state(id)) { | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |         case BreakpointInsertRequested: | 
					
						
							| 
									
										
										
										
											2011-09-30 09:56:57 +02:00
										 |  |  |             if (parameters.type == BreakpointByFileAndLine | 
					
						
							|  |  |  |                 && m_options->breakpointCorrection) { | 
					
						
							| 
									
										
										
										
											2011-05-09 17:02:46 +02:00
										 |  |  |                 if (lineCorrection.isNull()) | 
					
						
							|  |  |  |                     lineCorrection.reset(new BreakpointCorrectionContext(debuggerCore()->cppCodeModelSnapshot(), | 
					
						
							|  |  |  |                                                                          CPlusPlus::CppModelManagerInterface::instance()->workingCopy())); | 
					
						
							|  |  |  |                 response.lineNumber = lineCorrection->fixLineNumber(parameters.fileName, parameters.lineNumber); | 
					
						
							|  |  |  |                 postCommand(cdbAddBreakpointCommand(response, m_sourcePathMappings, id, false), 0); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 postCommand(cdbAddBreakpointCommand(parameters, m_sourcePathMappings, id, false), 0); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             if (!parameters.enabled) | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |                 postCommand("bd " + QByteArray::number(id.majorPart()), 0); | 
					
						
							| 
									
										
										
										
											2010-11-18 17:44:04 +01:00
										 |  |  |             handler->notifyBreakpointInsertProceeding(id); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             handler->notifyBreakpointInsertOk(id); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             m_pendingBreakpointMap.insert(id, response); | 
					
						
							|  |  |  |             addedChanged = true; | 
					
						
							| 
									
										
										
										
											2011-03-16 13:38:27 +01:00
										 |  |  |             // Ensure enabled/disabled is correct in handler and line number is there.
 | 
					
						
							|  |  |  |             handler->setResponse(id, response); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             if (debugBreakpoints) | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |                 qDebug("Adding %d %s\n", id.toInternalId(), | 
					
						
							|  |  |  |                     qPrintable(response.toString())); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |         case BreakpointChangeRequested: | 
					
						
							| 
									
										
										
										
											2010-11-18 17:44:04 +01:00
										 |  |  |             handler->notifyBreakpointChangeProceeding(id); | 
					
						
							| 
									
										
										
										
											2011-03-16 13:38:27 +01:00
										 |  |  |             if (debugBreakpoints) | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |                 qDebug("Changing %d:\n    %s\nTo %s\n", id.toInternalId(), | 
					
						
							|  |  |  |                     qPrintable(handler->response(id).toString()), | 
					
						
							|  |  |  |                     qPrintable(parameters.toString())); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             if (parameters.enabled != handler->response(id).enabled) { | 
					
						
							|  |  |  |                 // Change enabled/disabled breakpoints without triggering update.
 | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |                 postCommand((parameters.enabled ? "be " : "bd ") | 
					
						
							|  |  |  |                     + QByteArray::number(id.majorPart()), 0); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |                 response.pending = false; | 
					
						
							|  |  |  |                 response.enabled = parameters.enabled; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                 handler->setResponse(id, response); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 // Delete and re-add, triggering update
 | 
					
						
							|  |  |  |                 addedChanged = true; | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |                 postCommand("bc " + QByteArray::number(id.majorPart()), 0); | 
					
						
							| 
									
										
										
										
											2011-03-07 09:24:44 +01:00
										 |  |  |                 postCommand(cdbAddBreakpointCommand(parameters, m_sourcePathMappings, id, false), 0); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |                 m_pendingBreakpointMap.insert(id, response); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             handler->notifyBreakpointChangeOk(id); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |         case BreakpointRemoveRequested: | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |             postCommand("bc " + QByteArray::number(id.majorPart()), 0); | 
					
						
							| 
									
										
										
										
											2010-11-22 17:06:08 +01:00
										 |  |  |             handler->notifyBreakpointRemoveProceeding(id); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             handler->notifyBreakpointRemoveOk(id); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             m_pendingBreakpointMap.remove(id); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     // List breakpoints and send responses
 | 
					
						
							|  |  |  |     if (addedChanged) | 
					
						
							|  |  |  |         postCommandSequence(CommandListBreakPoints); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  | // 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) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  |     // 1) Check cache.
 | 
					
						
							|  |  |  |     QMap<QString, NormalizedSourceFileName>::const_iterator it = m_normalizedFileCache.constFind(f); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     if (it != m_normalizedFileCache.constEnd()) | 
					
						
							|  |  |  |         return it.value(); | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  |     if (debugSourceMapping) | 
					
						
							|  |  |  |         qDebug(">sourceMapNormalizeFileNameFromDebugger %s", qPrintable(f)); | 
					
						
							|  |  |  |     // Do we have source path mappings? ->Apply.
 | 
					
						
							| 
									
										
										
										
											2011-03-07 09:24:44 +01:00
										 |  |  |     const QString fileName = cdbSourcePathMapping(QDir::toNativeSeparators(f), m_sourcePathMappings, | 
					
						
							|  |  |  |                                                   DebuggerToSource); | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  |     // Up/lower case normalization according to Windows.
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #ifdef Q_OS_WIN
 | 
					
						
							| 
									
										
										
										
											2011-04-28 11:11:34 +02:00
										 |  |  |     QString normalized = Utils::normalizePathName(fileName); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  |     QString normalized = fileName; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  |     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; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  | // Parse frame from GDBMI. Duplicate of the gdb code, but that
 | 
					
						
							|  |  |  | // has more processing.
 | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  | static StackFrames parseFrames(const GdbMi &gdbmi) | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     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(); | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  |             frame.usable = false; // To be decided after source path mapping.
 | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         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); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |     return rc; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  | unsigned CdbEngine::parseStackTrace(const GdbMi &data, bool sourceStepInto) | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-05 11:34:35 +01:00
										 |  |  |     // 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.
 | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |     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+"))) { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |             showMessage(QString::fromLatin1("Step into: Call instruction hit, performing additional step..."), LogMisc); | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |             return ParseStackStepInto; | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |         if (hasFile) { | 
					
						
							| 
									
										
										
										
											2011-03-04 17:28:15 +01:00
										 |  |  |             const NormalizedSourceFileName fileName = sourceMapNormalizeFileNameFromDebugger(frames.at(i).file); | 
					
						
							|  |  |  |             frames[i].file = fileName.fileName; | 
					
						
							|  |  |  |             frames[i].usable = fileName.exists; | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |             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); | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |         postCommandSequence(command->commandSequence); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-12-21 14:02:52 +01:00
										 |  |  |         showMessage(QString::fromLocal8Bit(command->errorMessage), LogError); | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  | void CdbEngine::handleExpression(const CdbExtensionCommandPtr &command) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int value = 0; | 
					
						
							|  |  |  |     if (command->success) { | 
					
						
							|  |  |  |         value = command->reply.toInt(); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2011-12-21 14:02:52 +01:00
										 |  |  |         showMessage(QString::fromLocal8Bit(command->errorMessage), LogError); | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     // 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."). | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |             arg(value).arg(cookie.id.toString()) : | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |             tr("Value 0 obtained from evaluating the condition of breakpoint %1, continuing."). | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |             arg(cookie.id.toString()); | 
					
						
							| 
									
										
										
										
											2011-03-23 15:08:57 +01:00
										 |  |  |         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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 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) { | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |         postExtensionCommand("stack", QByteArray(), 0, &CdbEngine::handleStackTrace, mask & ~CommandListStack); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (mask & CommandListRegisters) { | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |         QTC_ASSERT(threadsHandler()->currentThread() >= 0,  return); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         postExtensionCommand("registers", QByteArray(), 0, &CdbEngine::handleRegisters, mask & ~CommandListRegisters); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (mask & CommandListModules) { | 
					
						
							|  |  |  |         postExtensionCommand("modules", QByteArray(), 0, &CdbEngine::handleModules, mask & ~CommandListModules); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     if (mask & CommandListBreakPoints) { | 
					
						
							|  |  |  |         postExtensionCommand("breakpoints", QByteArray("-v"), 0, | 
					
						
							|  |  |  |                              &CdbEngine::handleBreakPoints, mask & ~CommandListBreakPoints); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  | void CdbEngine::handleWidgetAt(const CdbExtensionCommandPtr &reply) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool success = false; | 
					
						
							|  |  |  |     QString message; | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |         if (!reply->success) { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |             message = QString::fromLatin1(reply->errorMessage); | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Should be "namespace::QWidget:0x555"
 | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |         QString watchExp = QString::fromLatin1(reply->reply); | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |         const int sepPos = watchExp.lastIndexOf(QLatin1Char(':')); | 
					
						
							|  |  |  |         if (sepPos == -1) { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |             message = QString::fromLatin1("Invalid output: %1").arg(watchExp); | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // 0x000 -> nothing found
 | 
					
						
							|  |  |  |         if (!watchExp.mid(sepPos + 1).toULongLong(0, 0)) { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |             message = QString::fromLatin1("No widget could be found at %1, %2.").arg(m_watchPointX).arg(m_watchPointY); | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |             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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  | static inline void formatCdbBreakPointResponse(BreakpointModelId id, const BreakpointResponse &r, | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |                                                   QTextStream &str) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-06-21 16:45:23 +02:00
										 |  |  |     str << "Obtained breakpoint " << id << " (#" << r.id.majorPart() << ')'; | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     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) | 
					
						
							| 
									
										
										
										
											2011-02-25 15:27:13 +01:00
										 |  |  |         qDebug("CdbEngine::handleBreakPoints: success=%d: %s", reply->success, reply->reply.constData()); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     if (!reply->success) { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |         showMessage(QString::fromLatin1(reply->errorMessage), LogError); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     GdbMi value; | 
					
						
							|  |  |  |     value.fromString(reply->reply); | 
					
						
							|  |  |  |     if (value.type() != GdbMi::List) { | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |         showMessage(QString::fromLatin1("Unabled to parse breakpoints reply"), LogError); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         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; | 
					
						
							| 
									
										
										
										
											2011-06-30 11:01:10 +02:00
										 |  |  |         parseBreakPoint(breakPointG, &reportedResponse); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |         if (debugBreakpoints) | 
					
						
							| 
									
										
										
										
											2011-06-30 11:01:10 +02:00
										 |  |  |             qDebug("  Parsed %d: pending=%d %s\n", reportedResponse.id.majorPart(), | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |                 reportedResponse.pending, | 
					
						
							|  |  |  |                 qPrintable(reportedResponse.toString())); | 
					
						
							| 
									
										
										
										
											2011-06-30 11:01:10 +02:00
										 |  |  |         if (reportedResponse.id.isValid() && !reportedResponse.pending) { | 
					
						
							|  |  |  |             const BreakpointModelId mid = handler->findBreakpointByResponseId(reportedResponse.id); | 
					
						
							| 
									
										
										
										
											2012-04-17 08:01:25 +02:00
										 |  |  |             QTC_ASSERT(mid.isValid(), continue); | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |             const PendingBreakPointMap::iterator it = m_pendingBreakpointMap.find(mid); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             if (it != m_pendingBreakpointMap.end()) { | 
					
						
							|  |  |  |                 // Complete the response and set on handler.
 | 
					
						
							|  |  |  |                 BreakpointResponse ¤tResponse = it.value(); | 
					
						
							| 
									
										
										
										
											2011-06-21 16:45:23 +02:00
										 |  |  |                 currentResponse.id = reportedResponse.id; | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |                 currentResponse.address = reportedResponse.address; | 
					
						
							|  |  |  |                 currentResponse.module = reportedResponse.module; | 
					
						
							|  |  |  |                 currentResponse.pending = reportedResponse.pending; | 
					
						
							|  |  |  |                 currentResponse.enabled = reportedResponse.enabled; | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |                 formatCdbBreakPointResponse(mid, currentResponse, str); | 
					
						
							| 
									
										
										
										
											2011-03-16 13:38:27 +01:00
										 |  |  |                 if (debugBreakpoints) | 
					
						
							| 
									
										
										
										
											2011-06-30 11:01:10 +02:00
										 |  |  |                     qDebug("  Setting for %d: %s\n", currentResponse.id.majorPart(), | 
					
						
							| 
									
										
										
										
											2011-06-15 14:02:26 +02:00
										 |  |  |                         qPrintable(currentResponse.toString())); | 
					
						
							| 
									
										
										
										
											2011-06-24 16:25:30 +02:00
										 |  |  |                 handler->setResponse(mid, currentResponse); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |                 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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  | 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'."). | 
					
						
							| 
									
										
										
										
											2012-02-16 09:55:47 +02:00
										 |  |  |                     arg(QString::fromLatin1(stateName(state()))), LogWarning); | 
					
						
							| 
									
										
										
										
											2011-01-18 11:40:45 +01:00
										 |  |  |         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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  | void CdbEngine::handleCustomSpecialStop(const QVariant &v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (qVariantCanConvert<MemoryChangeCookie>(v)) { | 
					
						
							|  |  |  |         const MemoryChangeCookie changeData = qVariantValue<MemoryChangeCookie>(v); | 
					
						
							|  |  |  |         postCommand(cdbWriteMemoryCommand(changeData.address, changeData.data), 0); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-04-19 13:57:25 +02:00
										 |  |  |     if (qVariantCanConvert<MemoryViewCookie>(v)) { | 
					
						
							|  |  |  |         postFetchMemory(qVariantValue<MemoryViewCookie>(v)); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-28 14:33:35 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | } // namespace Internal
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } // namespace Debugger
 |