From c7f5ad598c33db89a109295d5547480faefe5f55 Mon Sep 17 00:00:00 2001 From: Yappa Date: Wed, 22 Feb 2023 23:50:08 -0800 Subject: [PATCH] Fix "return" key in disassembler widget (#3090) Fix graph jumps --- rizin | 2 +- src/common/CutterSeekable.cpp | 1 + src/common/DisassemblyPreview.cpp | 10 ++++++++ src/common/DisassemblyPreview.h | 6 +++++ src/translations | 2 +- src/widgets/DisassemblerGraphView.cpp | 36 ++++++++++++++++++++++++++- src/widgets/DisassemblyWidget.cpp | 12 ++++++--- 7 files changed, 63 insertions(+), 6 deletions(-) diff --git a/rizin b/rizin index 12898a365e..ac1ea726b0 160000 --- a/rizin +++ b/rizin @@ -1 +1 @@ -Subproject commit 12898a365e70c22892d78abdd2627e7269533c5f +Subproject commit ac1ea726b023695449b5dd3287e3699a95012f19 diff --git a/src/common/CutterSeekable.cpp b/src/common/CutterSeekable.cpp index f8b0805a6a..821badb4e9 100644 --- a/src/common/CutterSeekable.cpp +++ b/src/common/CutterSeekable.cpp @@ -66,6 +66,7 @@ void CutterSeekable::seekToReference(RVA offset) } RVA target; + // finds the xrefs for calls, lea, and jmp QList refs = Core()->getXRefs(offset, false, false); if (refs.length()) { diff --git a/src/common/DisassemblyPreview.cpp b/src/common/DisassemblyPreview.cpp index e80e6eaabb..1488f947f6 100644 --- a/src/common/DisassemblyPreview.cpp +++ b/src/common/DisassemblyPreview.cpp @@ -89,3 +89,13 @@ RVA DisassemblyPreview::readDisassemblyOffset(QTextCursor tc) return userData->line.offset; } + +RVA DisassemblyPreview::readDisassemblyArrow(QTextCursor tc) +{ + auto userData = getUserData(tc.block()); + if (!userData && userData->line.arrow != RVA_INVALID) { + return RVA_INVALID; + } + + return userData->line.arrow; +} diff --git a/src/common/DisassemblyPreview.h b/src/common/DisassemblyPreview.h index 81e67ee8dc..6ddcbe1294 100644 --- a/src/common/DisassemblyPreview.h +++ b/src/common/DisassemblyPreview.h @@ -41,5 +41,11 @@ bool showDisasPreview(QWidget *parent, const QPoint &pointOfEvent, const RVA off * @return The disassembly offset of the hovered asm text */ RVA readDisassemblyOffset(QTextCursor tc); + +/*! + * @brief Reads the arrow offset for the cursor position + * @return The jump address of the hovered asm text + */ +RVA readDisassemblyArrow(QTextCursor tc); } #endif diff --git a/src/translations b/src/translations index 41c0c778b9..6f96c37559 160000 --- a/src/translations +++ b/src/translations @@ -1 +1 @@ -Subproject commit 41c0c778b942577749ea2fed117e48a2cf3892df +Subproject commit 6f96c37559455beb30e5733476409e0a33a3c915 diff --git a/src/widgets/DisassemblerGraphView.cpp b/src/widgets/DisassemblerGraphView.cpp index 69f87e5b04..4edc9a899b 100644 --- a/src/widgets/DisassemblerGraphView.cpp +++ b/src/widgets/DisassemblerGraphView.cpp @@ -914,7 +914,41 @@ void DisassemblerGraphView::blockDoubleClicked(GraphView::GraphBlock &block, QMo QPoint pos) { Q_UNUSED(event); - seekable->seekToReference(getAddrForMouseEvent(block, &pos)); + RVA arrow = NULL; + RVA offset = getAddrForMouseEvent(block, &pos); + DisassemblyBlock *db = blockForAddress(offset); + + Instr lastInstruction = db->instrs.back(); + + // Handle the blocks without any paths + if (offset == lastInstruction.addr && db->false_path == RVA_INVALID + && db->true_path == RVA_INVALID) { + return; + } + + // Handle the blocks with just one path + if (offset == lastInstruction.addr && db->false_path == RVA_INVALID) { + seekable->seek(db->true_path); + return; + } + + // Handle blocks with two paths + if (offset == lastInstruction.addr && db->false_path != RVA_INVALID) { + // gets the offset for the next instruction + RVA nextOffset = lastInstruction.addr + lastInstruction.size; + // sets "arrow" to the path that isn't going to the next offset + if (db->false_path == nextOffset) { + arrow = db->true_path; + } else if (db->true_path == nextOffset) { + arrow = db->false_path; + } + + seekable->seek(arrow); + return; + } + + // Handle "call" instruction to functions + seekable->seekToReference(offset); } void DisassemblerGraphView::blockHelpEvent(GraphView::GraphBlock &block, QHelpEvent *event, diff --git a/src/widgets/DisassemblyWidget.cpp b/src/widgets/DisassemblyWidget.cpp index 4c18b57f5c..41bbada4f7 100644 --- a/src/widgets/DisassemblyWidget.cpp +++ b/src/widgets/DisassemblyWidget.cpp @@ -613,6 +613,13 @@ void DisassemblyWidget::moveCursorRelative(bool up, bool page) void DisassemblyWidget::jumpToOffsetUnderCursor(const QTextCursor &cursor) { + // Handles "jmp" and conditonal jump instructions + RVA arrow = DisassemblyPreview::readDisassemblyArrow(cursor); + if (arrow != RVA_INVALID) { + seekable->seek(arrow); + } + + // Handles "call" and "lea" instructions RVA offset = DisassemblyPreview::readDisassemblyOffset(cursor); seekable->seekToReference(offset); } @@ -627,9 +634,8 @@ bool DisassemblyWidget::eventFilter(QObject *obj, QEvent *event) jumpToOffsetUnderCursor(cursor); return true; - } else if (Config()->getPreviewValue() - && event->type() == QEvent::ToolTip - && obj == mDisasTextEdit->viewport()) { + } else if (Config()->getPreviewValue() && event->type() == QEvent::ToolTip + && obj == mDisasTextEdit->viewport()) { QHelpEvent *helpEvent = static_cast(event); auto cursorForWord = mDisasTextEdit->cursorForPosition(helpEvent->pos());