cutter/src/CutterApplication.cpp

464 lines
16 KiB
C++
Raw Normal View History

2018-06-23 16:59:23 +00:00
#include "common/PythonManager.h"
#include "common/CrashHandler.h"
#include "CutterApplication.h"
2019-02-09 13:05:06 +00:00
#include "plugins/PluginManager.h"
2019-03-14 09:28:42 +00:00
#include "CutterConfig.h"
#include "common/Decompiler.h"
#include "common/ResourcePaths.h"
#include <QApplication>
#include <QFileOpenEvent>
#include <QEvent>
#include <QMenu>
#include <QMessageBox>
#include <QCommandLineParser>
#include <QTextCodec>
#include <QStringList>
#include <QProcess>
#include <QPluginLoader>
#include <QDir>
2018-10-31 16:07:53 +00:00
#include <QTranslator>
#include <QLibraryInfo>
2019-03-14 09:28:42 +00:00
#include <QFontDatabase>
#ifdef Q_OS_WIN
2021-01-24 14:50:13 +00:00
# include <QtNetwork/QtNetwork>
#endif // Q_OS_WIN
2019-03-14 09:28:42 +00:00
#include <cstdlib>
#if CUTTER_RZGHIDRA_STATIC
2021-01-24 14:50:13 +00:00
# include <RzGhidraDecompiler.h>
2019-08-30 11:41:14 +00:00
#endif
2018-03-21 20:32:32 +00:00
CutterApplication::CutterApplication(int &argc, char **argv) : QApplication(argc, argv)
{
// Setup application information
setApplicationVersion(CUTTER_VERSION_FULL);
setWindowIcon(QIcon(":/img/cutter.svg"));
setAttribute(Qt::AA_UseHighDpiPixmaps);
2019-01-12 19:44:23 +00:00
setLayoutDirection(Qt::LeftToRight);
2021-01-24 14:50:13 +00:00
// WARN!!! Put initialization code below this line. Code above this line is mandatory to be run
// First
#ifdef Q_OS_WIN
// Hack to force Cutter load internet connection related DLL's
QSslSocket s;
s.sslConfiguration();
#endif // Q_OS_WIN
2018-11-01 22:23:01 +00:00
// Load translations
if (!loadTranslations()) {
qWarning() << "Cannot load translations";
2018-10-31 16:07:53 +00:00
}
2018-11-01 22:23:01 +00:00
// Load fonts
int ret = QFontDatabase::addApplicationFont(":/fonts/Anonymous Pro.ttf");
if (ret == -1) {
qWarning() << "Cannot load Anonymous Pro font.";
}
ret = QFontDatabase::addApplicationFont(":/fonts/Inconsolata-Regular.ttf");
if (ret == -1) {
qWarning() << "Cannot load Incosolata-Regular font.";
}
// Set QString codec to UTF-8
QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF-8"));
2021-01-24 14:50:13 +00:00
#if QT_VERSION < QT_VERSION_CHECK(5, 0, 0)
QTextCodec::setCodecForCStrings(QTextCodec::codecForName("UTF-8"));
QTextCodec::setCodecForTr(QTextCodec::codecForName("UTF-8"));
#endif
if (!parseCommandLineOptions()) {
std::exit(1);
}
2020-12-06 18:00:13 +00:00
// Check rizin version
QString rzversion = rz_core_version();
2020-10-28 12:28:04 +00:00
QString localVersion = "" RZ_GITTAP;
2020-12-06 18:00:13 +00:00
if (rzversion != localVersion) {
QMessageBox msg;
msg.setIcon(QMessageBox::Critical);
msg.setStandardButtons(QMessageBox::Yes | QMessageBox::No);
msg.setWindowTitle(QObject::tr("Version mismatch!"));
2021-01-24 14:50:13 +00:00
msg.setText(QString(QObject::tr("The version used to compile Cutter (%1) does not match "
"the binary version of rizin (%2). This could result in "
"unexpected behaviour. Are you sure you want to continue?"))
.arg(localVersion, rzversion));
if (msg.exec() == QMessageBox::No) {
std::exit(1);
}
}
2019-02-13 21:53:52 +00:00
#ifdef CUTTER_ENABLE_PYTHON
2018-06-23 16:59:23 +00:00
// Init python
if (!clOptions.pythonHome.isEmpty()) {
Python()->setPythonHome(clOptions.pythonHome);
2018-03-06 17:21:42 +00:00
}
2018-06-23 16:59:23 +00:00
Python()->initialize();
2019-02-13 21:53:52 +00:00
#endif
2018-06-23 16:59:23 +00:00
2019-09-25 12:36:30 +00:00
#ifdef Q_OS_WIN
2020-10-28 12:28:04 +00:00
// Redefine rz_sys_prefix() behaviour
qputenv("RZ_ALT_SRC_DIR", "1");
2019-09-25 12:36:30 +00:00
#endif
Core()->initialize(clOptions.enableRizinPlugins);
2019-02-20 17:52:11 +00:00
Core()->setSettings();
Config()->loadInitial();
2019-07-11 10:32:56 +00:00
Core()->loadCutterRC();
2018-03-06 17:21:42 +00:00
Config()->setOutputRedirectionEnabled(clOptions.outputRedirectionEnabled);
if (JSDecDecompiler::isAvailable()) {
Core()->registerDecompiler(new JSDecDecompiler(Core()));
}
#if CUTTER_RZGHIDRA_STATIC
Core()->registerDecompiler(new RzGhidraDecompiler(Core()));
2019-08-30 11:41:14 +00:00
#endif
Plugins()->loadPlugins(clOptions.enableCutterPlugins);
for (auto &plugin : Plugins()->getPlugins()) {
2019-07-15 16:07:24 +00:00
plugin->registerDecompilers();
}
2018-02-27 13:06:04 +00:00
mainWindow = new MainWindow();
installEventFilter(mainWindow);
// set up context menu shortcut display fix
#if QT_VERSION_CHECK(5, 10, 0) < QT_VERSION
setStyle(new CutterProxyStyle());
#endif // QT_VERSION_CHECK(5, 10, 0) < QT_VERSION
if (clOptions.args.empty()) {
2019-01-19 20:54:02 +00:00
// check if this is the first execution of Cutter in this computer
2021-01-24 14:50:13 +00:00
// Note: the execution after the preferences been reset, will be considered as
// first-execution
2019-01-19 20:54:02 +00:00
if (Config()->isFirstExecution()) {
mainWindow->displayWelcomeDialog();
}
2018-02-27 13:06:04 +00:00
mainWindow->displayNewFileDialog();
2018-03-21 20:32:32 +00:00
} else { // filename specified as positional argument
bool askOptions = clOptions.analLevel != AutomaticAnalysisLevel::Ask;
mainWindow->openNewFile(clOptions.fileOpenOptions, askOptions);
}
2019-08-30 11:41:14 +00:00
#ifdef APPIMAGE
{
2020-01-24 11:18:18 +00:00
auto appdir = QDir(QCoreApplication::applicationDirPath()); // appdir/bin
appdir.cdUp(); // appdir
auto sleighHome = appdir;
2021-01-24 14:50:13 +00:00
sleighHome.cd(
"share/rizin/plugins/rz_ghidra_sleigh"); // appdir/share/rizin/plugins/rz_ghidra_sleigh
2020-10-28 12:28:04 +00:00
Core()->setConfig("ghidra.sleighhome", sleighHome.absolutePath());
2020-01-24 11:18:18 +00:00
auto jsdecHome = appdir;
jsdecHome.cd("share/rizin/plugins/jsdec"); // appdir/share/rizin/plugins/jsdec
qputenv("JSDEC_HOME", jsdecHome.absolutePath().toLocal8Bit());
2019-08-30 11:41:14 +00:00
}
#endif
#ifdef Q_OS_MACOS
{
2020-10-28 12:28:04 +00:00
auto rzprefix = QDir(QCoreApplication::applicationDirPath()); // Contents/MacOS
rzprefix.cdUp(); // Contents
rzprefix.cd("Resources"); // Contents/Resources/rz
2020-01-24 20:15:34 +00:00
2020-10-28 12:28:04 +00:00
auto sleighHome = rzprefix;
2021-01-24 14:50:13 +00:00
sleighHome.cd(
"share/rizin/plugins/rz_ghidra_sleigh"); // Contents/Resources/rz/share/rizin/plugins/rz_ghidra_sleigh
2020-10-28 12:28:04 +00:00
Core()->setConfig("ghidra.sleighhome", sleighHome.absolutePath());
2020-01-24 20:15:34 +00:00
auto jsdecHome = rzprefix;
2021-01-24 14:50:13 +00:00
jsdecHome.cd(
"share/rizin/plugins/jsdec"); // Contents/Resources/rz/share/rizin/plugins/jsdec
qputenv("JSDEC_HOME", jsdecHome.absolutePath().toLocal8Bit());
2019-08-30 11:41:14 +00:00
}
#endif
#ifdef CUTTER_APPVEYOR_JSDEC
qputenv("JSDEC_HOME", "lib\\plugins\\jsdec");
2020-08-08 15:43:43 +00:00
#endif
2019-08-30 11:41:14 +00:00
#ifdef Q_OS_WIN
{
auto sleighHome = QDir(QCoreApplication::applicationDirPath());
2020-10-28 12:28:04 +00:00
sleighHome.cd("lib/plugins/rz_ghidra_sleigh");
Core()->setConfig("ghidra.sleighhome", sleighHome.absolutePath());
2019-08-30 11:41:14 +00:00
}
#endif
}
2018-02-27 13:06:04 +00:00
CutterApplication::~CutterApplication()
{
2019-02-09 13:05:06 +00:00
Plugins()->destroyPlugins();
2018-02-27 13:06:04 +00:00
delete mainWindow;
2019-02-13 21:53:52 +00:00
#ifdef CUTTER_ENABLE_PYTHON
2019-02-03 13:00:40 +00:00
Python()->shutdown();
2019-02-13 21:53:52 +00:00
#endif
2018-02-27 13:06:04 +00:00
}
void CutterApplication::launchNewInstance(const QStringList &args)
{
QProcess process(this);
process.setEnvironment(QProcess::systemEnvironment());
QStringList allArgs;
if (!clOptions.enableCutterPlugins) {
allArgs.push_back("--no-cutter-plugins");
}
if (!clOptions.enableRizinPlugins) {
allArgs.push_back("--no-rizin-plugins");
}
allArgs.append(args);
process.startDetached(qApp->applicationFilePath(), allArgs);
}
2018-02-27 13:06:04 +00:00
bool CutterApplication::event(QEvent *e)
{
2018-03-21 20:32:32 +00:00
if (e->type() == QEvent::FileOpen) {
QFileOpenEvent *openEvent = static_cast<QFileOpenEvent *>(e);
2018-03-21 20:32:32 +00:00
if (openEvent) {
if (m_FileAlreadyDropped) {
// We already dropped a file in macOS, let's spawn another instance
// (Like the File -> Open)
QString fileName = openEvent->file();
2021-01-24 14:50:13 +00:00
launchNewInstance({ fileName });
2018-03-21 20:32:32 +00:00
} else {
QString fileName = openEvent->file();
m_FileAlreadyDropped = true;
2018-02-27 13:06:04 +00:00
mainWindow->closeNewFileDialog();
InitialOptions options;
options.filename = fileName;
mainWindow->openNewFile(options);
}
}
}
return QApplication::event(e);
}
bool CutterApplication::loadTranslations()
{
const QString &language = Config()->getCurrLocale().bcp47Name();
if (language == QStringLiteral("en") || language.startsWith(QStringLiteral("en-"))) {
return true;
}
2021-01-24 14:50:13 +00:00
const auto &allLocales =
QLocale::matchingLocales(QLocale::AnyLanguage, QLocale::AnyScript, QLocale::AnyCountry);
bool cutterTrLoaded = false;
for (const QLocale &it : allLocales) {
const QString &langPrefix = it.bcp47Name();
if (langPrefix == language) {
QApplication::setLayoutDirection(it.textDirection());
QLocale::setDefault(it);
QTranslator *trCutter = new QTranslator;
QTranslator *trQtBase = new QTranslator;
QTranslator *trQt = new QTranslator;
const QStringList &cutterTrPaths = Cutter::getTranslationsDirectories();
for (const auto &trPath : cutterTrPaths) {
2021-01-24 14:50:13 +00:00
if (trCutter
&& trCutter->load(it, QLatin1String("cutter"), QLatin1String("_"), trPath)) {
installTranslator(trCutter);
cutterTrLoaded = true;
trCutter = nullptr;
}
if (trQt && trQt->load(it, "qt", "_", trPath)) {
installTranslator(trQt);
trQt = nullptr;
}
if (trQtBase && trQtBase->load(it, "qtbase", "_", trPath)) {
installTranslator(trQtBase);
trQtBase = nullptr;
}
}
2019-09-25 12:36:30 +00:00
if (trCutter) {
delete trCutter;
}
if (trQt) {
delete trQt;
}
if (trQtBase) {
delete trQtBase;
}
return true;
}
}
if (!cutterTrLoaded) {
qWarning() << "Cannot load Cutter's translation for " << language;
}
return false;
}
bool CutterApplication::parseCommandLineOptions()
{
// Keep this function in sync with documentation
QCommandLineParser cmd_parser;
cmd_parser.setApplicationDescription(
2021-01-24 14:50:13 +00:00
QObject::tr("A Qt and C++ GUI for rizin reverse engineering framework"));
cmd_parser.addHelpOption();
cmd_parser.addVersionOption();
cmd_parser.addPositionalArgument("filename", QObject::tr("Filename to open."));
2021-01-24 14:50:13 +00:00
QCommandLineOption analOption(
{ "A", "analysis" },
QObject::tr("Automatically open file and optionally start analysis. "
"Needs filename to be specified. May be a value between 0 and 2:"
" 0 = no analysis, 1 = aaa, 2 = aaaa (experimental)"),
QObject::tr("level"));
cmd_parser.addOption(analOption);
2021-01-24 14:50:13 +00:00
QCommandLineOption formatOption({ "F", "format" },
QObject::tr("Force using a specific file format (bin plugin)"),
QObject::tr("name"));
cmd_parser.addOption(formatOption);
2021-01-24 14:50:13 +00:00
QCommandLineOption baddrOption({ "B", "base" },
QObject::tr("Load binary at a specific base address"),
QObject::tr("base address"));
cmd_parser.addOption(baddrOption);
2021-01-24 14:50:13 +00:00
QCommandLineOption scriptOption("i", QObject::tr("Run script file"), QObject::tr("file"));
cmd_parser.addOption(scriptOption);
2021-01-24 14:50:13 +00:00
QCommandLineOption writeModeOption({ "w", "writemode" },
QObject::tr("Open file in write mode"));
cmd_parser.addOption(writeModeOption);
2021-01-24 14:50:13 +00:00
QCommandLineOption pythonHomeOption(
"pythonhome", QObject::tr("PYTHONHOME to use for embedded python interpreter"),
"PYTHONHOME");
cmd_parser.addOption(pythonHomeOption);
2021-01-24 14:50:13 +00:00
QCommandLineOption disableRedirectOption(
"no-output-redirect",
QObject::tr("Disable output redirection."
" Some of the output in console widget will not be visible."
" Use this option when debuging a crash or freeze and output "
" redirection is causing some messages to be lost."));
cmd_parser.addOption(disableRedirectOption);
2021-01-24 14:50:13 +00:00
QCommandLineOption disablePlugins("no-plugins", QObject::tr("Do not load plugins"));
cmd_parser.addOption(disablePlugins);
QCommandLineOption disableCutterPlugins("no-cutter-plugins",
QObject::tr("Do not load Cutter plugins"));
cmd_parser.addOption(disableCutterPlugins);
QCommandLineOption disableRizinPlugins("no-rizin-plugins",
2021-01-24 14:50:13 +00:00
QObject::tr("Do not load rizin plugins"));
cmd_parser.addOption(disableRizinPlugins);
cmd_parser.process(*this);
CutterCommandLineOptions opts;
opts.args = cmd_parser.positionalArguments();
if (cmd_parser.isSet(analOption)) {
bool analLevelSpecified = false;
int analLevel = cmd_parser.value(analOption).toInt(&analLevelSpecified);
if (!analLevelSpecified || analLevel < 0 || analLevel > 2) {
fprintf(stderr, "%s\n",
2021-01-24 14:50:13 +00:00
QObject::tr("Invalid Analysis Level. May be a value between 0 and 2.")
.toLocal8Bit()
.constData());
return false;
}
switch (analLevel) {
case 0:
opts.analLevel = AutomaticAnalysisLevel::None;
break;
case 1:
opts.analLevel = AutomaticAnalysisLevel::AAA;
break;
case 2:
opts.analLevel = AutomaticAnalysisLevel::AAAA;
break;
}
}
if (opts.args.empty() && opts.analLevel != AutomaticAnalysisLevel::Ask) {
fprintf(stderr, "%s\n",
2021-01-24 14:50:13 +00:00
QObject::tr("Filename must be specified to start analysis automatically.")
.toLocal8Bit()
.constData());
return false;
}
InitialOptions options;
if (!opts.args.isEmpty()) {
opts.fileOpenOptions.filename = opts.args[0];
opts.fileOpenOptions.forceBinPlugin = cmd_parser.value(formatOption);
if (cmd_parser.isSet(baddrOption)) {
bool ok;
RVA baddr = cmd_parser.value(baddrOption).toULongLong(&ok, 0);
if (ok) {
options.binLoadAddr = baddr;
}
}
switch (opts.analLevel) {
case AutomaticAnalysisLevel::Ask:
break;
case AutomaticAnalysisLevel::None:
opts.fileOpenOptions.analCmd = {};
break;
case AutomaticAnalysisLevel::AAA:
2021-01-24 14:50:13 +00:00
opts.fileOpenOptions.analCmd = { { "aaa", "Auto analysis" } };
break;
case AutomaticAnalysisLevel::AAAA:
2021-01-24 14:50:13 +00:00
opts.fileOpenOptions.analCmd = { { "aaaa", "Auto analysis (experimental)" } };
break;
}
opts.fileOpenOptions.script = cmd_parser.value(scriptOption);
opts.fileOpenOptions.writeEnabled = cmd_parser.isSet(writeModeOption);
}
if (cmd_parser.isSet(pythonHomeOption)) {
opts.pythonHome = cmd_parser.value(pythonHomeOption);
}
opts.outputRedirectionEnabled = !cmd_parser.isSet(disableRedirectOption);
if (cmd_parser.isSet(disablePlugins)) {
opts.enableCutterPlugins = false;
opts.enableRizinPlugins = false;
}
if (cmd_parser.isSet(disableCutterPlugins)) {
opts.enableCutterPlugins = false;
}
if (cmd_parser.isSet(disableRizinPlugins)) {
opts.enableRizinPlugins = false;
}
this->clOptions = opts;
return true;
}
void CutterProxyStyle::polish(QWidget *widget)
{
QProxyStyle::polish(widget);
#if QT_VERSION_CHECK(5, 10, 0) < QT_VERSION
// HACK: This is the only way I've found to force Qt (5.10 and newer) to
// display shortcuts in context menus on all platforms. It's ugly,
// but it gets the job done.
if (auto menu = qobject_cast<QMenu *>(widget)) {
const auto &actions = menu->actions();
for (auto action : actions) {
action->setShortcutVisibleInContextMenu(true);
}
}
#endif // QT_VERSION_CHECK(5, 10, 0) < QT_VERSION
}