From 330795a650acff44af000ad7f784f80e6a7f08d1 Mon Sep 17 00:00:00 2001 From: Itay Cohen Date: Sun, 4 Aug 2019 00:58:41 +0300 Subject: [PATCH] Improve InitialOptionsDialog and modify UI terminology for analysis (#1669) * Define CommandDescription struct * Refactor InitialOptionsDialog to to work with CommandDescription * Clean InitialOptionsDialog UI --- src/CutterApplication.cpp | 4 +- src/common/AnalTask.cpp | 8 +- src/common/InitialOptions.h | 12 +- src/dialogs/InitialOptionsDialog.cpp | 120 +++++++++---------- src/dialogs/InitialOptionsDialog.h | 10 +- src/dialogs/InitialOptionsDialog.ui | 168 +++------------------------ 6 files changed, 97 insertions(+), 225 deletions(-) diff --git a/src/CutterApplication.cpp b/src/CutterApplication.cpp index 4dd49aa4..df6c6371 100644 --- a/src/CutterApplication.cpp +++ b/src/CutterApplication.cpp @@ -173,10 +173,10 @@ CutterApplication::CutterApplication(int &argc, char **argv) : QApplication(argc options.analCmd = {}; break; case 1: - options.analCmd = { "aaa" }; + options.analCmd = { {"aaa", "Auto analysis"} }; break; case 2: - options.analCmd = { "aaaa" }; + options.analCmd = { {"aaaa", "Auto analysis (experimental)"} }; break; } } diff --git a/src/common/AnalTask.cpp b/src/common/AnalTask.cpp index 64089f61..41cccff7 100644 --- a/src/common/AnalTask.cpp +++ b/src/common/AnalTask.cpp @@ -96,13 +96,13 @@ void AnalTask::runTask() Core()->setConfig("prj.simple", true); if (!options.analCmd.empty()) { - log(tr("Analyzing...")); - for (const QString &cmd : options.analCmd) { + log(tr("Executing analysis...")); + for (const CommandDescription &cmd : options.analCmd) { if (isInterrupted()) { return; } - log(" " + tr("Running") + " " + cmd); - Core()->cmd(cmd); + log(cmd.description); + Core()->cmd(cmd.command); } log(tr("Analysis complete!")); } else { diff --git a/src/common/InitialOptions.h b/src/common/InitialOptions.h index 65a86edf..110810c5 100644 --- a/src/common/InitialOptions.h +++ b/src/common/InitialOptions.h @@ -4,6 +4,14 @@ #include "core/Cutter.h" +/** + * @brief The CommandDescription struct is a pair of a radare2 command and its description + */ +struct CommandDescription { + QString command; + QString description; +}; + struct InitialOptions { enum class Endianness { Auto, Little, Big }; @@ -29,8 +37,8 @@ struct InitialOptions QString pdbFile; QString script; - - QList analCmd = { "aaa" }; + + QList analCmd = { {"aaa", "Auto analysis"} }; QString shellcode; }; diff --git a/src/dialogs/InitialOptionsDialog.cpp b/src/dialogs/InitialOptionsDialog.cpp index 965b52b2..4a3f8043 100644 --- a/src/dialogs/InitialOptionsDialog.cpp +++ b/src/dialogs/InitialOptionsDialog.cpp @@ -50,8 +50,36 @@ InitialOptionsDialog::InitialOptionsDialog(MainWindow *main): ui->formatComboBox->addItem(plugin.name, QVariant::fromValue(plugin)); } + analysisCommands = { + { { "aa", tr("Analyze all symbols") }, new QCheckBox(), true }, + { { "aar", tr("Analyze instructions for references") }, new QCheckBox(), true }, + { { "aac", tr("Analyze function calls") }, new QCheckBox(), true }, + { { "aab", tr("Analyze all basic blocks") }, new QCheckBox(), false }, + { { "aao", tr("Analyze all objc references") }, new QCheckBox(), false }, + { { "avrr", tr("Recover class information from RTTI") }, new QCheckBox(), false }, + { { "aan", tr("Autoname functions based on context") }, new QCheckBox(), false }, + { { "aae", tr("Emulate code to find computed references") }, new QCheckBox(), false }, + { { "aat", tr("Analyze all consecutive functions") }, new QCheckBox(), false }, + { { "aaft", tr("Type and Argument matching analysis") }, new QCheckBox(), false }, + { { "aaT", tr("Analyze code after trap-sleds") }, new QCheckBox(), false }, + { { "aap", tr("Analyze function preludes") }, new QCheckBox(), false }, + { { "e! anal.jmp.tbl", tr("Analyze jump tables in switch statements") }, new QCheckBox(), false }, + { { "e! anal.pushret", tr("Analyze PUSH+RET as JMP") }, new QCheckBox(), false }, + { { "e! anal.hasnext", tr("Continue analysis after each function") }, new QCheckBox(), false }}; + + // Per each checkbox, set a tooltip desccribing it + AnalysisCommands item; + foreach (item, analysisCommands){ + item.checkbox->setText(item.commandDesc.description); + item.checkbox->setToolTip(item.commandDesc.command); + item.checkbox->setChecked(item.checked); + ui->verticalLayout_7->addWidget(item.checkbox); + } + + ui->hideFrame->setVisible(false); ui->analoptionsFrame->setVisible(false); + ui->advancedAnlysisLine->setVisible(false); updatePDBLayout(); @@ -86,30 +114,30 @@ void InitialOptionsDialog::updateCPUComboBox() ui->cpuComboBox->lineEdit()->setText(currentText); } +QList InitialOptionsDialog::getAnalysisCommands(const InitialOptions &options) { + QList commands; + for (auto& commandDesc: options.analCmd) { + commands << commandDesc.command; + } + return commands; +} + void InitialOptionsDialog::loadOptions(const InitialOptions &options) { if (options.analCmd.isEmpty()) { analLevel = 0; - } else if (options.analCmd == QList({ "aaa" })) { + } else if (options.analCmd.first().command == "aaa" ) { analLevel = 1; - } else if (options.analCmd == QList({ "aaaa" })) { + } else if (options.analCmd.first().command == "aaaa" ) { analLevel = 2; } else { analLevel = 3; - // TODO: These checks must always be in sync with getSelectedAdvancedAnalCmds(), which is dangerous - ui->aa_symbols->setChecked(options.analCmd.contains("aa")); - ui->aar_references->setChecked(options.analCmd.contains("aar")); - ui->aac_calls->setChecked(options.analCmd.contains("aac")); - ui->aab_basicblocks->setChecked(options.analCmd.contains("aab")); - ui->aan_rename->setChecked(options.analCmd.contains("aan")); - ui->aae_emulate->setChecked(options.analCmd.contains("aae")); - ui->aat_consecutive->setChecked(options.analCmd.contains("aat")); - ui->afta_typeargument->setChecked(options.analCmd.contains("afta")); - ui->aaT_aftertrap->setChecked(options.analCmd.contains("aaT")); - ui->aap_preludes->setChecked(options.analCmd.contains("aap")); - ui->jmptbl->setChecked(options.analCmd.contains("e! anal.jmptbl")); - ui->pushret->setChecked(options.analCmd.contains("e! anal.pushret")); - ui->hasnext->setChecked(options.analCmd.contains("e! anal.hasnext")); + AnalysisCommands item; + QList commands = getAnalysisCommands(options); + foreach (item, analysisCommands){ + qInfo() << item.commandDesc.command; + item.checkbox->setChecked(commands.contains(item.commandDesc.command)); + } } if (!options.script.isEmpty()) { @@ -171,55 +199,15 @@ QString InitialOptionsDialog::getSelectedOS() const return os.isValid() ? os.toString() : nullptr; } -QList InitialOptionsDialog::getSelectedAdvancedAnalCmds() const +QList InitialOptionsDialog::getSelectedAdvancedAnalCmds() const { - QList advanced = QList(); + QList advanced = QList(); if (ui->analSlider->value() == 3) { - // Enable analysis configurations before executing analysis commands - if (ui->jmptbl->isChecked()) { - advanced << "e! anal.jmptbl"; - } - if (ui->pushret->isChecked()) { - advanced << "e! anal.pushret"; - } - if (ui->hasnext->isChecked()) { - advanced << "e! anal.hasnext"; - } - if (ui->aa_symbols->isChecked()) { - advanced << "aa"; - } - if (ui->aar_references->isChecked()) { - advanced << "aar"; - } - if (ui->aac_calls->isChecked()) { - advanced << "aac"; - } - if (ui->aab_basicblocks->isChecked()) { - advanced << "aab"; - } - if (ui->aao_objc->isChecked()) { - advanced << "aao"; - } - if (ui->avrr_vtables->isChecked()) { - advanced << "avrr"; - } - if (ui->aan_rename->isChecked()) { - advanced << "aan"; - } - if (ui->aae_emulate->isChecked()) { - advanced << "aae"; - } - if (ui->aat_consecutive->isChecked()) { - advanced << "aat"; - } - if (ui->afta_typeargument->isChecked()) { - advanced << "afta"; - } - if (ui->aaT_aftertrap->isChecked()) { - advanced << "aaT"; - } - if (ui->aap_preludes->isChecked()) { - advanced << "aap"; + AnalysisCommands item; + foreach (item, analysisCommands){ + if(item.checkbox->isChecked()) { + advanced << item.commandDesc; + } } } return advanced; @@ -269,10 +257,10 @@ void InitialOptionsDialog::setupAndStartAnalysis(/*int level, QList adv int level = ui->analSlider->value(); switch (level) { case 1: - options.analCmd = { "aaa" }; + options.analCmd = { {"aaa", "Auto analysis"} }; break; case 2: - options.analCmd = { "aaaa" }; + options.analCmd = { {"aaaa", "Auto analysis (experimental}"} }; break; case 3: options.analCmd = getSelectedAdvancedAnalCmds(); @@ -346,8 +334,10 @@ void InitialOptionsDialog::on_analSlider_valueChanged(int value) ui->analCheckBox->setText(tr("Analysis: Enabled")); if (value == 3) { ui->analoptionsFrame->setVisible(true); + ui->advancedAnlysisLine->setVisible(true); } else { ui->analoptionsFrame->setVisible(false); + ui->advancedAnlysisLine->setVisible(false); } } } diff --git a/src/dialogs/InitialOptionsDialog.h b/src/dialogs/InitialOptionsDialog.h index 60534b50..f5c41f34 100644 --- a/src/dialogs/InitialOptionsDialog.h +++ b/src/dialogs/InitialOptionsDialog.h @@ -2,6 +2,7 @@ #define OPTIONSDIALOG_H #include +#include #include #include "common/InitialOptions.h" @@ -49,13 +50,20 @@ private: void updateCPUComboBox(); + struct AnalysisCommands { + CommandDescription commandDesc; + QCheckBox *checkbox; + bool checked; + }; + QList analysisCommands; + QList getAnalysisCommands(const InitialOptions &options); QString getSelectedArch() const; QString getSelectedCPU() const; int getSelectedBits() const; InitialOptions::Endianness getSelectedEndianness() const; QString getSelectedOS() const; - QList getSelectedAdvancedAnalCmds() const; + QList getSelectedAdvancedAnalCmds() const; public: void loadOptions(const InitialOptions &options); diff --git a/src/dialogs/InitialOptionsDialog.ui b/src/dialogs/InitialOptionsDialog.ui index 5b4600a2..60637c19 100644 --- a/src/dialogs/InitialOptionsDialog.ui +++ b/src/dialogs/InitialOptionsDialog.ui @@ -221,8 +221,8 @@ 0 0 - 564 - 867 + 567 + 418 @@ -269,9 +269,12 @@ 0 - + - 5 + 0 + + + QLayout::SetMinimumSize @@ -300,156 +303,19 @@ 5 - - - - Analyze all symbols (aa) - - - true - - - - - - - Analyze for references (aar) - - - true - - - - - - - Analyze function calls (aac) - - - true - - - - - - - Analyze all basic blocks (aab) - - - - - - - Autorename functions based on context (aan) - - - false - - - - - - - Analyze all objc references (aao) - - - false - - - - - - - Recover class information (avrr) - - - false - - - - - - - Experimental: - - - - - - - Emulate code to find computed references (aae) - - - - - - - Analyze for consecutive function (aat) - - - - - - - Type and Argument matching analysis (afta) - - - - - - - Analyze code after trap-sleds (aaT) - - - - - - - Analyze function preludes (aap) - - - - - - - Analyze jump tables in switch statements (e! anal.jmptbl) - - - - - - - Analyze push+ret as jmp (e! anal.pushret) - - - - - - - Continue analysis after each function (e! anal.hasnext) - - - - - - - Qt::Horizontal - - - - - - - - - 0 - - - QLayout::SetMinimumSize - + + + + Qt::Horizontal + + + false + + +