| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2011-01-11 16:28:15 +01:00
										 |  |  | ** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | ** Contact: Nokia Corporation (qt-info@nokia.com) | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-12-17 17:14:20 +01:00
										 |  |  | ** No Commercial Usage | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-12-17 17:14:20 +01:00
										 |  |  | ** This file contains pre-release code and may not be distributed. | 
					
						
							|  |  |  | ** You may use this file in accordance with the terms and conditions | 
					
						
							|  |  |  | ** contained in the Technology Preview License Agreement accompanying | 
					
						
							|  |  |  | ** this package. | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | ** GNU Lesser General Public License Usage | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Alternatively, 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-12-17 17:14:20 +01:00
										 |  |  | ** In addition, as a special exception, Nokia gives you certain additional | 
					
						
							|  |  |  | ** rights.  These rights are described in the Nokia Qt LGPL Exception | 
					
						
							|  |  |  | ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** If you have questions regarding the use of this file, please contact | 
					
						
							|  |  |  | ** Nokia at qt-info@nokia.com. | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:22:55 +01:00
										 |  |  | #include "cdbengine.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-10 10:14:23 +01:00
										 |  |  | #include "debuggerstartparameters.h"
 | 
					
						
							|  |  |  | #include "disassemblerlines.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:22:55 +01:00
										 |  |  | #include "cdboptions.h"
 | 
					
						
							|  |  |  | #include "cdboptionspage.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #include "bytearrayinputstream.h"
 | 
					
						
							|  |  |  | #include "breakpoint.h"
 | 
					
						
							|  |  |  | #include "breakhandler.h"
 | 
					
						
							|  |  |  | #include "stackframe.h"
 | 
					
						
							|  |  |  | #include "stackhandler.h"
 | 
					
						
							|  |  |  | #include "watchhandler.h"
 | 
					
						
							|  |  |  | #include "threadshandler.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-10 10:14:23 +01:00
										 |  |  | #include "moduleshandler.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #include "debuggeractions.h"
 | 
					
						
							|  |  |  | #include "debuggercore.h"
 | 
					
						
							|  |  |  | #include "registerhandler.h"
 | 
					
						
							| 
									
										
										
										
											2010-12-08 12:43:11 +01:00
										 |  |  | #include "disassembleragent.h"
 | 
					
						
							|  |  |  | #include "memoryagent.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  | #include "debuggerrunner.h"
 | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  | #include "debuggertooltipmanager.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #include "cdbparsehelpers.h"
 | 
					
						
							|  |  |  | #include "watchutils.h"
 | 
					
						
							|  |  |  | #include "gdb/gdbmi.h"
 | 
					
						
							| 
									
										
										
										
											2010-11-18 16:22:41 +01:00
										 |  |  | #include "shared/cdbsymbolpathlisteditor.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <coreplugin/icore.h>
 | 
					
						
							| 
									
										
										
										
											2010-11-22 17:06:08 +01:00
										 |  |  | #include <texteditor/itexteditor.h>
 | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  | #include <projectexplorer/toolchain.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>
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <QtCore/QCoreApplication>
 | 
					
						
							|  |  |  | #include <QtCore/QFileInfo>
 | 
					
						
							|  |  |  | #include <QtCore/QDir>
 | 
					
						
							|  |  |  | #include <QtCore/QDebug>
 | 
					
						
							|  |  |  | #include <QtCore/QTextStream>
 | 
					
						
							|  |  |  | #include <QtCore/QDateTime>
 | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  | #include <QtGui/QToolTip>
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  | #include <QtGui/QMainWindow>
 | 
					
						
							|  |  |  | #include <QtGui/QMessageBox>
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef Q_OS_WIN
 | 
					
						
							|  |  |  | #    include <utils/winutils.h>
 | 
					
						
							|  |  |  | #    include "dbgwinutils.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #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-01-14 16:50:31 +01:00
										 |  |  | enum { debugWatches = 0 }; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | enum { debugBreakpoints = 0 }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #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-02 12:43:42 +01:00
										 |  |  | struct SourceLocationCookie | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     explicit SourceLocationCookie(const QString &f = QString(), int l = 0) : | 
					
						
							|  |  |  |              fileName(f), lineNumber(l) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QString fileName; | 
					
						
							|  |  |  |     int lineNumber; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | Q_DECLARE_METATYPE(Debugger::Internal::SourceLocationCookie) | 
					
						
							| 
									
										
										
										
											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-01-19 10:25:32 +01:00
										 |  |  | static inline bool isConsole(const DebuggerStartParameters &sp) | 
					
						
							| 
									
										
										
										
											2011-02-02 12:43:42 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     return (sp.startMode == StartInternal || sp.startMode == StartExternal) | 
					
						
							|  |  |  |         && sp.useTerminal; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											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 AttachTcf: | 
					
						
							|  |  |  |     case AttachCore: | 
					
						
							|  |  |  |     case StartRemoteGdb: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  | static inline QString msgCdbDisabled(ToolChainType tc) | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     return CdbEngine::tr("The CDB debug engine required for %1 is currently disabled."). | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  |                       arg(ToolChain::toolChainName(tc)); | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | // 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-01-07 15:04:11 +01:00
										 |  |  |     if (!op || !op->options()->isValid()) { | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  |         *errorMessage = msgCdbDisabled(sp.toolChainType); | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!validMode(sp.startMode)) { | 
					
						
							|  |  |  |         *errorMessage = CdbEngine::tr("The CDB debug engine does not support start mode %1.").arg(sp.startMode); | 
					
						
							|  |  |  |         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-01-07 19:50:41 +01:00
										 |  |  | ConfigurationCheck checkCdbConfiguration(ToolChainType toolChain) | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  |     ConfigurationCheck check; | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  |     switch (toolChain) { | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  |     case ToolChain_MinGW: // Do our best
 | 
					
						
							|  |  |  |     case ToolChain_MSVC: | 
					
						
							|  |  |  |     case ToolChain_WINCE: | 
					
						
							|  |  |  |     case ToolChain_OTHER: | 
					
						
							|  |  |  |     case ToolChain_UNKNOWN: | 
					
						
							|  |  |  |     case ToolChain_INVALID: | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  |         if (!isCdbEngineEnabled()) { | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  |             check.errorMessage = msgCdbDisabled(toolChain); | 
					
						
							|  |  |  |             check.settingsPage = CdbOptionsPage::settingsId(); | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         //: %1 is something like "GCCE" or "Intel C++ Compiler (Linux)" (see ToolChain context)
 | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  |         check.errorMessage = CdbEngine::tr("The CDB debug engine does not support the %1 toolchain."). | 
					
						
							|  |  |  |                     arg(ToolChain::toolChainName(toolChain)); | 
					
						
							|  |  |  |         check.settingsPage = CdbOptionsPage::settingsId(); | 
					
						
							| 
									
										
										
										
											2011-01-07 15:04:11 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-01-07 19:50:41 +01:00
										 |  |  |     return check; | 
					
						
							| 
									
										
										
										
											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) : | 
					
						
							|  |  |  |     DebuggerEngine(sp, 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_inferiorPid(0), | 
					
						
							|  |  |  |     m_accessible(false), | 
					
						
							|  |  |  |     m_specialStopMode(NoSpecialStop), | 
					
						
							|  |  |  |     m_nextCommandToken(0), | 
					
						
							|  |  |  |     m_currentBuiltinCommandIndex(-1), | 
					
						
							|  |  |  |     m_extensionCommandPrefixBA("!"QT_CREATOR_CDB_EXT"."), | 
					
						
							|  |  |  |     m_operateByInstructionPending(true), | 
					
						
							|  |  |  |     m_operateByInstruction(true), // Default CDB setting
 | 
					
						
							|  |  |  |     m_notifyEngineShutdownOnTermination(false), | 
					
						
							|  |  |  |     m_hasDebuggee(false), | 
					
						
							| 
									
										
										
										
											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_wX86BreakpointCount(0), | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     m_inferiorPid = 0; | 
					
						
							|  |  |  |     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_wX86BreakpointCount = 0; | 
					
						
							|  |  |  |     m_watchPointX = m_watchPointY = 0; | 
					
						
							|  |  |  |     m_ignoreCdbOutput = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_outputBuffer.clear(); | 
					
						
							|  |  |  |     m_builtinCommandQueue.clear(); | 
					
						
							|  |  |  |     m_extensionCommandQueue.clear(); | 
					
						
							|  |  |  |     m_extensionMessageBuffer.clear(); | 
					
						
							|  |  |  |     m_pendingBreakpointMap.clear(); | 
					
						
							|  |  |  |     QTC_ASSERT(m_process.state() != QProcess::Running, Utils::SynchronousProcess::stopProcess(m_process); ) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | CdbEngine::~CdbEngine() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::operateByInstructionTriggered(bool operateByInstruction) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |     if (state() == InferiorStopOk) { | 
					
						
							|  |  |  |         syncOperateByInstruction(operateByInstruction); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         // To be set next time session becomes accessible
 | 
					
						
							|  |  |  |         m_operateByInstructionPending = 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; | 
					
						
							|  |  |  |     QTC_ASSERT(m_accessible, return; ) | 
					
						
							|  |  |  |     m_operateByInstruction = operateByInstruction; | 
					
						
							|  |  |  |     postCommand(m_operateByInstruction ? QByteArray("l-t") : QByteArray("l+t"), 0); | 
					
						
							|  |  |  |     postCommand(m_operateByInstruction ? QByteArray("l-s") : QByteArray("l+s"), 0); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |     const 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; | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |     const QByteArray iname = QByteArray(localsPrefixC) + exp.toAscii(); | 
					
						
							| 
									
										
										
										
											2011-01-25 13:59:25 +01:00
										 |  |  |     const QModelIndex index = watchHandler()->itemIndex(iname); | 
					
						
							| 
									
										
										
										
											2011-02-17 10:08:57 +01:00
										 |  |  |     if (!index.isValid()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     DebuggerTreeViewToolTipWidget *tw = new DebuggerTreeViewToolTipWidget; | 
					
						
							|  |  |  |     tw->setContext(context); | 
					
						
							|  |  |  |     tw->setDebuggerModel(LocalsWatch); | 
					
						
							|  |  |  |     tw->setExpression(exp); | 
					
						
							|  |  |  |     tw->acquireEngine(this); | 
					
						
							|  |  |  |     DebuggerToolTipManager::instance()->add(mousePos, tw); | 
					
						
							|  |  |  |     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() | 
					
						
							|  |  |  |                      << "/lib/" << (is64Bit ? QT_CREATOR_CDB_EXT"64" : QT_CREATOR_CDB_EXT"32") | 
					
						
							|  |  |  |                      << '/' << QT_CREATOR_CDB_EXT << ".dll"; | 
					
						
							|  |  |  |     return rc; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Determine environment for CDB.exe, start out with run config and
 | 
					
						
							|  |  |  | // add CDB extension path merged with system value should there be one.
 | 
					
						
							|  |  |  | static QStringList mergeEnvironment(QStringList runConfigEnvironment, | 
					
						
							|  |  |  |                                     QString cdbExtensionPath) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Determine CDB extension path from Qt Creator
 | 
					
						
							|  |  |  |     static const char cdbExtensionPathVariableC[] = "_NT_DEBUGGER_EXTENSION_PATH"; | 
					
						
							|  |  |  |     const QByteArray oldCdbExtensionPath = qgetenv(cdbExtensionPathVariableC); | 
					
						
							|  |  |  |     if (!oldCdbExtensionPath.isEmpty()) { | 
					
						
							|  |  |  |         cdbExtensionPath.append(QLatin1Char(';')); | 
					
						
							|  |  |  |         cdbExtensionPath.append(QString::fromLocal8Bit(oldCdbExtensionPath)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // We do not assume someone sets _NT_DEBUGGER_EXTENSION_PATH in the run
 | 
					
						
							|  |  |  |     // config, just to make sure, delete any existing entries
 | 
					
						
							|  |  |  |     const QString cdbExtensionPathVariableAssign = | 
					
						
							|  |  |  |             QLatin1String(cdbExtensionPathVariableC) + QLatin1Char('='); | 
					
						
							|  |  |  |     for (QStringList::iterator it = runConfigEnvironment.begin(); it != runConfigEnvironment.end() ; ) { | 
					
						
							|  |  |  |         if (it->startsWith(cdbExtensionPathVariableAssign)) { | 
					
						
							|  |  |  |             it = runConfigEnvironment.erase(it); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             ++it; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     runConfigEnvironment.append(cdbExtensionPathVariableAssign + | 
					
						
							|  |  |  |                                 QDir::toNativeSeparators(cdbExtensionPath)); | 
					
						
							|  |  |  |     return runConfigEnvironment; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int CdbEngine::elapsedLogTime() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const int elapsed = m_logTime.elapsed(); | 
					
						
							|  |  |  |     const int delta = elapsed - m_elapsedLogTime; | 
					
						
							|  |  |  |     m_elapsedLogTime = elapsed; | 
					
						
							|  |  |  |     return delta; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |     connect(m_consoleStub.data(), SIGNAL(processMessage(QString, bool)), | 
					
						
							|  |  |  |             SLOT(consoleStubMessage(QString, bool))); | 
					
						
							|  |  |  |     connect(m_consoleStub.data(), SIGNAL(processStarted()), | 
					
						
							|  |  |  |             SLOT(consoleStubProcessStarted())); | 
					
						
							|  |  |  |     connect(m_consoleStub.data(), SIGNAL(wrapperStopped()), | 
					
						
							|  |  |  |             SLOT(consoleStubExited())); | 
					
						
							|  |  |  |     m_consoleStub->setWorkingDirectory(sp.workingDirectory); | 
					
						
							|  |  |  |     m_consoleStub->setEnvironment(sp.environment); | 
					
						
							|  |  |  |     if (!m_consoleStub->start(sp.executable, sp.processArgs)) { | 
					
						
							|  |  |  |         *errorMessage = tr("The console process '%1' could not be started.").arg(sp.executable); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::consoleStubMessage(const QString &msg, bool isError) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("consoleStubProcessMessage() in %s error=%d %s", stateName(state()), isError, qPrintable(msg)); | 
					
						
							|  |  |  |     if (isError) { | 
					
						
							|  |  |  |         if (state() == EngineSetupRequested) { | 
					
						
							|  |  |  |             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupFailed") | 
					
						
							|  |  |  |             notifyEngineSetupFailed(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineIll") | 
					
						
							|  |  |  |             notifyEngineIll(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         nonModalMessageBox(QMessageBox::Critical, tr("Debugger Error"), msg); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         showMessage(msg, AppOutput); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::consoleStubProcessStarted() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("consoleStubProcessStarted() PID=%lld", m_consoleStub->applicationPID()); | 
					
						
							|  |  |  |     // Attach to console process.
 | 
					
						
							|  |  |  |     DebuggerStartParameters attachParameters = startParameters(); | 
					
						
							|  |  |  |     attachParameters.executable.clear(); | 
					
						
							|  |  |  |     attachParameters.processArgs.clear(); | 
					
						
							|  |  |  |     attachParameters.attachPID = m_consoleStub->applicationPID(); | 
					
						
							|  |  |  |     attachParameters.startMode = AttachExternal; | 
					
						
							|  |  |  |     showMessage(QString::fromLatin1("Attaching to %1...").arg(attachParameters.attachPID), LogMisc); | 
					
						
							|  |  |  |     QString errorMessage; | 
					
						
							|  |  |  |     if (!launchCDB(attachParameters, &errorMessage)) { | 
					
						
							|  |  |  |         showMessage(errorMessage, LogError); | 
					
						
							|  |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupFailed") | 
					
						
							|  |  |  |         notifyEngineSetupFailed(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::consoleStubExited() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::setupEngine() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug(">setupEngine"); | 
					
						
							|  |  |  |     // Nag to add symbol server
 | 
					
						
							|  |  |  |     if (CdbSymbolPathListEditor::promptToAddSymbolServer(CdbOptions::settingsGroup(), | 
					
						
							|  |  |  |                                                          &(m_options->symbolPaths))) | 
					
						
							|  |  |  |         m_options->toSettings(Core::ICore::instance()->settings()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  |     const bool launchConsole = isConsole(sp); | 
					
						
							|  |  |  |     m_effectiveStartMode = launchConsole ? AttachExternal : sp.startMode; | 
					
						
							|  |  |  |     const bool ok = launchConsole ? | 
					
						
							|  |  |  |                 startConsole(startParameters(), &errorMessage) : | 
					
						
							|  |  |  |                 launchCDB(startParameters(), &errorMessage); | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("<setupEngine ok=%d", ok); | 
					
						
							|  |  |  |     if (!ok) { | 
					
						
							|  |  |  |         showMessage(errorMessage, LogError); | 
					
						
							|  |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSetupFailed") | 
					
						
							|  |  |  |         notifyEngineSetupFailed(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CdbEngine::launchCDB(const DebuggerStartParameters &sp, QString *errorMessage) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("launchCDB startMode=%d", sp.startMode); | 
					
						
							| 
									
										
										
										
											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:
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     // Determine extension lib name and path to use
 | 
					
						
							|  |  |  |     // The extension is passed as relative name with the path variable set
 | 
					
						
							|  |  |  |     //(does not work with absolute path names)
 | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     const QFileInfo extensionFi(CdbEngine::extensionLibraryName(m_options->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; | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     const bool isRemote = sp.startMode == AttachToRemote; | 
					
						
							|  |  |  |     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") | 
					
						
							|  |  |  |               << QString::fromAscii(".idle_cmd " + m_extensionCommandPrefixBA + "idle"); | 
					
						
							|  |  |  |     if (sp.useTerminal) // Separate console
 | 
					
						
							|  |  |  |         arguments << QLatin1String("-2"); | 
					
						
							|  |  |  |     if (!m_options->symbolPaths.isEmpty()) | 
					
						
							|  |  |  |         arguments << QLatin1String("-y") << m_options->symbolPaths.join(QString(QLatin1Char(';'))); | 
					
						
							|  |  |  |     if (!m_options->sourcePaths.isEmpty()) | 
					
						
							|  |  |  |         arguments << QLatin1String("-srcpath") << m_options->sourcePaths.join(QString(QLatin1Char(';'))); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     case AttachToRemote: | 
					
						
							|  |  |  |         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); | 
					
						
							|  |  |  |         if (sp.startMode == AttachCrashedExternal) | 
					
						
							|  |  |  |             arguments << QLatin1String("-e") << sp.crashParameter << QLatin1String("-g"); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         *errorMessage = QString::fromLatin1("Internal error: Unsupported start mode %1.").arg(sp.startMode); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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 executable = m_options->executable; | 
					
						
							|  |  |  |     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(); | 
					
						
							|  |  |  |     m_process.setEnvironment(mergeEnvironment(sp.environment.toStringList(), 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"); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     attemptBreakpointSynchronization(); | 
					
						
							|  |  |  |     postExtensionCommand("pid", QByteArray(), 0, &CdbEngine::handlePid); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::runEngine() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("runEngine"); | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |     // Resume the threads frozen by the console stub.
 | 
					
						
							|  |  |  |     if (isConsole(startParameters())) | 
					
						
							|  |  |  |         postCommand("~* m", 0); | 
					
						
							| 
									
										
										
										
											2010-12-17 12:00:44 +01:00
										 |  |  |     foreach (const QString &breakEvent, m_options->breakEvents) | 
					
						
							|  |  |  |             postCommand(QByteArray("sxe ") + breakEvent.toAscii(), 0); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     postCommand("g", 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     if (m_accessible) { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         if (m_effectiveStartMode == AttachExternal || m_effectiveStartMode == 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()) { | 
					
						
							|  |  |  |         // detach: Wait for debugger to finish.
 | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |         if (m_effectiveStartMode == AttachExternal) | 
					
						
							| 
									
										
										
										
											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-01-19 10:25:32 +01:00
										 |  |  |         if (m_effectiveStartMode == AttachToRemote) { | 
					
						
							| 
									
										
										
										
											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-01-19 10:25:32 +01:00
										 |  |  |         STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyEngineSpontaneousShutdown") | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         notifyEngineSpontaneousShutdown(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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")); | 
					
						
							|  |  |  |         watchHandler()->insertData(item); | 
					
						
							|  |  |  |         showMessage(QString::fromLatin1("Unable to add watch item '%1'/'%2': %3"). | 
					
						
							|  |  |  |                     arg(QString::fromAscii(item.iname), QString::fromAscii(item.exp), | 
					
						
							|  |  |  |                         reply->errorMessage), LogError); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |     postExtensionCommand(isWatch ? "watches" : "locals", localsArguments, 0, &CdbEngine::handleLocals); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned CdbEngine::debuggerCapabilities() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return DisassemblerCapability | RegisterCapability | ShowMemoryCapability | 
					
						
							| 
									
										
										
										
											2011-01-14 16:50:31 +01:00
										 |  |  |            |WatchpointCapability|JumpToLineCapability|AddWatcherCapability | 
					
						
							| 
									
										
										
										
											2011-02-07 11:04:31 +01:00
										 |  |  |            |BreakOnThrowAndCatchCapability // Sort-of: Can break on throw().
 | 
					
						
							|  |  |  |            |BreakModuleCapability; | 
					
						
							| 
									
										
										
										
											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-02-08 12:08:59 +01:00
										 |  |  |     return m_effectiveStartMode != AttachToRemote && m_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()); | 
					
						
							| 
									
										
										
										
											2010-11-19 16:13:22 +01:00
										 |  |  |     if (canInterruptInferior()) { | 
					
						
							|  |  |  |         doInterruptInferior(NoSpecialStop); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         showMessage(tr("Interrupting is not possible in remote sessions."), LogError); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::doInterruptInferior(SpecialStopMode sm) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef Q_OS_WIN
 | 
					
						
							|  |  |  |     const SpecialStopMode oldSpecialMode = m_specialStopMode; | 
					
						
							|  |  |  |     m_specialStopMode = sm; | 
					
						
							|  |  |  |     QString errorMessage; | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |     showMessage(QString::fromLatin1("Interrupting process %1...").arg(m_inferiorPid), LogMisc); | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     if (!winDebugBreakProcess(m_inferiorPid, &errorMessage)) { | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         m_specialStopMode = oldSpecialMode; | 
					
						
							| 
									
										
										
										
											2011-02-08 12:08:59 +01:00
										 |  |  |         showMessage(QString::fromLatin1("Cannot interrupt process %1: %2").arg(m_inferiorPid).arg(errorMessage), LogError); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     Q_UNUSED(sm) | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeRunToLine(const QString &fileName, int lineNumber) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Add one-shot breakpoint
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     BreakpointParameters bp(BreakpointByFileAndLine); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     bp.fileName = fileName; | 
					
						
							|  |  |  |     bp.lineNumber = lineNumber; | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     postCommand(cdbAddBreakpointCommand(bp, BreakpointId(-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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     postCommand(cdbAddBreakpointCommand(bp, BreakpointId(-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(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     QTC_ASSERT(regnr < registers.size(), return) | 
					
						
							|  |  |  |     // Value is decimal or 0x-hex-prefixed
 | 
					
						
							|  |  |  |     QByteArray cmd; | 
					
						
							|  |  |  |     ByteArrayInputStream str(cmd); | 
					
						
							|  |  |  |     str << "r " << registers.at(regnr).name << '=' << value; | 
					
						
							|  |  |  |     postCommand(cmd, 0); | 
					
						
							|  |  |  |     reloadRegisters(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeJumpToLine(const QString & fileName, int lineNumber) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QByteArray cmd; | 
					
						
							|  |  |  |     ByteArrayInputStream str(cmd); | 
					
						
							|  |  |  |     // Resolve source line address and go to that location
 | 
					
						
							|  |  |  |     str << "? `" << QDir::toNativeSeparators(fileName) << ':' << lineNumber << '`'; | 
					
						
							|  |  |  |     const QVariant cookie = qVariantFromValue(SourceLocationCookie(fileName, lineNumber)); | 
					
						
							|  |  |  |     postBuiltinCommand(cmd, 0, &CdbEngine::handleJumpToLineAddressResolution, 0, cookie); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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
 | 
					
						
							|  |  |  |     QByteArray answer = cmd->reply.front(); | 
					
						
							|  |  |  |     const int equalPos = answer.indexOf(" = "); | 
					
						
							|  |  |  |     if (equalPos == -1) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     answer.remove(0, equalPos + 3); | 
					
						
							|  |  |  |     QTC_ASSERT(qVariantCanConvert<SourceLocationCookie>(cmd->cookie), return ; ) | 
					
						
							|  |  |  |     const SourceLocationCookie cookie = qvariant_cast<SourceLocationCookie>(cmd->cookie); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QByteArray registerCmd; | 
					
						
							|  |  |  |     ByteArrayInputStream str(registerCmd); | 
					
						
							| 
									
										
										
										
											2010-12-13 17:19:21 +01:00
										 |  |  |     // PC-register depending on 64/32bit.
 | 
					
						
							|  |  |  |     str << "r " << (m_options->is64bit ? "rip" : "eip") << "=0x" << answer; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     postCommand(registerCmd, 0); | 
					
						
							| 
									
										
										
										
											2010-12-16 19:06:33 +01:00
										 |  |  |     gotoLocation(Location(cookie.fileName, cookie.lineNumber)); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |     str << m_extensionCommandPrefixBA << "assign " << w->iname << '=' << value.toString(); | 
					
						
							|  |  |  |     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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::executeDebuggerCommand(const QString &command) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     postCommand(command.toLocal8Bit(), QuietCommand); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Post command without callback
 | 
					
						
							|  |  |  | void CdbEngine::postCommand(const QByteArray &cmd, unsigned flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("CdbEngine::postCommand %dms '%s' %u %s\n", | 
					
						
							|  |  |  |                elapsedLogTime(), cmd.constData(), flags, stateName(state())); | 
					
						
							|  |  |  |     if (!(flags & QuietCommand)) | 
					
						
							|  |  |  |         showMessage(QString::fromLocal8Bit(cmd), LogInput); | 
					
						
							|  |  |  |     m_process.write(cmd + '\n'); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Post a built-in-command producing free-format output with a callback.
 | 
					
						
							|  |  |  | // In order to catch the output, it is enclosed in 'echo' commands
 | 
					
						
							|  |  |  | // printing a specially formatted token to be identifiable in the output.
 | 
					
						
							|  |  |  | void CdbEngine::postBuiltinCommand(const QByteArray &cmd, unsigned flags, | 
					
						
							|  |  |  |                                    BuiltinCommandHandler handler, | 
					
						
							|  |  |  |                                    unsigned nextCommandFlag, | 
					
						
							|  |  |  |                                    const QVariant &cookie) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_accessible) { | 
					
						
							|  |  |  |         const QString msg = QString::fromLatin1("Attempt to issue builtin command '%1' to non-accessible session (%2)") | 
					
						
							|  |  |  |                 .arg(QString::fromLocal8Bit(cmd), QString::fromAscii(stateName(state()))); | 
					
						
							|  |  |  |         showMessage(msg, LogError); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!flags & QuietCommand) | 
					
						
							|  |  |  |         showMessage(QString::fromLocal8Bit(cmd), LogInput); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const int token = m_nextCommandToken++; | 
					
						
							|  |  |  |     CdbBuiltinCommandPtr pendingCommand(new CdbBuiltinCommand(cmd, token, flags, handler, nextCommandFlag, cookie)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_builtinCommandQueue.push_back(pendingCommand); | 
					
						
							|  |  |  |     // Enclose command in echo-commands for token
 | 
					
						
							|  |  |  |     QByteArray fullCmd; | 
					
						
							|  |  |  |     ByteArrayInputStream str(fullCmd); | 
					
						
							|  |  |  |     str << ".echo \"" << m_tokenPrefix << token << "<\"\n" | 
					
						
							|  |  |  |             << cmd << "\n.echo \"" << m_tokenPrefix << token << ">\"\n"; | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("CdbEngine::postBuiltinCommand %dms '%s' flags=%u token=%d %s next=%u, cookie='%s', pending=%d, sequence=0x%x", | 
					
						
							|  |  |  |                elapsedLogTime(), cmd.constData(), flags, token, stateName(state()), nextCommandFlag, qPrintable(cookie.toString()), | 
					
						
							|  |  |  |                m_builtinCommandQueue.size(), nextCommandFlag); | 
					
						
							|  |  |  |     if (debug > 1) | 
					
						
							|  |  |  |         qDebug("CdbEngine::postBuiltinCommand: resulting command '%s'\n", | 
					
						
							|  |  |  |                fullCmd.constData()); | 
					
						
							|  |  |  |     m_process.write(fullCmd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Post an extension command producing one-line output with a callback,
 | 
					
						
							|  |  |  | // pass along token for identification in queue.
 | 
					
						
							|  |  |  | void CdbEngine::postExtensionCommand(const QByteArray &cmd, | 
					
						
							|  |  |  |                                      const QByteArray &arguments, | 
					
						
							|  |  |  |                                      unsigned flags, | 
					
						
							|  |  |  |                                      ExtensionCommandHandler handler, | 
					
						
							|  |  |  |                                      unsigned nextCommandFlag, | 
					
						
							|  |  |  |                                      const QVariant &cookie) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_accessible) { | 
					
						
							|  |  |  |         const QString msg = QString::fromLatin1("Attempt to issue extension command '%1' to non-accessible session (%2)") | 
					
						
							|  |  |  |                 .arg(QString::fromLocal8Bit(cmd), QString::fromAscii(stateName(state()))); | 
					
						
							|  |  |  |         showMessage(msg, LogError); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const int token = m_nextCommandToken++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Format full command with token to be recognizeable in the output
 | 
					
						
							|  |  |  |     QByteArray fullCmd; | 
					
						
							|  |  |  |     ByteArrayInputStream str(fullCmd); | 
					
						
							|  |  |  |     str << m_extensionCommandPrefixBA << cmd << " -t " << token; | 
					
						
							|  |  |  |     if (!arguments.isEmpty()) | 
					
						
							|  |  |  |         str <<  ' ' << arguments; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     QTC_ASSERT(index < frames.size(), return; ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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.
 | 
					
						
							|  |  |  |         watchHandler()->beginCycle(); | 
					
						
							|  |  |  |         watchHandler()->endCycle(); | 
					
						
							|  |  |  |         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) { | 
					
						
							|  |  |  |         watchHandler()->beginCycle(); | 
					
						
							|  |  |  |         watchHandler()->endCycle(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const StackFrame frame = stackHandler()->currentFrame(); | 
					
						
							|  |  |  |     if (!frame.isUsable()) { | 
					
						
							|  |  |  |         watchHandler()->beginCycle(); | 
					
						
							|  |  |  |         watchHandler()->endCycle(); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2010-11-23 13:36:39 +01:00
										 |  |  |     // Uninitialized variables if desired
 | 
					
						
							|  |  |  |     if (debuggerCore()->boolSetting(UseCodeModel)) { | 
					
						
							|  |  |  |         QStringList uninitializedVariables; | 
					
						
							|  |  |  |         getUninitializedVariables(debuggerCore()->cppCodeModelSnapshot(), | 
					
						
							|  |  |  |                                   frame.function, frame.file, frame.line, &uninitializedVariables); | 
					
						
							|  |  |  |         if (!uninitializedVariables.isEmpty()) { | 
					
						
							|  |  |  |             str << blankSeparator << "-u "; | 
					
						
							|  |  |  |             int i = 0; | 
					
						
							|  |  |  |             foreach(const QString &u, uninitializedVariables) { | 
					
						
							|  |  |  |                 if (i++) | 
					
						
							|  |  |  |                     str << ','; | 
					
						
							|  |  |  |                 str << localsPrefixC << u; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2011-01-11 14:58:32 +01:00
										 |  |  |     str << blankSeparator << frameIndex; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     watchHandler()->beginCycle(); | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  |     postExtensionCommand("locals", arguments, 0, &CdbEngine::handleLocals, 0, QVariant(forNewStackFrame)); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::selectThread(int index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (index < 0 || index == threadsHandler()->currentThread()) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resetLocation(); | 
					
						
							|  |  |  |     const int newThreadId = threadsHandler()->threads().at(index).id; | 
					
						
							|  |  |  |     threadsHandler()->setCurrentThread(index); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QByteArray cmd = "~" + QByteArray::number(newThreadId) + " s"; | 
					
						
							|  |  |  |     postBuiltinCommand(cmd, 0, &CdbEngine::dummyHandler, CommandListStack); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | void CdbEngine::fetchDisassembler(DisassemblerAgent *agent) | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     QTC_ASSERT(m_accessible, return;) | 
					
						
							|  |  |  |     QByteArray cmd; | 
					
						
							|  |  |  |     ByteArrayInputStream str(cmd); | 
					
						
							|  |  |  |     str <<  "u " << hex << hexPrefixOn << agent->address() << " L40"; | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     const QVariant cookie = qVariantFromValue<DisassemblerAgent*>(agent); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     postBuiltinCommand(cmd, 0, &CdbEngine::handleDisassembler, 0, cookie); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parse: "00000000`77606060 cc              int     3"
 | 
					
						
							|  |  |  | void CdbEngine::handleDisassembler(const CdbBuiltinCommandPtr &command) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  |     QTC_ASSERT(qVariantCanConvert<DisassemblerAgent*>(command->cookie), return;) | 
					
						
							|  |  |  |     DisassemblerAgent *agent = qvariant_cast<DisassemblerAgent*>(command->cookie); | 
					
						
							| 
									
										
										
										
											2010-11-24 16:51:02 +01:00
										 |  |  |     DisassemblerLines disassemblerLines; | 
					
						
							|  |  |  |     foreach(const QByteArray &line, command->reply) | 
					
						
							|  |  |  |         disassemblerLines.appendLine(DisassemblerLine(QString::fromLatin1(line))); | 
					
						
							|  |  |  |     agent->setContents(disassemblerLines); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-17 15:34:18 +01:00
										 |  |  |     if (!m_accessible) { | 
					
						
							|  |  |  |         qWarning("Internal error: Attempt to read memory from inaccessible session: %s", Q_FUNC_INFO); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("CdbEngine::fetchMemory %llu bytes from 0x%llx", length, addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QByteArray args; | 
					
						
							|  |  |  |     ByteArrayInputStream str(args); | 
					
						
							|  |  |  |     str << addr << ' ' << length; | 
					
						
							|  |  |  |     const QVariant cookie = qVariantFromValue<MemoryViewCookie>(MemoryViewCookie(agent, editor, addr, length)); | 
					
						
							|  |  |  |     postExtensionCommand("memory", args, 0, &CdbEngine::handleMemory, 0, cookie); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::handleMemory(const CdbExtensionCommandPtr &command) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QTC_ASSERT(qVariantCanConvert<MemoryViewCookie>(command->cookie), return;) | 
					
						
							|  |  |  |     const MemoryViewCookie memViewCookie = qvariant_cast<MemoryViewCookie>(command->cookie); | 
					
						
							|  |  |  |     if (command->success) { | 
					
						
							|  |  |  |         const QByteArray data = QByteArray::fromBase64(command->reply); | 
					
						
							|  |  |  |         if (unsigned(data.size()) == memViewCookie.length) | 
					
						
							|  |  |  |             memViewCookie.agent->addLazyData(memViewCookie.editorToken, | 
					
						
							|  |  |  |                                              memViewCookie.address, data); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         showMessage(QString::fromLocal8Bit(command->errorMessage), LogError); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::reloadModules() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     postCommandSequence(CommandListModules); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::loadSymbols(const QString & /* moduleName */) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::loadAllSymbols() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::requestModuleSymbols(const QString &moduleName) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Q_UNUSED(moduleName) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::reloadRegisters() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     postCommandSequence(CommandListRegisters); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::reloadSourceFiles() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::reloadFullStack() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debug) | 
					
						
							|  |  |  |         qDebug("%s", Q_FUNC_INFO); | 
					
						
							|  |  |  |     postCommandSequence(CommandListStack); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::handlePid(const CdbExtensionCommandPtr &reply) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (reply->success) { | 
					
						
							|  |  |  |         m_inferiorPid = reply->reply.toUInt(); | 
					
						
							| 
									
										
										
										
											2011-01-13 14:27:52 +01:00
										 |  |  |         showMessage(QString::fromLatin1("Inferior pid: %1.").arg(m_inferiorPid), LogMisc); | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parse CDB gdbmi register syntax
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | static inline 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 += ')'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     reg.value = QString::fromAscii(gdbmiReg.findChild("value").data()); | 
					
						
							|  |  |  |     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; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |                 module.moduleName = QString::fromAscii(gdbmiModule.findChild("name").data()); | 
					
						
							|  |  |  |                 module.modulePath = QString::fromAscii(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)); | 
					
						
							|  |  |  |             registerHandler()->setRegisters(registers); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             showMessage(QString::fromLatin1("Parse error in registers response."), LogError); | 
					
						
							|  |  |  |             qWarning("Parse error in registers response:\n%s", reply->reply.constData()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }  else { | 
					
						
							|  |  |  |         showMessage(QString::fromLatin1("Failed to determine registers: %1"). | 
					
						
							|  |  |  |                     arg(QLatin1String(reply->errorMessage)), LogError); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     postCommandSequence(reply->commandSequence); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::handleLocals(const CdbExtensionCommandPtr &reply) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (reply->success) { | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |         QTC_ASSERT(root.isList(), return ; ) | 
					
						
							|  |  |  |         if (debugLocals) { | 
					
						
							|  |  |  |             qDebug() << root.toString(true, 4); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Courtesy of GDB engine
 | 
					
						
							|  |  |  |         foreach (const GdbMi &child, root.children()) { | 
					
						
							|  |  |  |             WatchData dummy; | 
					
						
							|  |  |  |             dummy.iname = child.findChild("iname").data(); | 
					
						
							|  |  |  |             dummy.name = QLatin1String(child.findChild("name").data()); | 
					
						
							|  |  |  |             parseWatchData(watchHandler()->expandedINames(), dummy, child, &watchData); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         watchHandler()->insertBulkData(watchData); | 
					
						
							|  |  |  |         watchHandler()->endCycle(); | 
					
						
							|  |  |  |         if (debugLocals) { | 
					
						
							|  |  |  |             QDebug nsp = qDebug().nospace(); | 
					
						
							|  |  |  |             nsp << "Obtained " << watchData.size() << " items:\n"; | 
					
						
							| 
									
										
										
										
											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
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-02-11 15:00:13 +01:00
										 |  |  |         const bool forNewStackFrame = reply->cookie.toBool(); | 
					
						
							|  |  |  |         if (forNewStackFrame) | 
					
						
							|  |  |  |             emit stackFrameCompleted(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         showMessage(QString::fromLatin1(reply->errorMessage), LogError); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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-01-25 18:35:31 +01:00
										 |  |  | unsigned CdbEngine::examineStopReason(const GdbMi &stopReason, | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |                                       QString *message, | 
					
						
							| 
									
										
										
										
											2011-01-17 15:18:13 +01:00
										 |  |  |                                       QString *exceptionBoxMessage) | 
					
						
							| 
									
										
										
										
											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)."). | 
					
						
							|  |  |  |                     arg(QString::fromAscii(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.
 | 
					
						
							|  |  |  |         BreakpointId id = 0; | 
					
						
							|  |  |  |         int number = 0; | 
					
						
							|  |  |  |         const GdbMi breakpointIdG = stopReason.findChild("breakpointId"); | 
					
						
							|  |  |  |         if (breakpointIdG.isValid()) { | 
					
						
							|  |  |  |             id = breakpointIdG.data().toULongLong(); | 
					
						
							|  |  |  |             if (id) | 
					
						
							|  |  |  |                 number = breakHandler()->response(id).number; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         if (id && breakHandler()->type(id) == Watchpoint) { | 
					
						
							|  |  |  |             *message = msgWatchpointTriggered(id, number, breakHandler()->address(id), QString::number(threadId)); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             *message = msgBreakpointTriggered(id, number, QString::number(threadId)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							|  |  |  |         // It is possible to hit on a startup trap while stepping (if something
 | 
					
						
							|  |  |  |         // pulls DLLs. Avoid showing a 'stopped' Message box.
 | 
					
						
							|  |  |  |         if (exception.exceptionCode == winExceptionStartupCompleteTrap) | 
					
						
							|  |  |  |             return StopNotifyStop; | 
					
						
							| 
									
										
										
										
											2011-01-17 15:18:13 +01:00
										 |  |  |         // WOW 64 breakpoint: Report in log and continue the first one,
 | 
					
						
							|  |  |  |         // subsequent ones are caused by interrupting the application.
 | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         if (exception.exceptionCode == winExceptionWX86Breakpoint) { | 
					
						
							| 
									
										
										
										
											2011-01-17 15:18:13 +01:00
										 |  |  |             if (m_wX86BreakpointCount++) { | 
					
						
							|  |  |  |                 *message = tr("Interrupted (%1)").arg(description); | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |                 rc |= StopReportStatusMessage|StopNotifyStop; | 
					
						
							| 
									
										
										
										
											2011-01-17 15:18:13 +01:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 *message = description; | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |                 rc |= StopIgnoreContinue|StopReportLog; | 
					
						
							| 
									
										
										
										
											2011-01-17 15:18:13 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-02-08 13:00:08 +01:00
										 |  |  |             return rc; | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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
										 |  |  |     GdbMi stopReason; | 
					
						
							|  |  |  |     stopReason.fromString(messageBA); | 
					
						
							|  |  |  |     int forcedThreadId = -1; | 
					
						
							|  |  |  |     const unsigned stopFlags = examineStopReason(stopReason, &message, &exceptionBoxMessage); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |     // Ignore things like WOW64
 | 
					
						
							|  |  |  |     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) { | 
					
						
							|  |  |  |         if (state() == InferiorStopRequested) { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorStopOk") | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |             notifyInferiorStopOk(); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2011-01-19 10:25:32 +01:00
										 |  |  |             STATE_DEBUG(state(), Q_FUNC_INFO, __LINE__, "notifyInferiorSpontaneousStop") | 
					
						
							| 
									
										
										
										
											2011-01-12 13:49:51 +01:00
										 |  |  |             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 { | 
					
						
							|  |  |  |                 showMessage(QString::fromAscii(stopReason.findChild("stackerror").data()), LogError); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							|  |  |  |             showMessage(QString::fromAscii(stopReason.findChild("threaderror").data()), LogError); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // 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") { | 
					
						
							|  |  |  |         showStatusMessage(QString::fromAscii(message),  5000); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (what == "session_accessible") { | 
					
						
							|  |  |  |         if (!m_accessible) { | 
					
						
							|  |  |  |             m_accessible = true; | 
					
						
							|  |  |  |             handleSessionAccessible(message.toULong()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (what == "session_inaccessible") { | 
					
						
							|  |  |  |         if (m_accessible) { | 
					
						
							|  |  |  |             m_accessible = false; | 
					
						
							|  |  |  |             handleSessionInaccessible(message.toULong()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (what == "session_idle") { | 
					
						
							|  |  |  |         handleSessionIdle(message); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (what == "exception") { | 
					
						
							|  |  |  |         WinException exception; | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |         QTC_ASSERT(tokenEndPos != -1, return) | 
					
						
							|  |  |  |         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); | 
					
						
							|  |  |  |         QTC_ASSERT(remainingChunksEndPos != -1, return) | 
					
						
							|  |  |  |         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); | 
					
						
							|  |  |  |         QTC_ASSERT(whatEndPos != -1, return) | 
					
						
							|  |  |  |         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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CdbEngine::acceptsBreakpoint(BreakpointId id) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return DebuggerEngine::isCppBreakpoint(breakHandler()->breakpointData(id)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | void CdbEngine::attemptBreakpointSynchronization() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											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?
 | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |     foreach (BreakpointId 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; | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |     const BreakpointIds ids = handler->engineBreakpointIds(this); | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |     foreach (BreakpointId id, ids) { | 
					
						
							|  |  |  |         switch (handler->state(id)) { | 
					
						
							|  |  |  |         case BreakpointInsertRequested: | 
					
						
							|  |  |  |         case BreakpointRemoveRequested: | 
					
						
							|  |  |  |         case BreakpointChangeRequested: | 
					
						
							|  |  |  |             changed = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case BreakpointInserted: { | 
					
						
							|  |  |  |             // Collect the new modules matching the files.
 | 
					
						
							|  |  |  |             // In the future, that information should be obtained from the build system.
 | 
					
						
							|  |  |  |             const BreakpointParameters &data = handler->breakpointData(id); | 
					
						
							|  |  |  |             if (data.type == BreakpointByFileAndLine && !data.module.isEmpty()) | 
					
						
							|  |  |  |                 m_fileNameModuleHash.insert(data.fileName, data.module); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |     foreach (BreakpointId 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); | 
					
						
							|  |  |  |         // 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-02-03 16:26:23 +01:00
										 |  |  |             postCommand(cdbAddBreakpointCommand(parameters, id, false), 0); | 
					
						
							|  |  |  |             if (!parameters.enabled) | 
					
						
							|  |  |  |                 postCommand("bd " + QByteArray::number(id), 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; | 
					
						
							|  |  |  |             if (debugBreakpoints) | 
					
						
							|  |  |  |                 qDebug("Adding %llu %s\n", id, 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-02-03 16:26:23 +01:00
										 |  |  |             if (parameters.enabled != handler->response(id).enabled) { | 
					
						
							|  |  |  |                 // Change enabled/disabled breakpoints without triggering update.
 | 
					
						
							|  |  |  |                 postCommand((parameters.enabled ? "be " : "bd ") + QByteArray::number(id), 0); | 
					
						
							|  |  |  |                 response.pending = false; | 
					
						
							|  |  |  |                 response.enabled = parameters.enabled; | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |                 postCommand("bc " + QByteArray::number(id), 0); | 
					
						
							|  |  |  |                 postCommand(cdbAddBreakpointCommand(parameters, id, false), 0); | 
					
						
							|  |  |  |                 m_pendingBreakpointMap.insert(id, response); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             handler->notifyBreakpointChangeOk(id); | 
					
						
							|  |  |  |             if (debugBreakpoints) | 
					
						
							|  |  |  |                 qDebug("Changing %llu %s\n", id, qPrintable(response.toString())); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |         case BreakpointRemoveRequested: | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  |             postCommand("bc " + QByteArray::number(id), 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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString CdbEngine::normalizeFileName(const QString &f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QMap<QString, QString>::const_iterator it = m_normalizedFileCache.constFind(f); | 
					
						
							|  |  |  |     if (it != m_normalizedFileCache.constEnd()) | 
					
						
							|  |  |  |         return it.value(); | 
					
						
							|  |  |  |     const QString winF = QDir::toNativeSeparators(f); | 
					
						
							|  |  |  | #ifdef Q_OS_WIN
 | 
					
						
							| 
									
										
										
										
											2010-11-18 16:50:44 +01:00
										 |  |  |     QString normalized = winNormalizeFileName(winF); | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     QString normalized = winF; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     if (normalized.isEmpty()) { // At least upper case drive letter
 | 
					
						
							|  |  |  |         normalized = winF; | 
					
						
							|  |  |  |         if (normalized.size() > 2 && normalized.at(1) == QLatin1Char(':')) | 
					
						
							|  |  |  |             normalized[0] = normalized.at(0).toUpper(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     m_normalizedFileCache.insert(f, normalized); | 
					
						
							|  |  |  |     return normalized; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							| 
									
										
										
										
											2010-12-01 16:37:34 +01:00
										 |  |  |             frame.usable = QFileInfo(frame.file).isFile(); | 
					
						
							| 
									
										
										
										
											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+"))) { | 
					
						
							|  |  |  |             showMessage(QString::fromAscii("Step into: Call instruction hit, performing additional step..."), LogMisc); | 
					
						
							|  |  |  |             return ParseStackStepInto; | 
					
						
							| 
									
										
										
										
											2010-11-22 13:50:40 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-01-25 18:35:31 +01:00
										 |  |  |         if (hasFile) { | 
					
						
							|  |  |  |             frames[i].file = QDir::cleanPath(normalizeFileName(frames.at(i).file)); | 
					
						
							|  |  |  |             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 { | 
					
						
							|  |  |  |         showMessage(command->errorMessage, LogError); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  |         QTC_ASSERT(threadsHandler()->currentThread() >= 0,  return; ) | 
					
						
							|  |  |  |         postExtensionCommand("registers", QByteArray(), 0, &CdbEngine::handleRegisters, mask & ~CommandListRegisters); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (mask & CommandListModules) { | 
					
						
							|  |  |  |         postExtensionCommand("modules", QByteArray(), 0, &CdbEngine::handleModules, mask & ~CommandListModules); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  |             message = QString::fromAscii(reply->errorMessage); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Should be "namespace::QWidget:0x555"
 | 
					
						
							|  |  |  |         QString watchExp = QString::fromAscii(reply->reply); | 
					
						
							|  |  |  |         const int sepPos = watchExp.lastIndexOf(QLatin1Char(':')); | 
					
						
							|  |  |  |         if (sepPos == -1) { | 
					
						
							|  |  |  |             message = QString::fromAscii("Invalid output: %1").arg(watchExp); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // 0x000 -> nothing found
 | 
					
						
							|  |  |  |         if (!watchExp.mid(sepPos + 1).toULongLong(0, 0)) { | 
					
						
							|  |  |  |             message = QString::fromAscii("No widget could be found at %1, %2.").arg(m_watchPointX).arg(m_watchPointY); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Turn into watch expression: "*(namespace::QWidget*)0x555"
 | 
					
						
							|  |  |  |         watchExp.replace(sepPos, 1, QLatin1String("*)")); | 
					
						
							|  |  |  |         watchExp.insert(0, QLatin1String("*(")); | 
					
						
							|  |  |  |         watchHandler()->watchExpression(watchExp); | 
					
						
							|  |  |  |         success = true; | 
					
						
							|  |  |  |     } while (false); | 
					
						
							|  |  |  |     if (!success) | 
					
						
							|  |  |  |         showMessage(message, LogWarning); | 
					
						
							|  |  |  |     m_watchPointX = m_watchPointY = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-03 16:26:23 +01:00
										 |  |  | static inline void formatCdbBreakPointResponse(BreakpointId id, const BreakpointResponse &r, | 
					
						
							|  |  |  |                                                   QTextStream &str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     str << "Obtained breakpoint " << id << " (#" << r.number << ')'; | 
					
						
							|  |  |  |     if (r.pending) { | 
					
						
							|  |  |  |         str << ", pending"; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         str.setIntegerBase(16); | 
					
						
							|  |  |  |         str << ", at 0x" << r.address; | 
					
						
							|  |  |  |         str.setIntegerBase(10); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!r.enabled) | 
					
						
							|  |  |  |         str << ", disabled"; | 
					
						
							|  |  |  |     if (!r.module.isEmpty()) | 
					
						
							|  |  |  |         str << ", module: '" << r.module << '\''; | 
					
						
							|  |  |  |     str << '\n'; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::handleBreakPoints(const CdbExtensionCommandPtr &reply) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (debugBreakpoints) | 
					
						
							|  |  |  |         qDebug("CdbEngine::handleBreakPoints: sucess=%d: %s", reply->success, reply->reply.constData()); | 
					
						
							|  |  |  |     if (!reply->success) { | 
					
						
							|  |  |  |         showMessage(QString::fromAscii(reply->errorMessage), LogError); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     GdbMi value; | 
					
						
							|  |  |  |     value.fromString(reply->reply); | 
					
						
							|  |  |  |     if (value.type() != GdbMi::List) { | 
					
						
							|  |  |  |         showMessage(QString::fromAscii("Unabled to parse breakpoints reply"), LogError); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     handleBreakPoints(value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::handleBreakPoints(const GdbMi &value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Report all obtained parameters back. Note that not all parameters are reported
 | 
					
						
							|  |  |  |     // back, so, match by id and complete
 | 
					
						
							|  |  |  |     if (debugBreakpoints) | 
					
						
							|  |  |  |         qDebug("\nCdbEngine::handleBreakPoints with %d", value.childCount()); | 
					
						
							|  |  |  |     QString message; | 
					
						
							|  |  |  |     QTextStream str(&message); | 
					
						
							|  |  |  |     BreakHandler *handler = breakHandler(); | 
					
						
							|  |  |  |     foreach (const GdbMi &breakPointG, value.children()) { | 
					
						
							|  |  |  |         BreakpointResponse reportedResponse; | 
					
						
							|  |  |  |         const BreakpointId id = parseBreakPoint(breakPointG, &reportedResponse); | 
					
						
							|  |  |  |         if (debugBreakpoints) | 
					
						
							|  |  |  |             qDebug("  Parsed %llu: pending=%d %s\n", id, reportedResponse.pending, | 
					
						
							|  |  |  |                    qPrintable(reportedResponse.toString())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!reportedResponse.pending) { | 
					
						
							|  |  |  |             const PendingBreakPointMap::iterator it = m_pendingBreakpointMap.find(id); | 
					
						
							|  |  |  |             if (it != m_pendingBreakpointMap.end()) { | 
					
						
							|  |  |  |                 // Complete the response and set on handler.
 | 
					
						
							|  |  |  |                 BreakpointResponse ¤tResponse = it.value(); | 
					
						
							|  |  |  |                 currentResponse.number = reportedResponse.number; | 
					
						
							|  |  |  |                 currentResponse.address = reportedResponse.address; | 
					
						
							|  |  |  |                 currentResponse.module = reportedResponse.module; | 
					
						
							|  |  |  |                 currentResponse.pending = reportedResponse.pending; | 
					
						
							|  |  |  |                 currentResponse.enabled = reportedResponse.enabled; | 
					
						
							|  |  |  |                 formatCdbBreakPointResponse(id, currentResponse, str); | 
					
						
							|  |  |  |                 handler->setResponse(id, currentResponse); | 
					
						
							|  |  |  |                 m_pendingBreakpointMap.erase(it); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } // not pending reported
 | 
					
						
							|  |  |  |     } // foreach
 | 
					
						
							|  |  |  |     if (m_pendingBreakpointMap.empty()) { | 
					
						
							|  |  |  |         str << QLatin1String("All breakpoints have been resolved.\n"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         str << QString::fromLatin1("%1 breakpoint(s) pending...\n").arg(m_pendingBreakpointMap.size()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     showMessage(message, LogMisc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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'."). | 
					
						
							|  |  |  |                     arg(QString::fromAscii(stateName(state()))), LogWarning); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CdbEngine::postWidgetAtCommand() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QByteArray arguments = QByteArray::number(m_watchPointX); | 
					
						
							|  |  |  |     arguments.append(' '); | 
					
						
							|  |  |  |     arguments.append(QByteArray::number(m_watchPointY)); | 
					
						
							|  |  |  |     postExtensionCommand("widgetat", arguments, 0, &CdbEngine::handleWidgetAt, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 11:06:15 +01:00
										 |  |  | } // namespace Internal
 | 
					
						
							| 
									
										
										
										
											2010-11-18 13:51:15 +01:00
										 |  |  | } // namespace Debugger
 |