diff --git a/src/plugins/debugger/debuggeractions.h b/src/plugins/debugger/debuggeractions.h index b8a60faba9f8d65f34f1ba20d091c1c75f1a709e..364cb65176dc7120a27b60136766027c01c8a856 100644 --- a/src/plugins/debugger/debuggeractions.h +++ b/src/plugins/debugger/debuggeractions.h @@ -154,6 +154,7 @@ struct DebuggerManagerActions QAction *runToFunctionAction; QAction *jumpToLineAction1; // in the Debug menu QAction *jumpToLineAction2; // in the text editor context menu + QAction *returnFromFunctionAction; QAction *nextAction; QAction *snapshotAction; QAction *watchAction1; // in the Debug menu diff --git a/src/plugins/debugger/debuggerconstants.h b/src/plugins/debugger/debuggerconstants.h index 9b6cdb636c720979a31aea0616f700391b82baee..94c3aef9872ca660e59039d44a0b6fedf07bc968 100644 --- a/src/plugins/debugger/debuggerconstants.h +++ b/src/plugins/debugger/debuggerconstants.h @@ -53,9 +53,11 @@ const char * const C_GDBDEBUGGER = "Gdb Debugger"; const char * const GDBRUNNING = "Gdb.Running"; const char * const DEBUGGER_COMMON_SETTINGS_ID = "A.Common"; -const char * const DEBUGGER_COMMON_SETTINGS_NAME = QT_TRANSLATE_NOOP("Debugger", "Common"); +const char * const DEBUGGER_COMMON_SETTINGS_NAME = + QT_TRANSLATE_NOOP("Debugger", "Common"); const char * const DEBUGGER_SETTINGS_CATEGORY = "O.Debugger"; -const char * const DEBUGGER_SETTINGS_TR_CATEGORY = QT_TRANSLATE_NOOP("Debugger", "Debugger"); +const char * const DEBUGGER_SETTINGS_TR_CATEGORY = + QT_TRANSLATE_NOOP("Debugger", "Debugger"); namespace Internal { enum { debug = 0 }; @@ -65,9 +67,10 @@ namespace Internal { const char * const LD_PRELOAD_ENV_VAR = "LD_PRELOAD"; #endif -} +} // namespace Internal } // namespace Constants + enum DebuggerState { DebuggerNotReady, // Debugger not started @@ -121,6 +124,7 @@ enum DebuggerCapabilities ReloadModuleCapability = 0x80, ReloadModuleSymbolsCapability = 0x100, BreakOnThrowAndCatchCapability = 0x200, + ReturnFromFunctionCapability = 0x400, }; enum LogChannel diff --git a/src/plugins/debugger/debuggermanager.cpp b/src/plugins/debugger/debuggermanager.cpp index 06fad14ba01e86788a830746074a0a831eb57f9a..eb2936ff98bd4054e1c0329e441800777dfb5063 100644 --- a/src/plugins/debugger/debuggermanager.cpp +++ b/src/plugins/debugger/debuggermanager.cpp @@ -485,7 +485,11 @@ void DebuggerManager::init() d->m_actions.runToLineAction1 = new QAction(tr("Run to Line"), this); d->m_actions.runToLineAction2 = new QAction(tr("Run to Line"), this); - d->m_actions.runToFunctionAction = new QAction(tr("Run to Outermost Function"), this); + d->m_actions.runToFunctionAction = + new QAction(tr("Run to Outermost Function"), this); + + d->m_actions.returnFromFunctionAction = + new QAction(tr("Immediately Return From Inner Function"), this); d->m_actions.jumpToLineAction1 = new QAction(tr("Jump to Line"), this); d->m_actions.jumpToLineAction2 = new QAction(tr("Jump to Line"), this); @@ -524,6 +528,8 @@ void DebuggerManager::init() this, SLOT(jumpToLineExec())); connect(d->m_actions.jumpToLineAction2, SIGNAL(triggered()), this, SLOT(jumpToLineExec())); + connect(d->m_actions.returnFromFunctionAction, SIGNAL(triggered()), + this, SLOT(returnExec())); connect(d->m_actions.watchAction1, SIGNAL(triggered()), this, SLOT(addToWatchWindow())); connect(d->m_actions.watchAction2, SIGNAL(triggered()), @@ -1185,6 +1191,13 @@ void DebuggerManager::nextExec() d->m_engine->nextExec(); } +void DebuggerManager::returnExec() +{ + QTC_ASSERT(d->m_engine, return); + resetLocation(); + d->m_engine->returnExec(); +} + void DebuggerManager::watchPoint() { if (QAction *action = qobject_cast<QAction *>(sender())) @@ -1742,8 +1755,8 @@ void DebuggerManager::setState(DebuggerState state, bool forced) d->m_actions.watchAction1->setEnabled(true); d->m_actions.watchAction2->setEnabled(true); d->m_actions.breakAction->setEnabled(true); - d->m_actions.snapshotAction->setEnabled( - stopped && (engineCapabilities & SnapshotCapability)); + d->m_actions.snapshotAction-> + setEnabled(stopped && (engineCapabilities & SnapshotCapability)); const bool interruptIsExit = !running; if (interruptIsExit) { @@ -1762,6 +1775,8 @@ void DebuggerManager::setState(DebuggerState state, bool forced) d->m_actions.runToLineAction1->setEnabled(stopped); d->m_actions.runToLineAction2->setEnabled(stopped); d->m_actions.runToFunctionAction->setEnabled(stopped); + d->m_actions.returnFromFunctionAction-> + setEnabled(stopped && (engineCapabilities & ReturnFromFunctionCapability)); const bool canJump = stopped && (engineCapabilities & JumpToLineCapability); d->m_actions.jumpToLineAction1->setEnabled(canJump); diff --git a/src/plugins/debugger/debuggermanager.h b/src/plugins/debugger/debuggermanager.h index 55f03a055f614658adaa69c36f9bb655d200bbfc..9912221fdd0467998de411f046a11c9d137973e6 100644 --- a/src/plugins/debugger/debuggermanager.h +++ b/src/plugins/debugger/debuggermanager.h @@ -235,6 +235,7 @@ public slots: void stepOutExec(); void nextExec(); void continueExec(); + void returnExec(); void detachDebugger(); void makeSnapshot(); diff --git a/src/plugins/debugger/debuggerplugin.cpp b/src/plugins/debugger/debuggerplugin.cpp index 67788f74ed96c2533c93c016e2599badd0eba0f3..8c0358323b5d4304fa9e634aa795a7377489c646 100644 --- a/src/plugins/debugger/debuggerplugin.cpp +++ b/src/plugins/debugger/debuggerplugin.cpp @@ -119,6 +119,7 @@ const char * const RUN_TO_LINE2 = "Debugger.RunToLine2"; const char * const RUN_TO_FUNCTION = "Debugger.RunToFunction"; const char * const JUMP_TO_LINE1 = "Debugger.JumpToLine1"; const char * const JUMP_TO_LINE2 = "Debugger.JumpToLine2"; +const char * const RETURN_FROM_FUNCTION = "Debugger.ReturnFromFunction"; const char * const SNAPSHOT = "Debugger.Snapshot"; const char * const TOGGLE_BREAK = "Debugger.ToggleBreak"; const char * const BREAK_BY_FUNCTION = "Debugger.BreakByFunction"; @@ -799,6 +800,10 @@ bool DebuggerPlugin::initialize(const QStringList &arguments, QString *errorMess Constants::JUMP_TO_LINE1, debuggercontext); mdebug->addAction(cmd); + cmd = am->registerAction(actions.returnFromFunctionAction, + Constants::RETURN_FROM_FUNCTION, debuggercontext); + mdebug->addAction(cmd); + #ifdef USE_REVERSE_DEBUGGING cmd = am->registerAction(actions.reverseDirectionAction, Constants::REVERSE, debuggercontext); diff --git a/src/plugins/debugger/gdb/gdbengine.cpp b/src/plugins/debugger/gdb/gdbengine.cpp index 0947340b925ea28f4351d37dfffff576f941f499..27777a29e739d9e86a3e46c4f149107040a7d44c 100644 --- a/src/plugins/debugger/gdb/gdbengine.cpp +++ b/src/plugins/debugger/gdb/gdbengine.cpp @@ -1711,7 +1711,8 @@ unsigned GdbEngine::debuggerCapabilities() const | AutoDerefPointersCapability | DisassemblerCapability | RegisterCapability | ShowMemoryCapability | JumpToLineCapability | ReloadModuleCapability - | ReloadModuleSymbolsCapability | BreakOnThrowAndCatchCapability; + | ReloadModuleSymbolsCapability | BreakOnThrowAndCatchCapability + | ReturnFromFunctionCapability; } void GdbEngine::continueInferiorInternal() @@ -1902,6 +1903,21 @@ void GdbEngine::jumpToLineExec(const QString &fileName, int lineNumber) #endif } +void GdbEngine::returnExec() +{ + QTC_ASSERT(state() == InferiorStopped, qDebug() << state()); + setTokenBarrier(); + setState(InferiorRunningRequested); + showStatusMessage(tr("Immediate return from function requested..."), 5000); + postCommand("-exec-finish", RunRequest, CB(handleExecReturn)); +} + +void GdbEngine::handleExecReturn(const GdbResponse &response) +{ + if (response.resultClass == GdbResultDone) { + updateAll(); + } +} /*! \fn void GdbEngine::setTokenBarrier() \brief Discard the results of all pending watch-updating commands. diff --git a/src/plugins/debugger/gdb/gdbengine.h b/src/plugins/debugger/gdb/gdbengine.h index 1c127a5dbb527496545bf9da753ff10031e1a77f..479a6d64b3e25b3ef9aefb8b54c39bb8a06c00a0 100644 --- a/src/plugins/debugger/gdb/gdbengine.h +++ b/src/plugins/debugger/gdb/gdbengine.h @@ -296,10 +296,12 @@ private: ////////// Inferior Management ////////// virtual void runToFunctionExec(const QString &functionName); // void handleExecRunToFunction(const GdbResponse &response); virtual void jumpToLineExec(const QString &fileName, int lineNumber); + virtual void returnExec(); void handleExecContinue(const GdbResponse &response); void handleExecStep(const GdbResponse &response); void handleExecNext(const GdbResponse &response); + void handleExecReturn(const GdbResponse &response); qint64 inferiorPid() const { return m_manager->inferiorPid(); } void handleInferiorPidChanged(qint64 pid) { manager()->notifyInferiorPidChanged(pid); } diff --git a/src/plugins/debugger/idebuggerengine.h b/src/plugins/debugger/idebuggerengine.h index b60e8c7fb340cdf2cbcbee8389424893e3bb076a..6409ae5cb9569f2c7feec39fabe0be93fd768d3e 100644 --- a/src/plugins/debugger/idebuggerengine.h +++ b/src/plugins/debugger/idebuggerengine.h @@ -82,6 +82,7 @@ public: virtual void nextExec() = 0; virtual void stepIExec() = 0; virtual void nextIExec() = 0; + virtual void returnExec() {} virtual void continueInferior() = 0; virtual void interruptInferior() = 0;