Files
qt-creator/src/plugins/debugger/gdb/gdbengine.h

457 lines
16 KiB
C
Raw Normal View History

/**************************************************************************
2008-12-02 12:01:29 +01:00
**
** This file is part of Qt Creator
**
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
2008-12-02 12:01:29 +01:00
**
** Contact: Nokia Corporation (qt-info@nokia.com)
2008-12-02 12:01:29 +01:00
**
** Commercial Usage
**
** Licensees holding valid Qt Commercial licenses may use this file in
** accordance with the Qt Commercial License Agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and Nokia.
**
** 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.
**
** If you are unsure which license is appropriate for your use, please
2009-08-14 09:30:56 +02:00
** contact the sales department at http://qt.nokia.com/contact.
2008-12-02 12:01:29 +01:00
**
**************************************************************************/
2008-12-02 15:08:31 +01:00
2008-12-02 12:01:29 +01:00
#ifndef DEBUGGER_GDBENGINE_H
#define DEBUGGER_GDBENGINE_H
#include "idebuggerengine.h"
2009-09-10 13:09:42 +02:00
#include "debuggermanager.h" // only for StartParameters
#include "gdbmi.h"
#include "watchutils.h"
2009-09-21 11:09:38 +02:00
#include "outputcollector.h"
2008-12-02 12:01:29 +01:00
#include <QtCore/QByteArray>
#include <QtCore/QHash>
#include <QtCore/QMap>
#include <QtCore/QObject>
#include <QtCore/QProcess>
#include <QtCore/QPoint>
2009-09-21 11:09:38 +02:00
#include <QtCore/QSet>
#include <QtCore/QTextCodec>
#include <QtCore/QTime>
2008-12-02 12:01:29 +01:00
#include <QtCore/QVariant>
QT_BEGIN_NAMESPACE
class QAction;
class QAbstractItemModel;
class QWidget;
QT_END_NAMESPACE
namespace Debugger {
namespace Internal {
2009-09-21 11:09:38 +02:00
class AbstractGdbAdapter;
2008-12-02 12:01:29 +01:00
class DebuggerManager;
class IDebuggerManagerAccessForEngines;
class GdbResultRecord;
class GdbMi;
class WatchData;
class BreakpointData;
enum DebuggingHelperState
2008-12-02 12:01:29 +01:00
{
DebuggingHelperUninitialized,
DebuggingHelperLoadTried,
DebuggingHelperAvailable,
DebuggingHelperUnavailable,
2008-12-02 12:01:29 +01:00
};
class GdbEngine : public IDebuggerEngine
{
Q_OBJECT
public:
2009-09-21 11:09:38 +02:00
explicit GdbEngine(DebuggerManager *parent);
2008-12-02 12:01:29 +01:00
~GdbEngine();
2009-09-21 11:09:38 +02:00
void setGdbAdapter(AbstractGdbAdapter *adapter);
2008-12-02 12:01:29 +01:00
signals:
void gdbInputAvailable(int channel, const QString &msg);
void gdbOutputAvailable(int channel, const QString &msg);
void applicationOutputAvailable(const QString &output);
2008-12-02 12:01:29 +01:00
private:
friend class PlainGdbAdapter;
friend class TrkGdbAdapter;
2009-09-11 12:30:53 +02:00
const DebuggerStartParameters &startParameters() const
2009-09-17 14:17:43 +02:00
{ return *m_startParameters; }
2008-12-02 12:01:29 +01:00
//
// IDebuggerEngine implementation
//
void stepExec();
void stepOutExec();
void nextExec();
void stepIExec();
void nextIExec();
void shutdown();
void setToolTipExpression(const QPoint &mousePos, TextEditor::ITextEditor *editor, int cursorPos);
2009-09-17 14:38:15 +02:00
void startDebugger(const DebuggerStartParametersPtr &sp);
2008-12-02 12:01:29 +01:00
void exitDebugger();
2009-09-16 13:43:03 +02:00
void exitDebugger2();
void detachDebugger();
2008-12-02 12:01:29 +01:00
void continueInferior();
void interruptInferior();
void runToLineExec(const QString &fileName, int lineNumber);
void runToFunctionExec(const QString &functionName);
void jumpToLineExec(const QString &fileName, int lineNumber);
void activateFrame(int index);
void selectThread(int index);
Q_SLOT void attemptBreakpointSynchronization();
void assignValueInDebugger(const QString &expr, const QString &value);
void executeDebuggerCommand(const QString & command);
void watchPoint(const QPoint &);
2008-12-02 12:01:29 +01:00
void loadSymbols(const QString &moduleName);
void loadAllSymbols();
virtual QList<Symbol> moduleSymbols(const QString &moduleName);
2008-12-02 12:01:29 +01:00
void fetchMemory(MemoryViewAgent *agent, quint64 addr, quint64 length);
void handleFetchMemory(const GdbResultRecord &record, const QVariant &cookie);
void fetchDisassembler(DisassemblerViewAgent *agent,
const StackFrame &frame);
void fetchDisassemblerByAddress(DisassemblerViewAgent *agent,
bool useMixedMode);
void handleFetchDisassemblerByLine(const GdbResultRecord &record,
const QVariant &cookie);
void handleFetchDisassemblerByAddress1(const GdbResultRecord &record,
const QVariant &cookie);
void handleFetchDisassemblerByAddress0(const GdbResultRecord &record,
const QVariant &cookie);
Q_SLOT void setDebugDebuggingHelpers(const QVariant &on);
Q_SLOT void setUseDebuggingHelpers(const QVariant &on);
2008-12-02 12:01:29 +01:00
//
// Own stuff
//
2008-12-02 12:01:29 +01:00
int currentFrame() const;
bool supportsThreads() const;
2009-09-17 16:51:27 +02:00
void gotoLocation(const StackFrame &frame, bool setLocationMarker);
2008-12-02 12:01:29 +01:00
void initializeConnections();
void initializeVariables();
2008-12-02 12:01:29 +01:00
QString fullName(const QString &fileName);
// get one usable name out of these, try full names first
QString fullName(const QStringList &candidates);
void handleResult(const GdbResultRecord &, int type, const QVariant &);
public: // otherwise the Qt flag macros are unhappy
enum GdbCommandFlag {
NoFlags = 0,
NeedsStop = 1,
Discardable = 2,
RebuildModel = 4,
WatchUpdate = Discardable | RebuildModel,
EmbedToken = 8
};
Q_DECLARE_FLAGS(GdbCommandFlags, GdbCommandFlag)
2009-09-11 12:30:53 +02:00
private:
2009-09-21 11:09:38 +02:00
typedef void (GdbEngine::*GdbCommandCallback)
(const GdbResultRecord &record, const QVariant &cookie);
typedef void (AbstractGdbAdapter::*AdapterCallback)
(const GdbResultRecord &record, const QVariant &cookie);
struct GdbCommand
{
2009-09-21 11:09:38 +02:00
GdbCommand()
: flags(0), callback(0), adapterCallback(0), callbackName(0)
{}
int flags;
GdbCommandCallback callback;
2009-09-21 11:09:38 +02:00
AdapterCallback adapterCallback;
const char *callbackName;
QString command;
QVariant cookie;
QTime postTime;
};
2008-12-02 12:01:29 +01:00
// type and cookie are sender-internal data, opaque for the "event
// queue". resultNeeded == true increments m_pendingResults on
// send and decrements on receipt, effectively preventing
// watch model updates before everything is finished.
2009-09-21 11:09:38 +02:00
void flushCommand(const GdbCommand &cmd);
void postCommand(const QString &command,
GdbCommandFlags flags,
GdbCommandCallback callback = 0,
const char *callbackName = 0,
const QVariant &cookie = QVariant());
void postCommand(const QString &command,
GdbCommandCallback callback = 0,
const char *callbackName = 0,
const QVariant &cookie = QVariant());
2009-09-21 11:09:38 +02:00
void postCommand(const QString &command,
AdapterCallback callback,
const char *callbackName,
const QVariant &cookie = QVariant());
void postCommand(const QString &command,
GdbCommandFlags flags,
AdapterCallback callback,
const char *callbackName,
const QVariant &cookie = QVariant());
2009-09-21 11:09:38 +02:00
void postCommandHelper(const GdbCommand &cmd);
2008-12-02 12:01:29 +01:00
void setTokenBarrier();
void updateLocals();
private slots:
void gdbProcError(QProcess::ProcessError error);
void readGdbStandardOutput();
void readGdbStandardError();
void readUploadStandardOutput();
void readUploadStandardError();
void readDebugeeOutput(const QByteArray &data);
void uploadProcError(QProcess::ProcessError error);
void emitStartFailed();
2008-12-02 12:01:29 +01:00
2009-09-21 11:09:38 +02:00
void handleAdapterStarted();
void handleAdapterStartFailed(const QString &msg);
void handleInferiorPrepared();
void handleInferiorPreparationFailed(const QString &msg);
void handleInferiorStarted();
void handleInferiorStartFailed(const QString &msg);
void handleInferiorShutDown();
void handleInferiorShutdownFailed(const QString &msg);
void handleAdapterShutDown();
void handleAdapterShutdownFailed(const QString &msg);
2008-12-02 12:01:29 +01:00
private:
int terminationIndex(const QByteArray &buffer, int &length);
void handleResponse(const QByteArray &buff);
void handleStart(const GdbResultRecord &response, const QVariant &);
void handleAttach(const GdbResultRecord &, const QVariant &);
2009-02-18 15:27:56 +01:00
void handleAqcuiredInferior();
2008-12-02 12:01:29 +01:00
void handleAsyncOutput(const GdbMi &data);
2009-09-21 11:09:38 +02:00
void handleStop1(const GdbResultRecord &, const QVariant &cookie);
void handleStop2(const GdbResultRecord &, const QVariant &cookie);
void handleStop2(const GdbMi &data);
2008-12-02 12:01:29 +01:00
void handleResultRecord(const GdbResultRecord &response);
void handleFileExecAndSymbols(const GdbResultRecord &response, const QVariant &);
2009-07-21 14:37:51 +02:00
void handleExecContinue(const GdbResultRecord &response, const QVariant &);
2009-09-21 11:09:38 +02:00
//void handleExecRun(const GdbResultRecord &response, const QVariant &);
void handleExecJumpToLine(const GdbResultRecord &response, const QVariant &);
void handleExecRunToFunction(const GdbResultRecord &response, const QVariant &);
void handleInfoShared(const GdbResultRecord &response, const QVariant &);
void handleInfoProc(const GdbResultRecord &response, const QVariant &);
void handleInfoThreads(const GdbResultRecord &response, const QVariant &);
void handleShowVersion(const GdbResultRecord &response, const QVariant &);
void handleQueryPwd(const GdbResultRecord &response, const QVariant &);
void handleQuerySources(const GdbResultRecord &response, const QVariant &);
void handleTargetCore(const GdbResultRecord &, const QVariant &);
void handleExit(const GdbResultRecord &, const QVariant &);
2009-09-16 13:43:03 +02:00
void handleExitHelper(const GdbResultRecord &, const QVariant &);
void handleSetTargetAsync(const GdbResultRecord &, const QVariant &);
void handleTargetRemote(const GdbResultRecord &, const QVariant &);
void handleWatchPoint(const GdbResultRecord &, const QVariant &);
2009-02-16 13:29:57 +01:00
void debugMessage(const QString &msg);
bool showToolTip();
2008-12-02 12:01:29 +01:00
// Convenience
DebuggerManager *manager() { return m_manager; }
void showStatusMessage(const QString &msg, int timeout = -1)
{ m_manager->showStatusMessage(msg, timeout); }
int status() const { return m_manager->status(); }
QMainWindow *mainWindow() const { return m_manager->mainWindow(); }
2009-09-17 14:17:43 +02:00
DebuggerStartMode startMode() const { return m_startParameters->startMode; }
qint64 inferiorPid() const { return m_manager->inferiorPid(); }
void handleChildren(const WatchData &parent, const GdbMi &child,
QList<WatchData> *insertions);
const bool m_dumperInjectionLoad;
QTextCodec *m_outputCodec;
QTextCodec::ConverterState m_outputCodecState;
2008-12-02 12:01:29 +01:00
QByteArray m_inbuffer;
AbstractGdbAdapter *m_gdbAdapter;
QProcess m_uploadProc;
2008-12-02 12:01:29 +01:00
QHash<int, GdbCommand> m_cookieForToken;
2008-12-02 12:01:29 +01:00
QHash<int, QByteArray> m_customOutputForToken;
QByteArray m_pendingConsoleStreamOutput;
QByteArray m_pendingTargetStreamOutput;
QByteArray m_pendingLogStreamOutput;
// contains the first token number for the current round
// of evaluation. Responses with older tokens are considers
// out of date and discarded.
int m_oldestAcceptableToken;
int m_gdbVersion; // 6.8.0 is 680
2009-02-23 16:06:23 +01:00
int m_gdbBuildVersion; // MAC only?
2008-12-02 12:01:29 +01:00
// awful hack to keep track of used files
QMap<QString, QString> m_shortToFullName;
QMap<QString, QString> m_fullToShortName;
2008-12-02 12:01:29 +01:00
//
// Breakpoint specific stuff
//
void handleBreakList(const GdbResultRecord &record, const QVariant &);
2008-12-02 12:01:29 +01:00
void handleBreakList(const GdbMi &table);
void handleBreakIgnore(const GdbResultRecord &record, const QVariant &cookie);
void handleBreakInsert(const GdbResultRecord &record, const QVariant &cookie);
void handleBreakInsert1(const GdbResultRecord &record, const QVariant &cookie);
void handleBreakCondition(const GdbResultRecord &record, const QVariant &cookie);
void handleBreakInfo(const GdbResultRecord &record, const QVariant &cookie);
2008-12-02 12:01:29 +01:00
void extractDataFromInfoBreak(const QString &output, BreakpointData *data);
void breakpointDataFromOutput(BreakpointData *data, const GdbMi &bkpt);
void sendInsertBreakpoint(int index);
//
// Modules specific stuff
//
void reloadModules();
void handleModulesList(const GdbResultRecord &record, const QVariant &);
2008-12-02 12:01:29 +01:00
//
// Register specific stuff
//
Q_SLOT void reloadRegisters();
void setRegisterValue(int nr, const QString &value);
void handleRegisterListNames(const GdbResultRecord &record, const QVariant &);
void handleRegisterListValues(const GdbResultRecord &record, const QVariant &);
2008-12-02 12:01:29 +01:00
//
// Source file specific stuff
//
void reloadSourceFiles();
2008-12-02 12:01:29 +01:00
//
// Stack specific stuff
//
void handleStackListFrames(const GdbResultRecord &record, const QVariant &cookie);
void handleStackSelectThread(const GdbResultRecord &, const QVariant &);
void handleStackListThreads(const GdbResultRecord &record, const QVariant &cookie);
Q_SLOT void reloadStack();
Q_SLOT void reloadFullStack();
2008-12-02 12:01:29 +01:00
//
// Tooltip specific stuff
//
void sendToolTipCommand(const QString &command, const QString &cookie);
//
// Watch specific stuff
//
// FIXME: BaseClass. called to improve situation for a watch item
void updateSubItem(const WatchData &data);
void updateWatchData(const WatchData &data);
Q_SLOT void updateWatchDataHelper(const WatchData &data);
void rebuildModel();
2008-12-02 12:01:29 +01:00
void insertData(const WatchData &data);
void sendWatchParameters(const QByteArray &params0);
void createGdbVariable(const WatchData &data);
void maybeHandleInferiorPidChanged(const QString &pid);
void tryLoadDebuggingHelpers();
void tryQueryDebuggingHelpers();
Q_SLOT void recheckDebuggingHelperAvailability();
void runDebuggingHelper(const WatchData &data, bool dumpChildren);
void runDirectDebuggingHelper(const WatchData &data, bool dumpChildren);
bool hasDebuggingHelperForType(const QString &type) const;
2008-12-02 12:01:29 +01:00
void handleVarListChildren(const GdbResultRecord &record,
const QVariant &cookie);
2008-12-02 12:01:29 +01:00
void handleVarCreate(const GdbResultRecord &record,
const QVariant &cookie);
void handleVarAssign(const GdbResultRecord &, const QVariant &);
2008-12-02 12:01:29 +01:00
void handleEvaluateExpression(const GdbResultRecord &record,
const QVariant &cookie);
//void handleToolTip(const GdbResultRecord &record,
// const QVariant &cookie);
void handleQueryDebuggingHelper(const GdbResultRecord &record, const QVariant &);
void handleDebuggingHelperValue1(const GdbResultRecord &record,
const QVariant &cookie);
void handleDebuggingHelperValue2(const GdbResultRecord &record,
const QVariant &cookie);
void handleDebuggingHelperValue3(const GdbResultRecord &record,
const QVariant &cookie);
void handleDebuggingHelperEditValue(const GdbResultRecord &record);
void handleDebuggingHelperSetup(const GdbResultRecord &record, const QVariant &);
void handleStackListLocals(const GdbResultRecord &record, const QVariant &);
void handleStackListArguments(const GdbResultRecord &record, const QVariant &);
2008-12-02 12:01:29 +01:00
void handleVarListChildrenHelper(const GdbMi &child,
const WatchData &parent);
void setWatchDataType(WatchData &data, const GdbMi &mi);
void setWatchDataDisplayedType(WatchData &data, const GdbMi &mi);
2008-12-03 16:51:56 +01:00
void setLocals(const QList<GdbMi> &locals);
bool startModeAllowsDumpers() const;
QString parseDisassembler(const GdbMi &lines);
2008-12-02 12:01:29 +01:00
int m_pendingRequests;
QSet<QString> m_processedNames;
2008-12-02 12:01:29 +01:00
QtDumperHelper m_dumperHelper;
2008-12-02 12:01:29 +01:00
DebuggingHelperState m_debuggingHelperState;
2008-12-02 12:01:29 +01:00
QList<GdbMi> m_currentFunctionArgs;
QString m_currentFrame;
QMap<QString, QString> m_varToType;
2009-09-21 11:09:38 +02:00
typedef void (GdbEngine::*Continuation)();
// function called after all previous responses have been received
Continuation m_continuationAfterDone;
void handleInitialBreakpointsSet();
bool m_waitingForFirstBreakpointToBeHit;
bool m_modulesListOutdated;
QList<GdbCommand> m_commandsToRunOnTemporaryBreak;
2009-02-16 18:28:15 +01:00
DebuggerManager * const m_manager;
IDebuggerManagerAccessForEngines * const qq;
2009-09-17 14:17:43 +02:00
DebuggerStartParametersPtr m_startParameters;
// make sure to re-initialize new members in initializeVariables();
2009-09-21 11:09:38 +02:00
public:
OutputCollector m_outputCollector;
2008-12-02 12:01:29 +01:00
};
} // namespace Internal
} // namespace Debugger
Q_DECLARE_OPERATORS_FOR_FLAGS(Debugger::Internal::GdbEngine::GdbCommandFlags)
2008-12-02 12:01:29 +01:00
#endif // DEBUGGER_GDBENGINE_H