cutter/src/menus/DisassemblyContextMenu.cpp

958 lines
30 KiB
C++
Raw Normal View History

#include "DisassemblyContextMenu.h"
#include "dialogs/preferences/PreferencesDialog.h"
#include "dialogs/EditInstructionDialog.h"
#include "dialogs/CommentsDialog.h"
#include "dialogs/FlagDialog.h"
#include "dialogs/RenameDialog.h"
#include "dialogs/XrefsDialog.h"
#include "dialogs/EditVariablesDialog.h"
2018-08-04 18:05:56 +00:00
#include "dialogs/SetToDataDialog.h"
#include "dialogs/EditFunctionDialog.h"
#include "dialogs/LinkTypeDialog.h"
#include "MainWindow.h"
2019-03-16 12:41:45 +00:00
#include <QtCore>
#include <QShortcut>
#include <QJsonArray>
#include <QClipboard>
#include <QApplication>
2019-03-16 12:41:45 +00:00
#include <QPushButton>
DisassemblyContextMenu::DisassemblyContextMenu(QWidget *parent, MainWindow *mainWindow)
2017-11-19 17:49:29 +00:00
: QMenu(parent),
offset(0),
canCopy(false),
mainWindow(mainWindow)
2018-08-04 18:05:56 +00:00
{
initAction(&actionCopy, tr("Copy"), SLOT(on_actionCopy_triggered()), getCopySequence());
addAction(&actionCopy);
initAction(&actionCopyAddr, tr("Copy address"), SLOT(on_actionCopyAddr_triggered()),
getCopyAddressSequence());
2018-08-04 18:05:56 +00:00
addAction(&actionCopyAddr);
initAction(&showInSubmenu, tr("Show in"), nullptr);
addAction(&showInSubmenu);
copySeparator = addSeparator();
2018-08-04 18:05:56 +00:00
initAction(&actionAddComment, tr("Add Comment"),
SLOT(on_actionAddComment_triggered()), getCommentSequence());
addAction(&actionAddComment);
initAction(&actionAddFlag, tr("Add Flag"),
SLOT(on_actionAddFlag_triggered()), getAddFlagSequence());
addAction(&actionAddFlag);
initAction(&actionRename, tr("Rename"),
SLOT(on_actionRename_triggered()), getRenameSequence());
addAction(&actionRename);
initAction(&actionEditFunction, tr("Edit function"),
SLOT(on_actionEditFunction_triggered()));
addAction(&actionEditFunction);
2018-08-04 18:05:56 +00:00
initAction(&actionRenameUsedHere, tr("Rename Flag/Fcn/Var Used Here"),
SLOT(on_actionRenameUsedHere_triggered()), getRenameUsedHereSequence());
addAction(&actionRenameUsedHere);
initAction(&actionSetFunctionVarTypes, tr("Re-type function local vars"),
SLOT(on_actionSetFunctionVarTypes_triggered()), getRetypeSequence());
addAction(&actionSetFunctionVarTypes);
2018-08-04 18:05:56 +00:00
initAction(&actionDeleteComment, tr("Delete comment"), SLOT(on_actionDeleteComment_triggered()));
addAction(&actionDeleteComment);
initAction(&actionDeleteFlag, tr("Delete flag"), SLOT(on_actionDeleteFlag_triggered()));
addAction(&actionDeleteFlag);
2018-09-26 12:43:37 +00:00
initAction(&actionDeleteFunction, tr("Undefine function"),
SLOT(on_actionDeleteFunction_triggered()));
2018-08-04 18:05:56 +00:00
addAction(&actionDeleteFunction);
2019-06-12 07:08:15 +00:00
initAction(&actionAnalyzeFunction, tr("Define function here"),
2018-09-26 12:43:37 +00:00
SLOT(on_actionAnalyzeFunction_triggered()));
addAction(&actionAnalyzeFunction);
2018-12-03 15:16:17 +00:00
addSeparator();
2018-08-04 18:05:56 +00:00
addSetBaseMenu();
addSetBitsMenu();
structureOffsetMenu = addMenu(tr("Structure offset"));
connect(structureOffsetMenu, SIGNAL(triggered(QAction*)),
this, SLOT(on_actionStructureOffsetMenu_triggered(QAction*)));
initAction(&actionLinkType, tr("Link Type to Address"),
SLOT(on_actionLinkType_triggered()), getLinkTypeSequence());
addAction(&actionLinkType);
2019-06-12 07:08:15 +00:00
addSetAsMenu();
2018-08-04 18:05:56 +00:00
addSeparator();
initAction(&actionXRefs, tr("Show X-Refs"),
SLOT(on_actionXRefs_triggered()), getXRefSequence());
addAction(&actionXRefs);
initAction(&actionDisplayOptions, tr("Show Options"),
SLOT(on_actionDisplayOptions_triggered()), getDisplayOptionsSequence());
addSeparator();
addEditMenu();
addSeparator();
addDebugMenu();
connect(this, &DisassemblyContextMenu::aboutToShow,
this, &DisassemblyContextMenu::aboutToShowSlot);
}
DisassemblyContextMenu::~DisassemblyContextMenu()
{
}
void DisassemblyContextMenu::addSetBaseMenu()
{
setBaseMenu = addMenu(tr("Set Immediate Base to..."));
initAction(&actionSetBaseBinary, tr("Binary"));
2017-11-28 13:50:41 +00:00
setBaseMenu->addAction(&actionSetBaseBinary);
2018-08-04 18:05:56 +00:00
connect(&actionSetBaseBinary, &QAction::triggered, this, [this] { setBase("b"); });
initAction(&actionSetBaseOctal, tr("Octal"));
2017-11-28 13:50:41 +00:00
setBaseMenu->addAction(&actionSetBaseOctal);
2018-08-04 18:05:56 +00:00
connect(&actionSetBaseOctal, &QAction::triggered, this, [this] { setBase("o"); });
initAction(&actionSetBaseDecimal, tr("Decimal"));
2017-11-28 13:50:41 +00:00
setBaseMenu->addAction(&actionSetBaseDecimal);
2018-08-04 18:05:56 +00:00
connect(&actionSetBaseDecimal, &QAction::triggered, this, [this] { setBase("d"); });
initAction(&actionSetBaseHexadecimal, tr("Hexadecimal"));
2017-11-28 13:50:41 +00:00
setBaseMenu->addAction(&actionSetBaseHexadecimal);
2018-08-04 18:05:56 +00:00
connect(&actionSetBaseHexadecimal, &QAction::triggered, this, [this] { setBase("h"); });
initAction(&actionSetBasePort, tr("Network Port"));
2017-11-28 13:50:41 +00:00
setBaseMenu->addAction(&actionSetBasePort);
2018-08-04 18:05:56 +00:00
connect(&actionSetBasePort, &QAction::triggered, this, [this] { setBase("p"); });
initAction(&actionSetBaseIPAddr, tr("IP Address"));
2017-11-28 13:50:41 +00:00
setBaseMenu->addAction(&actionSetBaseIPAddr);
2018-08-04 18:05:56 +00:00
connect(&actionSetBaseIPAddr, &QAction::triggered, this, [this] { setBase("i"); });
initAction(&actionSetBaseSyscall, tr("Syscall"));
2017-11-28 13:50:41 +00:00
setBaseMenu->addAction(&actionSetBaseSyscall);
2018-08-04 18:05:56 +00:00
connect(&actionSetBaseSyscall, &QAction::triggered, this, [this] { setBase("S"); });
initAction(&actionSetBaseString, tr("String"));
2017-11-28 13:50:41 +00:00
setBaseMenu->addAction(&actionSetBaseString);
2018-08-04 18:05:56 +00:00
connect(&actionSetBaseString, &QAction::triggered, this, [this] { setBase("s"); });
}
void DisassemblyContextMenu::addSetBitsMenu()
{
setBitsMenu = addMenu(tr("Set current bits to..."));
2018-02-12 09:48:06 +00:00
2018-08-04 18:05:56 +00:00
initAction(&actionSetBits16, "16");
2018-02-12 09:48:06 +00:00
setBitsMenu->addAction(&actionSetBits16);
2018-08-04 18:05:56 +00:00
connect(&actionSetBits16, &QAction::triggered, this, [this] { setBits(16); });
initAction(&actionSetBits32, "32");
2018-02-12 09:48:06 +00:00
setBitsMenu->addAction(&actionSetBits32);
2018-08-04 18:05:56 +00:00
connect(&actionSetBits32, &QAction::triggered, this, [this] { setBits(32); });
initAction(&actionSetBits64, "64");
2018-02-12 09:48:06 +00:00
setBitsMenu->addAction(&actionSetBits64);
2018-08-04 18:05:56 +00:00
connect(&actionSetBits64, &QAction::triggered, this, [this] { setBits(64); });
}
2018-02-12 09:48:06 +00:00
2019-06-12 07:08:15 +00:00
void DisassemblyContextMenu::addSetAsMenu()
{
setAsMenu = addMenu(tr("Set as..."));
initAction(&actionSetToCode, tr("Code"),
SLOT(on_actionSetToCode_triggered()), getSetToCodeSequence());
setAsMenu->addAction(&actionSetToCode);
initAction(&actionSetAsString, tr("String"),
SLOT(on_actionSetAsString_triggered()), getSetAsStringSequence());
setAsMenu->addAction(&actionSetAsString);
addSetToDataMenu();
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::addSetToDataMenu()
{
2019-06-12 07:08:15 +00:00
setToDataMenu = setAsMenu->addMenu(tr("Data..."));
2018-08-04 18:05:56 +00:00
initAction(&actionSetToDataByte, tr("Byte"));
setToDataMenu->addAction(&actionSetToDataByte);
connect(&actionSetToDataByte, &QAction::triggered, this, [this] { setToData(1); });
initAction(&actionSetToDataWord, tr("Word"));
setToDataMenu->addAction(&actionSetToDataWord);
connect(&actionSetToDataWord, &QAction::triggered, this, [this] { setToData(2); });
initAction(&actionSetToDataDword, tr("Dword"));
setToDataMenu->addAction(&actionSetToDataDword);
connect(&actionSetToDataDword, &QAction::triggered, this, [this] { setToData(4); });
initAction(&actionSetToDataQword, tr("Qword"));
setToDataMenu->addAction(&actionSetToDataQword);
connect(&actionSetToDataQword, &QAction::triggered, this, [this] { setToData(8); });
initAction(&actionSetToDataEx, "...",
SLOT(on_actionSetToDataEx_triggered()), getSetToDataExSequence());
setToDataMenu->addAction(&actionSetToDataEx);
auto switchAction = new QAction();
initAction(switchAction, "Switch Data",
SLOT(on_actionSetToData_triggered()), getSetToDataSequence());
}
void DisassemblyContextMenu::addEditMenu()
{
editMenu = addMenu(tr("Edit"));
initAction(&actionEditInstruction, tr("Instruction"), SLOT(on_actionEditInstruction_triggered()));
editMenu->addAction(&actionEditInstruction);
2018-08-04 18:05:56 +00:00
initAction(&actionNopInstruction, tr("Nop Instruction"), SLOT(on_actionNopInstruction_triggered()));
editMenu->addAction(&actionNopInstruction);
2018-08-04 18:05:56 +00:00
initAction(&actionEditBytes, tr("Bytes"), SLOT(on_actionEditBytes_triggered()));
editMenu->addAction(&actionEditBytes);
2018-06-12 08:43:14 +00:00
2018-08-04 18:05:56 +00:00
initAction(&actionJmpReverse, tr("Reverse Jump"), SLOT(on_actionJmpReverse_triggered()));
editMenu->addAction(&actionJmpReverse);
2017-11-28 13:50:41 +00:00
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::addDebugMenu()
{
2018-08-04 18:05:56 +00:00
debugMenu = addMenu(tr("Debug"));
initAction(&actionAddBreakpoint, tr("Add/remove breakpoint"),
SLOT(on_actionAddBreakpoint_triggered()), getAddBPSequence());
debugMenu->addAction(&actionAddBreakpoint);
initAction(&actionContinueUntil, tr("Continue until line"),
SLOT(on_actionContinueUntil_triggered()));
debugMenu->addAction(&actionContinueUntil);
initAction(&actionSetPC, "Set PC", SLOT(on_actionSetPC_triggered()));
debugMenu->addAction(&actionSetPC);
}
QVector<DisassemblyContextMenu::ThingUsedHere> DisassemblyContextMenu::getThingUsedHere(RVA offset)
{
QVector<ThingUsedHere> result;
const QJsonArray array = Core()->cmdj("anj @ " + QString::number(offset)).array();
result.reserve(array.size());
for (const auto &thing : array) {
auto obj = thing.toObject();
RVA offset = obj["offset"].toVariant().toULongLong();
QString name = obj["name"].toString();
QString typeString = obj["type"].toString();
ThingUsedHere::Type type = ThingUsedHere::Type::Address;
if (typeString == "var") {
type = ThingUsedHere::Type::Var;
} else if (typeString == "flag") {
type = ThingUsedHere::Type::Flag;
} else if (typeString == "function") {
type = ThingUsedHere::Type::Function;
} else if (typeString == "address") {
type = ThingUsedHere::Type::Address;
}
result.push_back(ThingUsedHere{name, offset, type});
}
return result;
}
void DisassemblyContextMenu::updateTargetMenuActions(const QVector<ThingUsedHere> &targets)
{
for (auto action : showTargetMenuActions) {
removeAction(action);
auto menu = action->menu();
if (menu) {
menu->deleteLater();
}
action->deleteLater();
}
showTargetMenuActions.clear();
for (auto &target : targets) {
QString name;
if (target.name.isEmpty()) {
name = tr("%1 (used here)").arg(RAddressString(target.offset));
} else {
name = tr("%1 (%2)").arg(target.name, RAddressString(target.offset));
}
auto action = new QAction(name, this);
showTargetMenuActions.append(action);
auto menu = mainWindow->createShowInMenu(this, target.offset);
action->setMenu(menu);
QAction *copyAddress = new QAction(tr("Copy address"), menu);
RVA offset = target.offset;
connect(copyAddress, &QAction::triggered, copyAddress, [offset]() {
QClipboard *clipboard = QApplication::clipboard();
clipboard->setText(RAddressString(offset));
});
menu->addSeparator();
menu->addAction(copyAddress);
}
insertActions(copySeparator, showTargetMenuActions);
}
2017-11-28 13:50:41 +00:00
void DisassemblyContextMenu::setOffset(RVA offset)
{
this->offset = offset;
this->actionSetFunctionVarTypes.setVisible(true);
2017-11-28 13:50:41 +00:00
}
2017-12-02 15:43:21 +00:00
void DisassemblyContextMenu::setCanCopy(bool enabled)
{
this->canCopy = enabled;
}
void DisassemblyContextMenu::setCurHighlightedWord(const QString &text)
{
this->curHighlightedWord = text;
}
2017-11-28 13:50:41 +00:00
void DisassemblyContextMenu::aboutToShowSlot()
{
// check if set immediate base menu makes sense
2018-03-21 20:32:32 +00:00
QJsonObject instObject = Core()->cmdj("aoj @ " + QString::number(
offset)).array().first().toObject();
2017-11-28 13:50:41 +00:00
auto keys = instObject.keys();
bool immBase = keys.contains("val") || keys.contains("ptr");
2018-08-04 18:05:56 +00:00
setBaseMenu->menuAction()->setVisible(immBase);
setBitsMenu->menuAction()->setVisible(true);
// Create structure offset menu if it makes sense
QString memBaseReg; // Base register
QVariant memDisp; // Displacement
if (instObject.contains("opex") && instObject["opex"].toObject().contains("operands")) {
// Loop through both the operands of the instruction
for (const QJsonValue value: instObject["opex"].toObject()["operands"].toArray()) {
QJsonObject operand = value.toObject();
if (operand.contains("type") && operand["type"].toString() == "mem" &&
operand.contains("base") && !operand["base"].toString().contains("bp") &&
operand.contains("disp") && operand["disp"].toVariant().toLongLong() > 0) {
// The current operand is the one which has an immediate displacement
memBaseReg = operand["base"].toString();
memDisp = operand["disp"].toVariant();
break;
}
}
}
if (memBaseReg.isEmpty()) {
// hide structure offset menu
structureOffsetMenu->menuAction()->setVisible(false);
} else {
// show structure offset menu
structureOffsetMenu->menuAction()->setVisible(true);
structureOffsetMenu->clear();
// Get the possible offsets using the "tas" command
// TODO: add tasj command to radare2 and then use it here
2019-03-23 10:54:34 +00:00
QStringList ret = Core()->cmdList("tas " + memDisp.toString());
for (const QString &val : ret) {
if (val.isEmpty()) {
continue;
}
structureOffsetMenu->addAction("[" + memBaseReg + " + " + val + "]")->setData(val);
}
if (structureOffsetMenu->isEmpty()) {
// No possible offset was found so hide the menu
structureOffsetMenu->menuAction()->setVisible(false);
}
}
actionAnalyzeFunction.setVisible(true);
QString comment = Core()->cmd("CC." + RAddressString(offset));
2018-03-21 20:32:32 +00:00
if (comment.isNull() || comment.isEmpty()) {
actionDeleteComment.setVisible(false);
actionAddComment.setText(tr("Add Comment"));
2018-03-21 20:32:32 +00:00
} else {
actionDeleteComment.setVisible(true);
actionAddComment.setText(tr("Edit Comment"));
}
2017-12-02 15:43:21 +00:00
actionCopy.setVisible(canCopy);
copySeparator->setVisible(canCopy);
RCore *core = Core()->core();
RAnalFunction *fcn = r_anal_get_fcn_at (core->anal, offset, R_ANAL_FCN_TYPE_NULL);
RAnalFunction *in_fcn = Core()->functionAt(offset);
RFlagItem *f = r_flag_get_i (core->flags, offset);
actionDeleteFlag.setVisible(f ? true : false);
actionDeleteFunction.setVisible(fcn ? true : false);
2018-03-21 20:32:32 +00:00
if (fcn) {
actionAnalyzeFunction.setVisible(false);
actionRename.setVisible(true);
actionRename.setText(tr("Rename function \"%1\"").arg(fcn->name));
2018-03-21 20:32:32 +00:00
} else if (f) {
actionRename.setVisible(true);
actionRename.setText(tr("Rename flag \"%1\"").arg(f->name));
2018-03-21 20:32:32 +00:00
} else {
actionRename.setVisible(false);
}
2017-11-30 14:16:39 +00:00
2018-11-05 21:51:27 +00:00
// Only show retype for local vars if in a function
if (in_fcn) {
auto vars = Core()->getVariables(offset);
actionSetFunctionVarTypes.setVisible(!vars.empty());
actionEditFunction.setVisible(true);
actionEditFunction.setText(tr("Edit function \"%1\"").arg(in_fcn->name));
} else {
actionSetFunctionVarTypes.setVisible(false);
actionEditFunction.setVisible(false);
}
2017-11-30 14:16:39 +00:00
2018-11-05 21:51:27 +00:00
// Only show "rename X used here" if there is something to rename
auto thingsUsedHere = getThingUsedHere(offset);
if (!thingsUsedHere.isEmpty()) {
2017-11-30 14:16:39 +00:00
actionRenameUsedHere.setVisible(true);
auto &thingUsedHere = thingsUsedHere.first();
if (thingUsedHere.type == ThingUsedHere::Type::Address) {
RVA offset = thingUsedHere.offset;
2017-11-30 21:18:09 +00:00
actionRenameUsedHere.setText(tr("Add flag at %1 (used here)").arg(RAddressString(offset)));
} else if (thingUsedHere.type == ThingUsedHere::Type::Function) {
actionRenameUsedHere.setText(tr("Rename \"%1\"").arg(thingUsedHere.name));
} else {
actionRenameUsedHere.setText(tr("Rename \"%1\" (used here)").arg(thingUsedHere.name));
2017-11-30 21:18:09 +00:00
}
2018-03-21 20:32:32 +00:00
} else {
2017-11-30 14:16:39 +00:00
actionRenameUsedHere.setVisible(false);
}
updateTargetMenuActions(thingsUsedHere);
2018-11-05 21:51:27 +00:00
// Decide to show Reverse jmp option
showReverseJmpQuery();
if (showInSubmenu.menu() != nullptr) {
showInSubmenu.menu()->deleteLater();
}
showInSubmenu.setMenu(mainWindow->createShowInMenu(this, offset));
2018-11-05 21:51:27 +00:00
// Only show debug options if we are currently debugging
2018-08-04 18:05:56 +00:00
debugMenu->menuAction()->setVisible(Core()->currentlyDebugging);
QString progCounterName = Core()->getRegisterName("PC");
actionSetPC.setText("Set " + progCounterName + " here");
2017-12-02 15:43:21 +00:00
}
QKeySequence DisassemblyContextMenu::getCopySequence() const
{
return QKeySequence::Copy;
2017-11-28 13:50:41 +00:00
}
QKeySequence DisassemblyContextMenu::getCommentSequence() const
{
2018-01-16 14:29:33 +00:00
return {Qt::Key_Semicolon};
2017-11-28 13:50:41 +00:00
}
QKeySequence DisassemblyContextMenu::getCopyAddressSequence() const
{
return {Qt::CTRL + Qt::SHIFT + Qt::Key_C};
}
2018-08-04 18:05:56 +00:00
QKeySequence DisassemblyContextMenu::getSetToCodeSequence() const
{
return {Qt::Key_C};
}
2018-11-16 21:27:07 +00:00
QKeySequence DisassemblyContextMenu::getSetAsStringSequence() const
{
return {Qt::Key_A};
}
2018-08-04 18:05:56 +00:00
QKeySequence DisassemblyContextMenu::getSetToDataSequence() const
{
return {Qt::Key_D};
}
QKeySequence DisassemblyContextMenu::getSetToDataExSequence() const
{
return {Qt::Key_Asterisk};
}
2017-11-28 13:50:41 +00:00
QKeySequence DisassemblyContextMenu::getAddFlagSequence() const
{
return {}; //TODO insert correct sequence
}
QKeySequence DisassemblyContextMenu::getRenameSequence() const
{
return {Qt::Key_N};
}
2017-11-30 14:16:39 +00:00
QKeySequence DisassemblyContextMenu::getRenameUsedHereSequence() const
{
return {Qt::SHIFT + Qt::Key_N};
}
QKeySequence DisassemblyContextMenu::getRetypeSequence() const
{
return {Qt::Key_Y};
}
2017-11-28 13:50:41 +00:00
QKeySequence DisassemblyContextMenu::getXRefSequence() const
{
return {Qt::Key_X};
}
QKeySequence DisassemblyContextMenu::getDisplayOptionsSequence() const
{
return {}; //TODO insert correct sequence
}
QKeySequence DisassemblyContextMenu::getLinkTypeSequence() const
{
return {Qt::Key_L};
}
QList<QKeySequence> DisassemblyContextMenu::getAddBPSequence() const
{
return {Qt::Key_F2, Qt::CTRL + Qt::Key_B};
}
void DisassemblyContextMenu::on_actionEditInstruction_triggered()
{
EditInstructionDialog e(EDIT_TEXT, this);
e.setWindowTitle(tr("Edit Instruction at %1").arg(RAddressString(offset)));
QString oldInstructionOpcode = Core()->getInstructionOpcode(offset);
QString oldInstructionBytes = Core()->getInstructionBytes(offset);
e.setInstruction(oldInstructionOpcode);
if (e.exec()) {
QString userInstructionOpcode = e.getInstruction();
if (userInstructionOpcode != oldInstructionOpcode) {
Core()->editInstruction(offset, userInstructionOpcode);
2018-11-05 21:51:27 +00:00
// Check if the write failed
auto newInstructionBytes = Core()->getInstructionBytes(offset);
if (newInstructionBytes == oldInstructionBytes) {
if (!writeFailed()) {
Core()->editInstruction(offset, userInstructionOpcode);
}
}
}
}
}
void DisassemblyContextMenu::on_actionNopInstruction_triggered()
{
QString oldBytes = Core()->getInstructionBytes(offset);
Core()->nopInstruction(offset);
QString newBytes = Core()->getInstructionBytes(offset);
if (oldBytes == newBytes) {
if (!writeFailed()) {
Core()->nopInstruction(offset);
}
}
}
void DisassemblyContextMenu::showReverseJmpQuery()
{
QString type;
QJsonArray array = Core()->cmdj("pdj 1 @ " + RAddressString(offset)).array();
2018-03-21 20:32:32 +00:00
if (array.isEmpty()) {
return;
}
type = array.first().toObject()["type"].toString();
2018-03-21 20:32:32 +00:00
if (type == "cjmp") {
actionJmpReverse.setVisible(true);
2018-03-21 20:32:32 +00:00
} else {
actionJmpReverse.setVisible(false);
}
}
void DisassemblyContextMenu::on_actionJmpReverse_triggered()
{
QString oldBytes = Core()->getInstructionBytes(offset);
Core()->jmpReverse(offset);
QString newBytes = Core()->getInstructionBytes(offset);
if (oldBytes == newBytes) {
if (!writeFailed()) {
Core()->jmpReverse(offset);
}
}
}
void DisassemblyContextMenu::on_actionEditBytes_triggered()
{
EditInstructionDialog e(EDIT_BYTES, this);
e.setWindowTitle(tr("Edit Bytes at %1").arg(RAddressString(offset)));
QString oldBytes = Core()->getInstructionBytes(offset);
e.setInstruction(oldBytes);
if (e.exec()) {
QString bytes = e.getInstruction();
2018-03-21 20:32:32 +00:00
if (bytes != oldBytes) {
Core()->editBytes(offset, bytes);
QString newBytes = Core()->getInstructionBytes(offset);
if (oldBytes == newBytes) {
if (!writeFailed()) {
Core()->editBytes(offset, bytes);
}
}
}
}
}
bool DisassemblyContextMenu::writeFailed()
{
QMessageBox msgBox;
msgBox.setIcon(QMessageBox::Icon::Critical);
msgBox.setWindowTitle(tr("Write error"));
msgBox.setText(
tr("Unable to complete write operation. Consider opening in write mode. \n\nWARNING: In write mode any changes will be commited to disk"));
msgBox.addButton(tr("OK"), QMessageBox::NoRole);
QAbstractButton *reopenButton = msgBox.addButton(tr("Reopen in write mode and try again"),
QMessageBox::YesRole);
msgBox.exec();
if (msgBox.clickedButton() == reopenButton) {
Core()->cmd("oo+");
return false;
}
return true;
}
2017-12-02 15:43:21 +00:00
void DisassemblyContextMenu::on_actionCopy_triggered()
{
emit copy();
}
void DisassemblyContextMenu::on_actionCopyAddr_triggered()
{
QClipboard *clipboard = QApplication::clipboard();
clipboard->setText(RAddressString(offset));
}
2018-06-12 08:43:14 +00:00
void DisassemblyContextMenu::on_actionAddBreakpoint_triggered()
{
Core()->toggleBreakpoint(offset);
2018-06-12 08:43:14 +00:00
}
void DisassemblyContextMenu::on_actionContinueUntil_triggered()
{
Core()->continueUntilDebug(RAddressString(offset));
}
void DisassemblyContextMenu::on_actionSetPC_triggered()
{
QString progCounterName = Core()->getRegisterName("PC");
Core()->setRegister(progCounterName, RAddressString(offset));
}
void DisassemblyContextMenu::on_actionAddComment_triggered()
{
CommentsDialog::addOrEditComment(offset, this);
}
void DisassemblyContextMenu::on_actionAnalyzeFunction_triggered()
{
2019-09-05 08:22:05 +00:00
RenameDialog dialog(mainWindow);
dialog.setWindowTitle(tr("Analyze function at %1").arg(RAddressString(offset)));
dialog.setPlaceholderText(tr("Function name"));
if (dialog.exec()) {
QString function_name = dialog.getName();
Core()->createFunctionAt(offset, function_name);
}
}
void DisassemblyContextMenu::on_actionAddFlag_triggered()
{
FlagDialog dialog(offset, this->parentWidget());
dialog.exec();
}
void DisassemblyContextMenu::on_actionRename_triggered()
{
2017-11-26 16:53:05 +00:00
RCore *core = Core()->core();
2019-09-05 08:22:05 +00:00
RenameDialog dialog(mainWindow);
2017-11-26 16:53:05 +00:00
RAnalFunction *fcn = r_anal_get_fcn_at (core->anal, offset, R_ANAL_FCN_TYPE_NULL);
RFlagItem *f = r_flag_get_i (core->flags, offset);
2018-03-21 20:32:32 +00:00
if (fcn) {
/* Rename function */
dialog.setWindowTitle(tr("Rename function %1").arg(fcn->name));
dialog.setName(fcn->name);
if (dialog.exec()) {
QString new_name = dialog.getName();
Core()->renameFunction(fcn->name, new_name);
2017-11-26 16:53:05 +00:00
}
2018-03-21 20:32:32 +00:00
} else if (f) {
/* Rename current flag */
dialog.setWindowTitle(tr("Rename flag %1").arg(f->name));
dialog.setName(f->name);
if (dialog.exec()) {
QString new_name = dialog.getName();
Core()->renameFlag(f->name, new_name);
2017-11-26 16:53:05 +00:00
}
2018-03-21 20:32:32 +00:00
} else {
return;
}
}
2017-11-30 14:16:39 +00:00
void DisassemblyContextMenu::on_actionRenameUsedHere_triggered()
{
auto array = getThingUsedHere(offset);
2018-03-21 20:32:32 +00:00
if (array.isEmpty()) {
2017-11-30 14:16:39 +00:00
return;
}
auto thingUsedHere = array.first();
2017-11-30 21:18:09 +00:00
2019-09-05 08:22:05 +00:00
RenameDialog dialog(mainWindow);
2017-11-30 21:18:09 +00:00
QString oldName;
auto type = thingUsedHere.type;
if (type == ThingUsedHere::Type::Address) {
RVA offset = thingUsedHere.offset;
dialog.setWindowTitle(tr("Add flag at %1").arg(RAddressString(offset)));
dialog.setName("label." + QString::number(offset, 16));
2018-03-21 20:32:32 +00:00
} else {
oldName = thingUsedHere.name;
dialog.setWindowTitle(tr("Rename %1").arg(oldName));
dialog.setName(oldName);
2017-11-30 21:18:09 +00:00
}
if (dialog.exec()) {
QString newName = dialog.getName().trimmed();
2018-03-21 20:32:32 +00:00
if (!newName.isEmpty()) {
2017-11-30 21:18:09 +00:00
Core()->cmd("an " + newName + " @ " + QString::number(offset));
if (type == ThingUsedHere::Type::Address || type == ThingUsedHere::Type::Address) {
Core()->triggerFlagsChanged();
} else if (type == ThingUsedHere::Type::Var) {
Core()->triggerVarsChanged();
} else if (type == ThingUsedHere::Type::Function) {
Core()->triggerFunctionRenamed(oldName, newName);
}
2017-11-30 21:18:09 +00:00
}
2017-11-30 14:16:39 +00:00
}
}
void DisassemblyContextMenu::on_actionSetFunctionVarTypes_triggered()
{
RAnalFunction *fcn = Core()->functionAt(offset);
if (!fcn) {
QMessageBox::critical(this, tr("Re-type function local vars"),
tr("You must be in a function to define variable types."));
return;
}
EditVariablesDialog dialog(Core()->getOffset(), curHighlightedWord, this);
if (dialog.empty()) { // don't show the dialog if there are no variables
return;
}
dialog.exec();
}
void DisassemblyContextMenu::on_actionXRefs_triggered()
{
XrefsDialog dialog(mainWindow, nullptr);
dialog.fillRefsForAddress(offset, RAddressString(offset), false);
dialog.exec();
}
void DisassemblyContextMenu::on_actionDisplayOptions_triggered()
{
PreferencesDialog dialog(this->window());
dialog.showSection(PreferencesDialog::Section::Disassembly);
dialog.exec();
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::on_actionSetToCode_triggered()
{
2018-08-04 18:05:56 +00:00
Core()->setToCode(offset);
}
2018-11-16 21:27:07 +00:00
void DisassemblyContextMenu::on_actionSetAsString_triggered()
{
Core()->setAsString(offset);
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::on_actionSetToData_triggered()
{
2018-08-04 18:05:56 +00:00
int size = Core()->sizeofDataMeta(offset);
if (size > 8 || (size && (size & (size - 1)))) {
return;
}
if (size == 0 || size == 8) {
size = 1;
} else {
size *= 2;
}
setToData(size);
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::on_actionSetToDataEx_triggered()
{
SetToDataDialog dialog(offset, this->window());
if (!dialog.exec()) {
2018-08-04 18:05:56 +00:00
return;
}
setToData(dialog.getItemSize(), dialog.getItemCount());
}
void DisassemblyContextMenu::on_actionStructureOffsetMenu_triggered(QAction *action)
{
Core()->applyStructureOffset(action->data().toString(), offset);
}
void DisassemblyContextMenu::on_actionLinkType_triggered()
{
2019-09-05 08:22:05 +00:00
LinkTypeDialog dialog(mainWindow);
if (!dialog.setDefaultAddress(curHighlightedWord)) {
dialog.setDefaultAddress(RAddressString(offset));
}
dialog.exec();
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::on_actionDeleteComment_triggered()
{
2018-08-04 18:05:56 +00:00
Core()->delComment(offset);
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::on_actionDeleteFlag_triggered()
{
2018-08-04 18:05:56 +00:00
Core()->delFlag(offset);
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::on_actionDeleteFunction_triggered()
{
2018-08-04 18:05:56 +00:00
Core()->delFunction(offset);
}
void DisassemblyContextMenu::on_actionEditFunction_triggered()
{
RCore *core = Core()->core();
2019-09-05 08:22:05 +00:00
EditFunctionDialog dialog(mainWindow);
RAnalFunction *fcn = r_anal_get_fcn_in(core->anal, offset, 0);
if (fcn) {
dialog.setWindowTitle(tr("Edit function %1").arg(fcn->name));
dialog.setNameText(fcn->name);
QString startAddrText = "0x" + QString::number(fcn->addr, 16);
dialog.setStartAddrText(startAddrText);
QString endAddrText = "0x" + QString::number(fcn->addr + fcn->_size, 16);
dialog.setEndAddrText(endAddrText);
QString stackSizeText;
stackSizeText.sprintf("%d", fcn->stack);
dialog.setStackSizeText(stackSizeText);
QStringList callConList = Core()->cmd("afcl").split("\n");
callConList.removeLast();
dialog.setCallConList(callConList);
dialog.setCallConSelected(fcn->cc);
if (dialog.exec()) {
QString new_name = dialog.getNameText();
Core()->renameFunction(fcn->name, new_name);
QString new_start_addr = dialog.getStartAddrText();
fcn->addr = Core()->math(new_start_addr);
QString new_end_addr = dialog.getEndAddrText();
Core()->cmd("afu " + new_end_addr);
QString new_stack_size = dialog.getStackSizeText();
fcn->stack = int(Core()->math(new_stack_size));
Core()->cmd("afc " + dialog.getCallConSelected());
emit Core()->functionsChanged();
}
}
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::setBase(QString base)
{
2018-08-04 18:05:56 +00:00
Core()->setImmediateBase(base, offset);
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::setBits(int bits)
2018-02-12 09:48:06 +00:00
{
2018-08-04 18:05:56 +00:00
Core()->setCurrentBits(bits, offset);
2018-02-12 09:48:06 +00:00
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::setToData(int size, int repeat)
2018-02-12 09:48:06 +00:00
{
2018-08-04 18:05:56 +00:00
Core()->setToData(offset, size, repeat);
2018-02-12 09:48:06 +00:00
}
2018-08-04 18:05:56 +00:00
QAction *DisassemblyContextMenu::addAnonymousAction(QString name, const char *slot,
QKeySequence keySequence)
2018-02-12 09:48:06 +00:00
{
auto action = new QAction(this);
2018-08-04 18:05:56 +00:00
addAction(action);
anonymousActions.append(action);
initAction(action, name, slot, keySequence);
return action;
2018-02-12 09:48:06 +00:00
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::initAction(QAction *action, QString name, const char *slot)
{
2018-08-04 18:05:56 +00:00
action->setParent(this);
parentWidget()->addAction(action);
2018-08-04 18:05:56 +00:00
action->setText(name);
if (slot) {
connect(action, SIGNAL(triggered(bool)), this, slot);
}
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::initAction(QAction *action, QString name,
const char *slot, QKeySequence keySequence)
{
2018-08-04 18:05:56 +00:00
initAction(action, name, slot);
if (keySequence.isEmpty()) {
return;
}
action->setShortcut(keySequence);
action->setShortcutContext(Qt::WidgetWithChildrenShortcut);
}
2018-08-04 18:05:56 +00:00
void DisassemblyContextMenu::initAction(QAction *action, QString name,
const char *slot, QList<QKeySequence> keySequenceList)
{
2018-08-04 18:05:56 +00:00
initAction(action, name, slot);
if (keySequenceList.empty()) {
return;
}
action->setShortcuts(keySequenceList);
action->setShortcutContext(Qt::WidgetWithChildrenShortcut);
2018-08-04 18:05:56 +00:00
}