2019-09-06 05:40:20 +00:00
|
|
|
#include "DecompilerWidget.h"
|
|
|
|
#include "ui_DecompilerWidget.h"
|
2020-06-22 20:25:30 +00:00
|
|
|
#include "menus/DecompilerContextMenu.h"
|
2017-12-15 10:52:47 +00:00
|
|
|
|
2018-10-17 07:55:53 +00:00
|
|
|
#include "common/Configuration.h"
|
|
|
|
#include "common/Helpers.h"
|
|
|
|
#include "common/TempConfig.h"
|
2019-07-12 08:57:07 +00:00
|
|
|
#include "common/SelectionHighlight.h"
|
2019-07-15 12:08:44 +00:00
|
|
|
#include "common/Decompiler.h"
|
2019-12-14 12:57:36 +00:00
|
|
|
#include "common/CutterSeekable.h"
|
2017-12-06 23:19:14 +00:00
|
|
|
|
2019-07-11 13:21:54 +00:00
|
|
|
#include <QTextEdit>
|
2019-07-12 08:57:07 +00:00
|
|
|
#include <QPlainTextEdit>
|
|
|
|
#include <QTextBlock>
|
|
|
|
#include <QObject>
|
|
|
|
#include <QTextBlockUserData>
|
|
|
|
|
2020-05-22 11:49:34 +00:00
|
|
|
DecompilerWidget::DecompilerWidget(MainWindow *main) :
|
|
|
|
MemoryDockWidget(MemoryWidgetType::Decompiler, main),
|
2020-06-22 20:25:30 +00:00
|
|
|
mCtxMenu(new DecompilerContextMenu(this, main)),
|
2020-06-08 21:29:26 +00:00
|
|
|
ui(new Ui::DecompilerWidget),
|
2020-06-29 19:08:02 +00:00
|
|
|
code(Decompiler::makeWarning(tr("Choose an offset and refresh to get decompiled code")),
|
|
|
|
&r_annotated_code_free)
|
2017-12-06 23:19:14 +00:00
|
|
|
{
|
2017-12-21 14:23:44 +00:00
|
|
|
ui->setupUi(this);
|
|
|
|
|
2019-07-11 13:21:54 +00:00
|
|
|
syntaxHighlighter = Config()->createSyntaxHighlighter(ui->textEdit->document());
|
2017-12-16 13:22:56 +00:00
|
|
|
|
2019-12-14 12:57:36 +00:00
|
|
|
// Event filter to intercept double clicks in the textbox
|
|
|
|
ui->textEdit->viewport()->installEventFilter(this);
|
|
|
|
|
2017-12-06 23:19:14 +00:00
|
|
|
setupFonts();
|
|
|
|
colorsUpdatedSlot();
|
|
|
|
|
2020-08-03 09:13:39 +00:00
|
|
|
connect(Config(), &Configuration::fontsUpdated, this, &DecompilerWidget::fontsUpdatedSlot);
|
|
|
|
connect(Config(), &Configuration::colorsUpdated, this, &DecompilerWidget::colorsUpdatedSlot);
|
|
|
|
connect(Core(), &CutterCore::registersChanged, this, &DecompilerWidget::highlightPC);
|
2020-06-22 20:25:30 +00:00
|
|
|
connect(mCtxMenu, &DecompilerContextMenu::copy, ui->textEdit, &QPlainTextEdit::copy);
|
2017-12-06 23:19:14 +00:00
|
|
|
|
2019-09-01 09:06:54 +00:00
|
|
|
decompiledFunctionAddr = RVA_INVALID;
|
|
|
|
decompilerWasBusy = false;
|
|
|
|
|
2017-12-21 14:23:44 +00:00
|
|
|
connect(ui->refreshButton, &QAbstractButton::clicked, this, [this]() {
|
2019-09-01 09:06:54 +00:00
|
|
|
doRefresh();
|
|
|
|
});
|
|
|
|
|
|
|
|
refreshDeferrer = createRefreshDeferrer([this]() {
|
|
|
|
doRefresh();
|
|
|
|
});
|
|
|
|
|
|
|
|
autoRefreshEnabled = Config()->getDecompilerAutoRefreshEnabled();
|
|
|
|
ui->autoRefreshCheckBox->setChecked(autoRefreshEnabled);
|
|
|
|
setAutoRefresh(autoRefreshEnabled);
|
|
|
|
connect(ui->autoRefreshCheckBox, &QCheckBox::stateChanged, this, [this](int state) {
|
|
|
|
setAutoRefresh(state == Qt::Checked);
|
|
|
|
Config()->setDecompilerAutoRefreshEnabled(autoRefreshEnabled);
|
|
|
|
doAutoRefresh();
|
2017-12-06 23:19:14 +00:00
|
|
|
});
|
|
|
|
|
2019-07-15 12:08:44 +00:00
|
|
|
auto decompilers = Core()->getDecompilers();
|
2019-07-16 18:33:05 +00:00
|
|
|
auto selectedDecompilerId = Config()->getSelectedDecompiler();
|
2020-01-21 16:43:30 +00:00
|
|
|
if (selectedDecompilerId.isEmpty()) {
|
|
|
|
// If no decompiler was previously chosen. set r2ghidra as default decompiler
|
|
|
|
selectedDecompilerId = "r2ghidra";
|
|
|
|
}
|
2019-07-15 12:08:44 +00:00
|
|
|
for (auto dec : decompilers) {
|
|
|
|
ui->decompilerComboBox->addItem(dec->getName(), dec->getId());
|
2019-07-16 18:33:05 +00:00
|
|
|
if (dec->getId() == selectedDecompilerId) {
|
|
|
|
ui->decompilerComboBox->setCurrentIndex(ui->decompilerComboBox->count() - 1);
|
|
|
|
}
|
2019-09-06 05:40:20 +00:00
|
|
|
connect(dec, &Decompiler::finished, this, &DecompilerWidget::decompilationFinished);
|
2019-07-15 12:08:44 +00:00
|
|
|
}
|
|
|
|
|
2019-08-28 17:01:12 +00:00
|
|
|
decompilerSelectionEnabled = decompilers.size() > 1;
|
|
|
|
ui->decompilerComboBox->setEnabled(decompilerSelectionEnabled);
|
|
|
|
|
|
|
|
if (decompilers.isEmpty()) {
|
|
|
|
ui->textEdit->setPlainText(tr("No Decompiler available."));
|
2018-09-08 07:12:08 +00:00
|
|
|
}
|
|
|
|
|
2020-06-29 19:08:02 +00:00
|
|
|
connect(ui->decompilerComboBox,
|
|
|
|
static_cast<void(QComboBox::*)(int)>(&QComboBox::currentIndexChanged), this,
|
|
|
|
&DecompilerWidget::decompilerSelected);
|
2019-07-12 08:57:07 +00:00
|
|
|
connectCursorPositionChanged(false);
|
2019-09-06 05:40:20 +00:00
|
|
|
connect(Core(), &CutterCore::seekChanged, this, &DecompilerWidget::seekChanged);
|
2019-08-07 11:38:22 +00:00
|
|
|
ui->textEdit->setContextMenuPolicy(Qt::CustomContextMenu);
|
2020-08-03 09:13:39 +00:00
|
|
|
connect(ui->textEdit, &QWidget::customContextMenuRequested,
|
|
|
|
this, &DecompilerWidget::showDisasContextMenu);
|
2019-08-07 11:38:22 +00:00
|
|
|
|
|
|
|
// refresh the widget when an action in this menu is triggered
|
2019-09-06 05:40:20 +00:00
|
|
|
connect(mCtxMenu, &QMenu::triggered, this, &DecompilerWidget::refreshDecompiler);
|
2020-06-29 19:08:02 +00:00
|
|
|
connect(Core(), &CutterCore::breakpointsChanged, this, &DecompilerWidget::setInfoForBreakpoints);
|
2019-08-07 11:38:22 +00:00
|
|
|
addActions(mCtxMenu->actions());
|
2019-07-12 08:57:07 +00:00
|
|
|
|
2019-08-28 17:01:12 +00:00
|
|
|
ui->progressLabel->setVisible(false);
|
2019-01-12 17:02:51 +00:00
|
|
|
doRefresh(RVA_INVALID);
|
2019-09-01 09:06:54 +00:00
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
connect(Core(), &CutterCore::refreshAll, this, &DecompilerWidget::doAutoRefresh);
|
|
|
|
connect(Core(), &CutterCore::functionRenamed, this, &DecompilerWidget::doAutoRefresh);
|
|
|
|
connect(Core(), &CutterCore::varsChanged, this, &DecompilerWidget::doAutoRefresh);
|
|
|
|
connect(Core(), &CutterCore::functionsChanged, this, &DecompilerWidget::doAutoRefresh);
|
|
|
|
connect(Core(), &CutterCore::flagsChanged, this, &DecompilerWidget::doAutoRefresh);
|
|
|
|
connect(Core(), &CutterCore::commentsChanged, this, &DecompilerWidget::doAutoRefresh);
|
|
|
|
connect(Core(), &CutterCore::instructionChanged, this, &DecompilerWidget::doAutoRefresh);
|
|
|
|
connect(Core(), &CutterCore::refreshCodeViews, this, &DecompilerWidget::doAutoRefresh);
|
2019-12-14 12:57:36 +00:00
|
|
|
|
|
|
|
// Esc to seek backward
|
|
|
|
QAction *seekPrevAction = new QAction(this);
|
|
|
|
seekPrevAction->setShortcut(Qt::Key_Escape);
|
|
|
|
seekPrevAction->setShortcutContext(Qt::WidgetWithChildrenShortcut);
|
|
|
|
addAction(seekPrevAction);
|
|
|
|
connect(seekPrevAction, &QAction::triggered, seekable, &CutterSeekable::seekPrev);
|
2017-12-06 23:19:14 +00:00
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
DecompilerWidget::~DecompilerWidget() = default;
|
2017-12-06 23:19:14 +00:00
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
Decompiler *DecompilerWidget::getCurrentDecompiler()
|
2019-09-01 09:06:54 +00:00
|
|
|
{
|
|
|
|
return Core()->getDecompilerById(ui->decompilerComboBox->currentData().toString());
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::setAutoRefresh(bool enabled)
|
2019-09-01 09:06:54 +00:00
|
|
|
{
|
|
|
|
autoRefreshEnabled = enabled;
|
|
|
|
updateRefreshButton();
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::doAutoRefresh()
|
2019-09-01 09:06:54 +00:00
|
|
|
{
|
|
|
|
if (!autoRefreshEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
doRefresh();
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::updateRefreshButton()
|
2019-09-01 09:06:54 +00:00
|
|
|
{
|
|
|
|
Decompiler *dec = getCurrentDecompiler();
|
|
|
|
ui->refreshButton->setEnabled(!autoRefreshEnabled && dec && !dec->isRunning());
|
|
|
|
if (dec && dec->isRunning() && dec->isCancelable()) {
|
|
|
|
ui->refreshButton->setText(tr("Cancel"));
|
|
|
|
} else {
|
|
|
|
ui->refreshButton->setText(tr("Refresh"));
|
|
|
|
}
|
|
|
|
}
|
2017-12-06 23:19:14 +00:00
|
|
|
|
2020-06-08 21:29:26 +00:00
|
|
|
static ut64 offsetForPosition(RAnnotatedCode &codeDecompiled, size_t pos)
|
|
|
|
{
|
|
|
|
size_t closestPos = SIZE_MAX;
|
|
|
|
ut64 closestOffset = UT64_MAX;
|
|
|
|
void *annotationi;
|
|
|
|
r_vector_foreach(&codeDecompiled.annotations, annotationi) {
|
|
|
|
RCodeAnnotation *annotation = (RCodeAnnotation *)annotationi;
|
|
|
|
if (annotation->type != R_CODE_ANNOTATION_TYPE_OFFSET || annotation->start > pos
|
|
|
|
|| annotation->end <= pos) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (closestPos != SIZE_MAX && closestPos >= annotation->start) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
closestPos = annotation->start;
|
|
|
|
closestOffset = annotation->offset.offset;
|
|
|
|
}
|
|
|
|
return closestOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t positionForOffset(RAnnotatedCode &codeDecompiled, ut64 offset)
|
|
|
|
{
|
|
|
|
size_t closestPos = SIZE_MAX;
|
|
|
|
ut64 closestOffset = UT64_MAX;
|
|
|
|
void *annotationi;
|
|
|
|
r_vector_foreach(&codeDecompiled.annotations, annotationi) {
|
|
|
|
RCodeAnnotation *annotation = (RCodeAnnotation *)annotationi;
|
|
|
|
if (annotation->type != R_CODE_ANNOTATION_TYPE_OFFSET || annotation->offset.offset > offset) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (closestOffset != UT64_MAX && closestOffset >= annotation->offset.offset) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
closestPos = annotation->start;
|
|
|
|
closestOffset = annotation->offset.offset;
|
|
|
|
}
|
|
|
|
return closestPos;
|
|
|
|
}
|
|
|
|
|
2020-06-29 19:08:02 +00:00
|
|
|
void DecompilerWidget::setInfoForBreakpoints()
|
|
|
|
{
|
|
|
|
if (mCtxMenu->getIsTogglingBreakpoints())
|
|
|
|
return;
|
|
|
|
// Get the range of the line
|
|
|
|
QTextCursor cursorForLine = ui->textEdit->textCursor();
|
|
|
|
cursorForLine.movePosition(QTextCursor::StartOfLine);
|
|
|
|
size_t startPos = cursorForLine.position();
|
|
|
|
cursorForLine.movePosition(QTextCursor::EndOfLine);
|
|
|
|
size_t endPos = cursorForLine.position();
|
|
|
|
gatherBreakpointInfo(*code, startPos, endPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DecompilerWidget::gatherBreakpointInfo(RAnnotatedCode &codeDecompiled, size_t startPos,
|
|
|
|
size_t endPos)
|
|
|
|
{
|
|
|
|
RVA firstOffset = RVA_MAX;
|
|
|
|
void *annotationi;
|
|
|
|
r_vector_foreach(&codeDecompiled.annotations, annotationi) {
|
|
|
|
RCodeAnnotation *annotation = (RCodeAnnotation *)annotationi;
|
|
|
|
if (annotation->type != R_CODE_ANNOTATION_TYPE_OFFSET) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if ((startPos <= annotation->start && annotation->start < endPos) || (startPos <= annotation->end
|
|
|
|
&& annotation->end < endPos)) {
|
|
|
|
firstOffset = (annotation->offset.offset < firstOffset) ? annotation->offset.offset : firstOffset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mCtxMenu->setFirstOffsetInLine(firstOffset);
|
|
|
|
QList<RVA> functionBreakpoints = Core()->getBreakpointsInFunction(decompiledFunctionAddr);
|
|
|
|
QVector<RVA> offsetList;
|
|
|
|
for (auto bpOffset : functionBreakpoints) {
|
|
|
|
size_t pos = positionForOffset(*code, bpOffset);
|
|
|
|
if (startPos <= pos && pos <= endPos) {
|
|
|
|
offsetList.push_back(bpOffset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
std::sort(offsetList.begin(), offsetList.end());
|
|
|
|
mCtxMenu->setAvailableBreakpoints(offsetList);
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::doRefresh(RVA addr)
|
2017-12-06 23:19:14 +00:00
|
|
|
{
|
2019-09-01 09:06:54 +00:00
|
|
|
if (!refreshDeferrer->attemptRefresh(nullptr)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-07-15 12:08:44 +00:00
|
|
|
if (ui->decompilerComboBox->currentIndex() < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-09-01 09:06:54 +00:00
|
|
|
Decompiler *dec = getCurrentDecompiler();
|
|
|
|
if (!dec) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dec->isRunning()) {
|
|
|
|
decompilerWasBusy = true;
|
2019-07-15 12:08:44 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-03-21 20:32:32 +00:00
|
|
|
if (addr == RVA_INVALID) {
|
2019-09-06 05:40:20 +00:00
|
|
|
ui->textEdit->setPlainText(tr("Click Refresh to generate Decompiler from current offset."));
|
2017-12-21 14:23:44 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-12-19 17:58:30 +00:00
|
|
|
// Clear all selections since we just refreshed
|
|
|
|
ui->textEdit->setExtraSelections({});
|
2019-09-03 14:25:28 +00:00
|
|
|
decompiledFunctionAddr = Core()->getFunctionStart(addr);
|
2019-08-28 17:01:12 +00:00
|
|
|
dec->decompileAt(addr);
|
|
|
|
if (dec->isRunning()) {
|
|
|
|
ui->progressLabel->setVisible(true);
|
|
|
|
ui->decompilerComboBox->setEnabled(false);
|
2019-09-01 09:06:54 +00:00
|
|
|
updateRefreshButton();
|
2019-08-28 17:01:12 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::refreshDecompiler()
|
2019-08-28 17:01:12 +00:00
|
|
|
{
|
2019-09-01 09:06:54 +00:00
|
|
|
doRefresh();
|
2020-06-29 19:08:02 +00:00
|
|
|
setInfoForBreakpoints();
|
2019-08-28 17:01:12 +00:00
|
|
|
}
|
|
|
|
|
2019-12-19 17:58:30 +00:00
|
|
|
QTextCursor DecompilerWidget::getCursorForAddress(RVA addr)
|
|
|
|
{
|
2020-06-08 21:29:26 +00:00
|
|
|
size_t pos = positionForOffset(*code, addr);
|
2019-12-19 17:58:30 +00:00
|
|
|
if (pos == SIZE_MAX || pos == 0) {
|
|
|
|
return QTextCursor();
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextCursor cursor = ui->textEdit->textCursor();
|
|
|
|
cursor.setPosition(pos);
|
|
|
|
return cursor;
|
|
|
|
}
|
|
|
|
|
2020-06-08 21:29:26 +00:00
|
|
|
void DecompilerWidget::decompilationFinished(RAnnotatedCode *codeDecompiled)
|
2019-08-28 17:01:12 +00:00
|
|
|
{
|
|
|
|
ui->progressLabel->setVisible(false);
|
|
|
|
ui->decompilerComboBox->setEnabled(decompilerSelectionEnabled);
|
2019-09-01 09:06:54 +00:00
|
|
|
updateRefreshButton();
|
2019-08-28 17:01:12 +00:00
|
|
|
|
2020-06-08 21:29:26 +00:00
|
|
|
this->code.reset(codeDecompiled);
|
|
|
|
QString codeString = QString::fromUtf8(this->code->code);
|
|
|
|
if (codeString.isEmpty()) {
|
2019-08-28 17:01:12 +00:00
|
|
|
ui->textEdit->setPlainText(tr("Cannot decompile at this address (Not a function?)"));
|
2017-12-15 10:52:47 +00:00
|
|
|
return;
|
2019-07-12 08:57:07 +00:00
|
|
|
} else {
|
|
|
|
connectCursorPositionChanged(true);
|
2020-06-08 21:29:26 +00:00
|
|
|
ui->textEdit->setPlainText(codeString);
|
2019-07-12 08:57:07 +00:00
|
|
|
connectCursorPositionChanged(false);
|
2019-09-01 09:06:54 +00:00
|
|
|
updateCursorPosition();
|
2019-12-19 17:58:30 +00:00
|
|
|
highlightPC();
|
|
|
|
highlightBreakpoints();
|
2019-09-01 09:06:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (decompilerWasBusy) {
|
|
|
|
decompilerWasBusy = false;
|
|
|
|
doAutoRefresh();
|
2017-12-06 23:19:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::decompilerSelected()
|
2019-07-16 18:33:05 +00:00
|
|
|
{
|
2019-08-23 20:06:44 +00:00
|
|
|
Config()->setSelectedDecompiler(ui->decompilerComboBox->currentData().toString());
|
2019-09-01 09:06:54 +00:00
|
|
|
if (autoRefreshEnabled) {
|
|
|
|
doRefresh();
|
|
|
|
}
|
2019-07-16 18:33:05 +00:00
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::connectCursorPositionChanged(bool disconnect)
|
2019-07-12 08:57:07 +00:00
|
|
|
{
|
|
|
|
if (disconnect) {
|
2020-06-29 19:08:02 +00:00
|
|
|
QObject::disconnect(ui->textEdit, &QPlainTextEdit::cursorPositionChanged, this,
|
|
|
|
&DecompilerWidget::cursorPositionChanged);
|
2019-07-12 08:57:07 +00:00
|
|
|
} else {
|
2020-06-29 19:08:02 +00:00
|
|
|
connect(ui->textEdit, &QPlainTextEdit::cursorPositionChanged, this,
|
|
|
|
&DecompilerWidget::cursorPositionChanged);
|
2019-07-12 08:57:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::cursorPositionChanged()
|
2019-07-12 08:57:07 +00:00
|
|
|
{
|
2020-06-22 20:25:30 +00:00
|
|
|
mCtxMenu->setCanCopy(ui->textEdit->textCursor().hasSelection());
|
2019-12-14 10:42:24 +00:00
|
|
|
// Do not perform seeks along with the cursor while selecting multiple lines
|
2020-06-22 20:25:30 +00:00
|
|
|
if (!ui->textEdit->textCursor().selectedText().isEmpty()) {
|
2019-12-14 10:42:24 +00:00
|
|
|
return;
|
|
|
|
}
|
2020-06-29 19:08:02 +00:00
|
|
|
|
2019-08-27 15:27:39 +00:00
|
|
|
size_t pos = ui->textEdit->textCursor().position();
|
2020-06-29 19:08:02 +00:00
|
|
|
|
|
|
|
|
|
|
|
setInfoForBreakpoints();
|
|
|
|
|
2020-06-08 21:29:26 +00:00
|
|
|
RVA offset = offsetForPosition(*code, pos);
|
2019-07-12 08:57:07 +00:00
|
|
|
if (offset != RVA_INVALID && offset != Core()->getOffset()) {
|
|
|
|
seekFromCursor = true;
|
|
|
|
Core()->seek(offset);
|
2019-08-07 11:38:22 +00:00
|
|
|
mCtxMenu->setOffset(offset);
|
2019-07-12 08:57:07 +00:00
|
|
|
seekFromCursor = false;
|
|
|
|
}
|
|
|
|
updateSelection();
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::seekChanged()
|
2019-07-12 08:57:07 +00:00
|
|
|
{
|
|
|
|
if (seekFromCursor) {
|
|
|
|
return;
|
|
|
|
}
|
2019-09-01 09:06:54 +00:00
|
|
|
|
|
|
|
if (autoRefreshEnabled) {
|
|
|
|
auto fcnAddr = Core()->getFunctionStart(Core()->getOffset());
|
|
|
|
if (fcnAddr == RVA_INVALID || fcnAddr != decompiledFunctionAddr) {
|
|
|
|
doRefresh();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-12 08:57:07 +00:00
|
|
|
updateCursorPosition();
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::updateCursorPosition()
|
2019-07-12 08:57:07 +00:00
|
|
|
{
|
|
|
|
RVA offset = Core()->getOffset();
|
2020-06-08 21:29:26 +00:00
|
|
|
size_t pos = positionForOffset(*code, offset);
|
2019-08-27 15:27:39 +00:00
|
|
|
if (pos == SIZE_MAX) {
|
|
|
|
return;
|
2019-07-12 08:57:07 +00:00
|
|
|
}
|
2019-10-20 08:58:58 +00:00
|
|
|
mCtxMenu->setOffset(offset);
|
2019-08-27 15:27:39 +00:00
|
|
|
connectCursorPositionChanged(true);
|
|
|
|
QTextCursor cursor = ui->textEdit->textCursor();
|
|
|
|
cursor.setPosition(pos);
|
|
|
|
ui->textEdit->setTextCursor(cursor);
|
|
|
|
updateSelection();
|
2019-07-12 08:57:07 +00:00
|
|
|
connectCursorPositionChanged(false);
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::setupFonts()
|
2017-12-06 23:19:14 +00:00
|
|
|
{
|
2019-10-12 05:50:10 +00:00
|
|
|
ui->textEdit->setFont(Config()->getFont());
|
2017-12-06 23:19:14 +00:00
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::updateSelection()
|
2019-07-12 08:57:07 +00:00
|
|
|
{
|
|
|
|
QList<QTextEdit::ExtraSelection> extraSelections;
|
|
|
|
|
|
|
|
// Highlight the current line
|
|
|
|
auto cursor = ui->textEdit->textCursor();
|
2019-08-27 15:27:39 +00:00
|
|
|
extraSelections.append(createLineHighlightSelection(cursor));
|
2019-07-12 08:57:07 +00:00
|
|
|
|
|
|
|
// Highlight all the words in the document same as the current one
|
|
|
|
cursor.select(QTextCursor::WordUnderCursor);
|
|
|
|
QString searchString = cursor.selectedText();
|
|
|
|
extraSelections.append(createSameWordsSelections(ui->textEdit, searchString));
|
|
|
|
|
|
|
|
ui->textEdit->setExtraSelections(extraSelections);
|
2019-12-19 17:58:30 +00:00
|
|
|
// Highlight PC after updating the selected line
|
|
|
|
highlightPC();
|
2019-07-12 08:57:07 +00:00
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
QString DecompilerWidget::getWindowTitle() const
|
2019-06-18 13:02:41 +00:00
|
|
|
{
|
2019-09-06 05:40:20 +00:00
|
|
|
return tr("Decompiler");
|
2019-06-18 13:02:41 +00:00
|
|
|
}
|
|
|
|
|
2019-10-12 05:50:10 +00:00
|
|
|
void DecompilerWidget::fontsUpdatedSlot()
|
2017-12-06 23:19:14 +00:00
|
|
|
{
|
|
|
|
setupFonts();
|
|
|
|
}
|
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::colorsUpdatedSlot()
|
2017-12-06 23:19:14 +00:00
|
|
|
{
|
|
|
|
}
|
2019-08-07 11:38:22 +00:00
|
|
|
|
2019-09-06 05:40:20 +00:00
|
|
|
void DecompilerWidget::showDisasContextMenu(const QPoint &pt)
|
2019-08-07 11:38:22 +00:00
|
|
|
{
|
|
|
|
mCtxMenu->exec(ui->textEdit->mapToGlobal(pt));
|
2019-09-01 09:06:54 +00:00
|
|
|
doRefresh();
|
2019-12-14 12:57:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DecompilerWidget::seekToReference()
|
|
|
|
{
|
|
|
|
size_t pos = ui->textEdit->textCursor().position();
|
2020-06-08 21:29:26 +00:00
|
|
|
RVA offset = offsetForPosition(*code, pos);
|
2019-12-14 12:57:36 +00:00
|
|
|
seekable->seekToReference(offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DecompilerWidget::eventFilter(QObject *obj, QEvent *event)
|
|
|
|
{
|
|
|
|
if (event->type() == QEvent::MouseButtonDblClick
|
2020-06-29 19:08:02 +00:00
|
|
|
&& (obj == ui->textEdit || obj == ui->textEdit->viewport())) {
|
2019-12-14 12:57:36 +00:00
|
|
|
QMouseEvent *mouseEvent = static_cast<QMouseEvent *>(event);
|
|
|
|
|
2020-06-29 19:08:02 +00:00
|
|
|
const QTextCursor &cursor = ui->textEdit->cursorForPosition(QPoint(mouseEvent->x(),
|
|
|
|
mouseEvent->y()));
|
2019-12-14 12:57:36 +00:00
|
|
|
seekToReference();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return MemoryDockWidget::eventFilter(obj, event);
|
|
|
|
}
|
2019-12-19 17:58:30 +00:00
|
|
|
|
|
|
|
void DecompilerWidget::highlightPC()
|
|
|
|
{
|
|
|
|
RVA PCAddress = Core()->getProgramCounterValue();
|
|
|
|
if (PCAddress == RVA_INVALID || (Core()->getFunctionStart(PCAddress) != decompiledFunctionAddr)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextCursor cursor = getCursorForAddress(PCAddress);
|
|
|
|
if (!cursor.isNull()) {
|
|
|
|
colorLine(createLineHighlightPC(cursor));
|
|
|
|
}
|
2020-06-22 20:25:30 +00:00
|
|
|
|
2019-12-19 17:58:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DecompilerWidget::highlightBreakpoints()
|
|
|
|
{
|
|
|
|
|
|
|
|
QList<RVA> functionBreakpoints = Core()->getBreakpointsInFunction(decompiledFunctionAddr);
|
|
|
|
QTextCursor cursor;
|
2020-06-29 19:08:02 +00:00
|
|
|
for (auto &bp : functionBreakpoints) {
|
2019-12-19 17:58:30 +00:00
|
|
|
if (bp == RVA_INVALID) {
|
|
|
|
continue;;
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor = getCursorForAddress(bp);
|
|
|
|
if (!cursor.isNull()) {
|
|
|
|
// Use a Block formatting since these lines are not updated frequently as selections and PC
|
|
|
|
QTextBlockFormat f;
|
|
|
|
f.setBackground(ConfigColor("gui.breakpoint_background"));
|
|
|
|
cursor.setBlockFormat(f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DecompilerWidget::colorLine(QTextEdit::ExtraSelection extraSelection)
|
|
|
|
{
|
|
|
|
QList<QTextEdit::ExtraSelection> extraSelections = ui->textEdit->extraSelections();
|
|
|
|
extraSelections.append(extraSelection);
|
|
|
|
ui->textEdit->setExtraSelections(extraSelections);
|
|
|
|
return true;
|
|
|
|
}
|