cutter/src/widgets/DecompilerWidget.cpp

634 lines
21 KiB
C++
Raw Normal View History

#include "DecompilerWidget.h"
#include "ui_DecompilerWidget.h"
#include "menus/DecompilerContextMenu.h"
2018-10-17 07:55:53 +00:00
#include "common/Configuration.h"
#include "common/Helpers.h"
#include "common/TempConfig.h"
#include "common/SelectionHighlight.h"
#include "common/Decompiler.h"
#include "common/CutterSeekable.h"
#include "core/MainWindow.h"
#include "common/DecompilerHighlighter.h"
2019-07-11 13:21:54 +00:00
#include <QTextEdit>
#include <QPlainTextEdit>
#include <QTextBlock>
#include <QClipboard>
#include <QObject>
#include <QTextBlockUserData>
#include <QScrollBar>
#include <QAbstractSlider>
2021-01-24 14:50:13 +00:00
DecompilerWidget::DecompilerWidget(MainWindow *main)
: MemoryDockWidget(MemoryWidgetType::Decompiler, main),
mCtxMenu(new DecompilerContextMenu(this, main)),
ui(new Ui::DecompilerWidget),
decompilerBusy(false),
seekFromCursor(false),
scrollerHorizontal(0),
scrollerVertical(0),
previousFunctionAddr(RVA_INVALID),
decompiledFunctionAddr(RVA_INVALID),
code(Decompiler::makeWarning(tr("Choose an offset and refresh to get decompiled code")),
&rz_annotated_code_free)
{
ui->setupUi(this);
2021-01-24 14:50:13 +00:00
setObjectName(main ? main->getUniqueObjectName(getWidgetType()) : getWidgetType());
updateWindowTitle();
setHighlighter(Config()->isDecompilerAnnotationHighlighterEnabled());
// Event filter to intercept double click and right click in the textbox
ui->textEdit->viewport()->installEventFilter(this);
setupFonts();
colorsUpdatedSlot();
connect(Config(), &Configuration::fontsUpdated, this, &DecompilerWidget::fontsUpdatedSlot);
connect(Config(), &Configuration::colorsUpdated, this, &DecompilerWidget::colorsUpdatedSlot);
connect(Core(), &CutterCore::registersChanged, this, &DecompilerWidget::highlightPC);
connect(mCtxMenu, &DecompilerContextMenu::copy, this, &DecompilerWidget::copy);
2021-01-24 14:50:13 +00:00
refreshDeferrer = createRefreshDeferrer([this]() { doRefresh(); });
auto decompilers = Core()->getDecompilers();
QString selectedDecompilerId = Config()->getSelectedDecompiler();
if (selectedDecompilerId.isEmpty()) {
2020-10-28 12:28:04 +00:00
// If no decompiler was previously chosen. set rz-ghidra as default decompiler
selectedDecompilerId = "ghidra";
}
for (Decompiler *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);
}
}
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
}
connect(ui->decompilerComboBox,
2021-01-24 14:50:13 +00:00
static_cast<void (QComboBox::*)(int)>(&QComboBox::currentIndexChanged), this,
&DecompilerWidget::decompilerSelected);
connectCursorPositionChanged(true);
connect(seekable, &CutterSeekable::seekableSeekChanged, this, &DecompilerWidget::seekChanged);
ui->textEdit->setContextMenuPolicy(Qt::CustomContextMenu);
2021-01-24 14:50:13 +00:00
connect(ui->textEdit, &QWidget::customContextMenuRequested, this,
&DecompilerWidget::showDecompilerContextMenu);
connect(Core(), &CutterCore::breakpointsChanged, this, &DecompilerWidget::updateBreakpoints);
mCtxMenu->addSeparator();
mCtxMenu->addAction(&syncAction);
addActions(mCtxMenu->actions());
ui->progressLabel->setVisible(false);
doRefresh();
connect(Core(), &CutterCore::refreshAll, this, &DecompilerWidget::doRefresh);
connect(Core(), &CutterCore::functionRenamed, this, &DecompilerWidget::doRefresh);
connect(Core(), &CutterCore::varsChanged, this, &DecompilerWidget::doRefresh);
connect(Core(), &CutterCore::functionsChanged, this, &DecompilerWidget::doRefresh);
connect(Core(), &CutterCore::flagsChanged, this, &DecompilerWidget::doRefresh);
connect(Core(), &CutterCore::commentsChanged, this, &DecompilerWidget::refreshIfChanged);
connect(Core(), &CutterCore::instructionChanged, this, &DecompilerWidget::refreshIfChanged);
connect(Core(), &CutterCore::refreshCodeViews, this, &DecompilerWidget::doRefresh);
// 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);
}
DecompilerWidget::~DecompilerWidget() = default;
QString DecompilerWidget::getWidgetType()
{
return "DecompilerWidget";
}
Decompiler *DecompilerWidget::getCurrentDecompiler()
{
return Core()->getDecompilerById(ui->decompilerComboBox->currentData().toString());
}
ut64 DecompilerWidget::offsetForPosition(size_t pos)
{
size_t closestPos = SIZE_MAX;
ut64 closestOffset = mCtxMenu->getFirstOffsetInLine();
void *iter;
2021-01-24 14:50:13 +00:00
rz_vector_foreach(&code->annotations, iter)
{
2020-10-28 12:28:04 +00:00
RzCodeAnnotation *annotation = (RzCodeAnnotation *)iter;
if (annotation->type != RZ_CODE_ANNOTATION_TYPE_OFFSET || annotation->start > pos
2021-01-24 14:50:13 +00:00
|| annotation->end <= pos) {
continue;
}
if (closestPos != SIZE_MAX && closestPos >= annotation->start) {
continue;
}
closestPos = annotation->start;
closestOffset = annotation->offset.offset;
}
return closestOffset;
}
size_t DecompilerWidget::positionForOffset(ut64 offset)
{
size_t closestPos = SIZE_MAX;
ut64 closestOffset = UT64_MAX;
void *iter;
2021-01-24 14:50:13 +00:00
rz_vector_foreach(&code->annotations, iter)
{
2020-10-28 12:28:04 +00:00
RzCodeAnnotation *annotation = (RzCodeAnnotation *)iter;
2021-01-24 14:50:13 +00:00
if (annotation->type != RZ_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;
}
void DecompilerWidget::updateBreakpoints(RVA addr)
{
if (!addressInRange(addr)) {
return;
}
setInfoForBreakpoints();
QTextCursor cursor = ui->textEdit->textCursor();
cursor.select(QTextCursor::Document);
cursor.setCharFormat(QTextCharFormat());
cursor.setBlockFormat(QTextBlockFormat());
ui->textEdit->setExtraSelections({});
highlightPC();
highlightBreakpoints();
updateSelection();
}
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);
}
2020-10-28 12:28:04 +00:00
void DecompilerWidget::gatherBreakpointInfo(RzAnnotatedCode &codeDecompiled, size_t startPos,
size_t endPos)
{
RVA firstOffset = RVA_MAX;
void *iter;
2021-01-24 14:50:13 +00:00
rz_vector_foreach(&codeDecompiled.annotations, iter)
{
2020-10-28 12:28:04 +00:00
RzCodeAnnotation *annotation = (RzCodeAnnotation *)iter;
if (annotation->type != RZ_CODE_ANNOTATION_TYPE_OFFSET) {
continue;
}
2021-01-24 14:50:13 +00:00
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 (RVA bpOffset : functionBreakpoints) {
size_t pos = positionForOffset(bpOffset);
if (startPos <= pos && pos <= endPos) {
offsetList.push_back(bpOffset);
}
}
std::sort(offsetList.begin(), offsetList.end());
mCtxMenu->setAvailableBreakpoints(offsetList);
}
void DecompilerWidget::refreshIfChanged(RVA addr)
{
if (addressInRange(addr)) {
doRefresh();
}
}
void DecompilerWidget::doRefresh()
{
RVA addr = seekable->getOffset();
if (!refreshDeferrer->attemptRefresh(nullptr)) {
return;
}
if (ui->decompilerComboBox->currentIndex() < 0) {
return;
}
Decompiler *dec = getCurrentDecompiler();
if (!dec) {
return;
}
2021-01-24 14:50:13 +00:00
// Disabling decompiler selection combo box and making progress label visible ahead of
// decompilation.
ui->progressLabel->setVisible(true);
ui->decompilerComboBox->setEnabled(false);
if (dec->isRunning()) {
if (!decompilerBusy) {
connect(dec, &Decompiler::finished, this, &DecompilerWidget::doRefresh);
}
return;
}
disconnect(dec, &Decompiler::finished, this, &DecompilerWidget::doRefresh);
// Clear all selections since we just refreshed
ui->textEdit->setExtraSelections({});
previousFunctionAddr = decompiledFunctionAddr;
decompiledFunctionAddr = Core()->getFunctionStart(addr);
updateWindowTitle();
if (decompiledFunctionAddr == RVA_INVALID) {
// No function was found, so making the progress label invisible and enabling
// the decompiler selection combo box as we are not waiting for any decompilation to finish.
ui->progressLabel->setVisible(false);
ui->decompilerComboBox->setEnabled(true);
setCode(Decompiler::makeWarning(
2021-01-24 14:50:13 +00:00
tr("No function found at this offset. "
"Seek to a function or define one in order to decompile it.")));
return;
}
mCtxMenu->setDecompiledFunctionAddress(decompiledFunctionAddr);
connect(dec, &Decompiler::finished, this, &DecompilerWidget::decompilationFinished);
decompilerBusy = true;
dec->decompileAt(addr);
}
void DecompilerWidget::refreshDecompiler()
{
doRefresh();
setInfoForBreakpoints();
}
QTextCursor DecompilerWidget::getCursorForAddress(RVA addr)
{
size_t pos = positionForOffset(addr);
if (pos == SIZE_MAX || pos == 0) {
return QTextCursor();
}
QTextCursor cursor = ui->textEdit->textCursor();
cursor.setPosition(pos);
return cursor;
}
2020-10-28 12:28:04 +00:00
void DecompilerWidget::decompilationFinished(RzAnnotatedCode *codeDecompiled)
{
bool isDisplayReset = false;
if (previousFunctionAddr == decompiledFunctionAddr) {
scrollerHorizontal = ui->textEdit->horizontalScrollBar()->sliderPosition();
scrollerVertical = ui->textEdit->verticalScrollBar()->sliderPosition();
isDisplayReset = true;
}
ui->progressLabel->setVisible(false);
ui->decompilerComboBox->setEnabled(decompilerSelectionEnabled);
mCtxMenu->setAnnotationHere(nullptr);
setCode(codeDecompiled);
Decompiler *dec = getCurrentDecompiler();
QObject::disconnect(dec, &Decompiler::finished, this, &DecompilerWidget::decompilationFinished);
decompilerBusy = false;
if (ui->textEdit->toPlainText().isEmpty()) {
setCode(Decompiler::makeWarning(tr("Cannot decompile at this address (Not a function?)")));
lowestOffsetInCode = RVA_MAX;
highestOffsetInCode = 0;
return;
} else {
updateCursorPosition();
highlightPC();
highlightBreakpoints();
lowestOffsetInCode = RVA_MAX;
highestOffsetInCode = 0;
void *iter;
2021-01-24 14:50:13 +00:00
rz_vector_foreach(&code->annotations, iter)
{
2020-10-28 12:28:04 +00:00
RzCodeAnnotation *annotation = (RzCodeAnnotation *)iter;
if (annotation->type == RZ_CODE_ANNOTATION_TYPE_OFFSET) {
if (lowestOffsetInCode > annotation->offset.offset) {
lowestOffsetInCode = annotation->offset.offset;
}
if (highestOffsetInCode < annotation->offset.offset) {
highestOffsetInCode = annotation->offset.offset;
}
}
}
}
if (isDisplayReset) {
ui->textEdit->horizontalScrollBar()->setSliderPosition(scrollerHorizontal);
ui->textEdit->verticalScrollBar()->setSliderPosition(scrollerVertical);
}
}
void DecompilerWidget::setAnnotationsAtCursor(size_t pos)
{
2020-10-28 12:28:04 +00:00
RzCodeAnnotation *annotationAtPos = nullptr;
void *iter;
2021-01-24 14:50:13 +00:00
rz_vector_foreach(&this->code->annotations, iter)
{
2020-10-28 12:28:04 +00:00
RzCodeAnnotation *annotation = (RzCodeAnnotation *)iter;
2021-01-24 14:50:13 +00:00
if (annotation->type == RZ_CODE_ANNOTATION_TYPE_OFFSET
|| annotation->type == RZ_CODE_ANNOTATION_TYPE_SYNTAX_HIGHLIGHT
|| annotation->start > pos || annotation->end <= pos) {
continue;
}
annotationAtPos = annotation;
break;
}
mCtxMenu->setAnnotationHere(annotationAtPos);
}
void DecompilerWidget::decompilerSelected()
2019-07-16 18:33:05 +00:00
{
Config()->setSelectedDecompiler(ui->decompilerComboBox->currentData().toString());
doRefresh();
2019-07-16 18:33:05 +00:00
}
void DecompilerWidget::connectCursorPositionChanged(bool connectPositionChange)
{
if (!connectPositionChange) {
disconnect(ui->textEdit, &QPlainTextEdit::cursorPositionChanged, this,
&DecompilerWidget::cursorPositionChanged);
} else {
connect(ui->textEdit, &QPlainTextEdit::cursorPositionChanged, this,
&DecompilerWidget::cursorPositionChanged);
}
}
void DecompilerWidget::cursorPositionChanged()
{
// Do not perform seeks along with the cursor while selecting multiple lines
if (!ui->textEdit->textCursor().selectedText().isEmpty()) {
return;
}
size_t pos = ui->textEdit->textCursor().position();
setAnnotationsAtCursor(pos);
setInfoForBreakpoints();
RVA offset = offsetForPosition(pos);
if (offset != RVA_INVALID && offset != seekable->getOffset()) {
seekFromCursor = true;
seekable->seek(offset);
mCtxMenu->setOffset(offset);
seekFromCursor = false;
}
updateSelection();
}
void DecompilerWidget::seekChanged()
{
if (seekFromCursor) {
return;
}
RVA fcnAddr = Core()->getFunctionStart(seekable->getOffset());
if (fcnAddr == RVA_INVALID || fcnAddr != decompiledFunctionAddr) {
doRefresh();
return;
}
updateCursorPosition();
}
void DecompilerWidget::updateCursorPosition()
{
RVA offset = seekable->getOffset();
size_t pos = positionForOffset(offset);
if (pos == SIZE_MAX) {
return;
}
mCtxMenu->setOffset(offset);
connectCursorPositionChanged(false);
QTextCursor cursor = ui->textEdit->textCursor();
cursor.setPosition(pos);
ui->textEdit->setTextCursor(cursor);
updateSelection();
connectCursorPositionChanged(true);
}
void DecompilerWidget::setupFonts()
{
ui->textEdit->setFont(Config()->getFont());
}
void DecompilerWidget::updateSelection()
{
QList<QTextEdit::ExtraSelection> extraSelections;
// Highlight the current line
QTextCursor cursor = ui->textEdit->textCursor();
extraSelections.append(createLineHighlightSelection(cursor));
// Highlight all the words in the document same as the current one
cursor.select(QTextCursor::WordUnderCursor);
QString searchString = cursor.selectedText();
mCtxMenu->setCurHighlightedWord(searchString);
extraSelections.append(createSameWordsSelections(ui->textEdit, searchString));
ui->textEdit->setExtraSelections(extraSelections);
// Highlight PC after updating the selected line
highlightPC();
}
QString DecompilerWidget::getWindowTitle() const
{
2020-12-07 07:57:11 +00:00
RzAnalysisFunction *fcn = Core()->functionAt(decompiledFunctionAddr);
QString windowTitle = tr("Decompiler");
if (fcn != NULL) {
windowTitle += " (" + QString(fcn->name) + ")";
} else {
windowTitle += " (Empty)";
}
return windowTitle;
}
void DecompilerWidget::fontsUpdatedSlot()
{
setupFonts();
}
void DecompilerWidget::colorsUpdatedSlot()
{
bool useAnotationHiglighter = Config()->isDecompilerAnnotationHighlighterEnabled();
if (useAnotationHiglighter != usingAnnotationBasedHighlighting) {
setHighlighter(useAnotationHiglighter);
}
}
void DecompilerWidget::showDecompilerContextMenu(const QPoint &pt)
{
mCtxMenu->exec(ui->textEdit->mapToGlobal(pt));
}
void DecompilerWidget::seekToReference()
{
size_t pos = ui->textEdit->textCursor().position();
RVA offset = offsetForPosition(pos);
seekable->seekToReference(offset);
}
bool DecompilerWidget::eventFilter(QObject *obj, QEvent *event)
{
if (event->type() == QEvent::MouseButtonDblClick
2021-01-24 14:50:13 +00:00
&& (obj == ui->textEdit || obj == ui->textEdit->viewport())) {
QMouseEvent *mouseEvent = static_cast<QMouseEvent *>(event);
ui->textEdit->setTextCursor(ui->textEdit->cursorForPosition(mouseEvent->pos()));
seekToReference();
return true;
}
if (event->type() == QEvent::MouseButtonPress
2021-01-24 14:50:13 +00:00
&& (obj == ui->textEdit || obj == ui->textEdit->viewport())) {
QMouseEvent *mouseEvent = static_cast<QMouseEvent *>(event);
if (mouseEvent->button() == Qt::RightButton && !ui->textEdit->textCursor().hasSelection()) {
ui->textEdit->setTextCursor(ui->textEdit->cursorForPosition(mouseEvent->pos()));
return true;
}
}
return MemoryDockWidget::eventFilter(obj, event);
}
void DecompilerWidget::highlightPC()
{
RVA PCAddress = Core()->getProgramCounterValue();
2021-01-24 14:50:13 +00:00
if (PCAddress == RVA_INVALID
|| (Core()->getFunctionStart(PCAddress) != decompiledFunctionAddr)) {
return;
}
QTextCursor cursor = getCursorForAddress(PCAddress);
if (!cursor.isNull()) {
colorLine(createLineHighlightPC(cursor));
}
}
void DecompilerWidget::highlightBreakpoints()
{
QList<RVA> functionBreakpoints = Core()->getBreakpointsInFunction(decompiledFunctionAddr);
QTextCursor cursor;
for (RVA &bp : functionBreakpoints) {
if (bp == RVA_INVALID) {
2021-01-24 14:50:13 +00:00
continue;
;
}
cursor = getCursorForAddress(bp);
if (!cursor.isNull()) {
2021-01-24 14:50:13 +00:00
// 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;
}
void DecompilerWidget::copy()
{
if (ui->textEdit->textCursor().hasSelection()) {
ui->textEdit->copy();
} else {
QTextCursor cursor = ui->textEdit->textCursor();
QClipboard *clipboard = QApplication::clipboard();
cursor.select(QTextCursor::WordUnderCursor);
if (!cursor.selectedText().isEmpty()) {
clipboard->setText(cursor.selectedText());
} else {
cursor.select(QTextCursor::LineUnderCursor);
clipboard->setText(cursor.selectedText());
}
}
}
bool DecompilerWidget::addressInRange(RVA addr)
{
if (lowestOffsetInCode <= addr && addr <= highestOffsetInCode) {
return true;
}
return false;
}
/**
2021-01-24 14:50:13 +00:00
* Convert annotation ranges from byte offsets in utf8 used by RzAnnotated code to QString QChars
* used by QString and Qt text editor.
2020-10-28 12:28:04 +00:00
* @param code - RzAnnotated code with annotations that need to be modified
* @return Decompiled code
*/
2020-10-28 12:28:04 +00:00
static QString remapAnnotationOffsetsToQString(RzAnnotatedCode &code)
{
QByteArray bytes(code.code);
std::vector<size_t> offsets;
offsets.reserve(bytes.size());
char c;
for (size_t off = 0; c = code.code[off], c; off++) {
if ((c & 0xc0) == 0x80) {
continue;
}
offsets.push_back(off);
}
auto mapPos = [&](size_t pos) {
auto it = std::upper_bound(offsets.begin(), offsets.end(), pos);
if (it != offsets.begin()) {
--it;
}
return it - offsets.begin();
};
void *iter;
2021-01-24 14:50:13 +00:00
rz_vector_foreach(&code.annotations, iter)
{
2020-10-28 12:28:04 +00:00
RzCodeAnnotation *annotation = (RzCodeAnnotation *)iter;
annotation->start = mapPos(annotation->start);
annotation->end = mapPos(annotation->end);
}
return QString::fromUtf8(code.code);
}
2020-10-28 12:28:04 +00:00
void DecompilerWidget::setCode(RzAnnotatedCode *code)
{
connectCursorPositionChanged(false);
2021-01-24 14:50:13 +00:00
if (auto highlighter = qobject_cast<DecompilerHighlighter *>(syntaxHighlighter.get())) {
highlighter->setAnnotations(code);
}
this->code.reset(code);
QString text = remapAnnotationOffsetsToQString(*this->code);
this->ui->textEdit->setPlainText(text);
connectCursorPositionChanged(true);
syntaxHighlighter->rehighlight();
}
void DecompilerWidget::setHighlighter(bool annotationBasedHighlighter)
{
usingAnnotationBasedHighlighting = annotationBasedHighlighter;
if (usingAnnotationBasedHighlighting) {
syntaxHighlighter.reset(new DecompilerHighlighter());
2021-01-24 14:50:13 +00:00
static_cast<DecompilerHighlighter *>(syntaxHighlighter.get())->setAnnotations(code.get());
} else {
syntaxHighlighter.reset(Config()->createSyntaxHighlighter(nullptr));
}
syntaxHighlighter->setDocument(ui->textEdit->document());
}