#include "content/views/view_disassembler.hpp" #include "hex/api/content_registry.hpp" #include #include #include #include #include #include using namespace std::literals::string_literals; using namespace wolv::literals; namespace hex::plugin::disasm { ViewDisassembler::ViewDisassembler() : View::Window("hex.disassembler.view.disassembler.name", ICON_VS_FILE_CODE) { EventProviderDeleted::subscribe(this, [this](const auto*) { m_disassembly.clear(); }); ContentRegistry::Interface::addMenuItem({ "hex.builtin.menu.edit", "hex.builtin.menu.edit.disassemble_range" }, ICON_VS_DEBUG_LINE_BY_LINE, 3100, CTRLCMD + SHIFT + Keys::D, [this] { ImGui::SetWindowFocus(this->getName().c_str()); this->getWindowOpenState() = true; m_range = ui::RegionType::Region; m_regionToDisassemble = ImHexApi::HexEditor::getSelection()->getRegion(); this->disassemble(); }, [this]{ return ImHexApi::HexEditor::isSelectionValid() && !this->m_disassemblerTask.isRunning(); }); } ViewDisassembler::~ViewDisassembler() { EventDataChanged::unsubscribe(this); EventRegionSelected::unsubscribe(this); EventProviderDeleted::unsubscribe(this); } void ViewDisassembler::disassemble() { m_disassembly.clear(); if (m_regionToDisassemble.getStartAddress() < m_imageBaseAddress) return; m_disassemblerTask = TaskManager::createTask("hex.disassembler.view.disassembler.disassembling"_lang, m_regionToDisassemble.getSize(), [this](auto &task) { csh capstoneHandle; cs_insn instruction; cs_mode mode = m_mode; // Create a capstone disassembler instance if (cs_open(Disassembler::toCapstoneArchitecture(m_architecture), mode, &capstoneHandle) == CS_ERR_OK) { // Tell capstone to skip data bytes cs_option(capstoneHandle, CS_OPT_SKIPDATA, CS_OPT_ON); auto provider = ImHexApi::Provider::get(); std::vector buffer(1_MiB, 0x00); const u64 codeOffset = m_regionToDisassemble.getStartAddress() - m_imageBaseAddress; // Read the data in chunks and disassemble it u64 instructionLoadAddress = m_imageLoadAddress + codeOffset; u64 instructionDataAddress = m_regionToDisassemble.getStartAddress(); bool hadError = false; while (instructionDataAddress < m_regionToDisassemble.getEndAddress()) { // Read a chunk of data size_t bufferSize = std::min(buffer.size(), (m_regionToDisassemble.getEndAddress() - instructionDataAddress)); provider->read(instructionDataAddress, buffer.data(), bufferSize); // Ask capstone to disassemble the data const u8 *code = buffer.data(); while (cs_disasm_iter(capstoneHandle, &code, &bufferSize, &instructionLoadAddress, &instruction)) { task.update(instructionDataAddress); // Convert the capstone instructions to our disassembly format Disassembly disassembly = { }; disassembly.address = instruction.address; disassembly.offset = instructionDataAddress - m_imageBaseAddress; disassembly.size = instruction.size; disassembly.mnemonic = instruction.mnemonic; disassembly.operators = instruction.op_str; for (u16 j = 0; j < instruction.size; j++) disassembly.bytes += hex::format("{0:02X} ", instruction.bytes[j]); disassembly.bytes.pop_back(); m_disassembly.push_back(disassembly); instructionDataAddress += instruction.size; hadError = false; } if (hadError) break; hadError = true; } cs_close(&capstoneHandle); } }); } void ViewDisassembler::exportToFile() { TaskManager::createTask("hex.ui.common.processing"_lang, TaskManager::NoProgress, [this](auto &) { TaskManager::doLater([this] { fs::openFileBrowser(fs::DialogMode::Save, {}, [this](const std::fs::path &path) { auto p = path; if (p.extension() != ".asm") p.replace_filename(hex::format("{}{}", p.filename().string(), ".asm")); auto file = wolv::io::File(p, wolv::io::File::Mode::Create); if (!file.isValid()) { ui::ToastError::open("hex.disassembler.view.disassembler.export.popup.error"_lang); return; } // As disassembly code can be quite long, we prefer writing each disassembled instruction to file for (const Disassembly& d : m_disassembly) { // We test for a "bugged" case that should never happen - the instruction should always have a mnemonic if (d.mnemonic.empty()) continue; if (d.operators.empty()) file.writeString(hex::format("{}\n", d.mnemonic)); else file.writeString(hex::format("{} {}\n", d.mnemonic, d.operators)); } }); }); }); } void ViewDisassembler::drawContent() { auto provider = ImHexApi::Provider::get(); if (ImHexApi::Provider::isValid() && provider->isReadable()) { ImGuiExt::Header("hex.disassembler.view.disassembler.position"_lang, true); // Draw region selection picker ui::regionSelectionPicker(&m_regionToDisassemble, provider, &m_range); // Draw base address input ImGuiExt::InputHexadecimal("hex.disassembler.view.disassembler.image_load_address"_lang, &m_imageLoadAddress, ImGuiInputTextFlags_CharsHexadecimal); // Draw code region start address input ImGui::BeginDisabled(m_range == ui::RegionType::EntireData); { ImGuiExt::InputHexadecimal("hex.disassembler.view.disassembler.image_base_address"_lang, &m_imageBaseAddress, ImGuiInputTextFlags_CharsHexadecimal); } ImGui::EndDisabled(); // Draw settings { ImGuiExt::Header("hex.ui.common.settings"_lang); // Draw architecture selector if (ImGui::Combo("hex.disassembler.view.disassembler.arch"_lang, reinterpret_cast(&m_architecture), Disassembler::ArchitectureNames.data(), Disassembler::getArchitectureSupportedCount())) m_mode = cs_mode(0); // Draw sub-settings for each architecture if (ImGuiExt::BeginBox()) { // Draw endian radio buttons. This setting is available for all architectures static int littleEndian = true; ImGui::RadioButton("hex.ui.common.little_endian"_lang, &littleEndian, true); ImGui::SameLine(); ImGui::RadioButton("hex.ui.common.big_endian"_lang, &littleEndian, false); ImGui::NewLine(); // Draw architecture specific settings switch (m_architecture) { case Architecture::ARM: { static int mode = CS_MODE_ARM; ImGui::RadioButton("hex.disassembler.view.disassembler.arm.arm"_lang, &mode, CS_MODE_ARM); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.arm.thumb"_lang, &mode, CS_MODE_THUMB); static int extraMode = 0; ImGui::RadioButton("hex.disassembler.view.disassembler.arm.default"_lang, &extraMode, 0); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.arm.cortex_m"_lang, &extraMode, CS_MODE_MCLASS); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.arm.armv8"_lang, &extraMode, CS_MODE_V8); m_mode = cs_mode(mode | extraMode); } break; case Architecture::MIPS: { static int mode = CS_MODE_MIPS32; ImGui::RadioButton("hex.disassembler.view.disassembler.mips.mips32"_lang, &mode, CS_MODE_MIPS32); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.mips.mips64"_lang, &mode, CS_MODE_MIPS64); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.mips.mips32R6"_lang, &mode, CS_MODE_MIPS32R6); ImGui::RadioButton("hex.disassembler.view.disassembler.mips.mips2"_lang, &mode, CS_MODE_MIPS2); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.mips.mips3"_lang, &mode, CS_MODE_MIPS3); static bool microMode; ImGui::Checkbox("hex.disassembler.view.disassembler.mips.micro"_lang, µMode); m_mode = cs_mode(mode | (microMode ? CS_MODE_MICRO : cs_mode(0))); } break; case Architecture::X86: { static int mode = CS_MODE_32; ImGui::RadioButton("hex.disassembler.view.disassembler.16bit"_lang, &mode, CS_MODE_16); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.32bit"_lang, &mode, CS_MODE_32); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.64bit"_lang, &mode, CS_MODE_64); m_mode = cs_mode(mode); } break; case Architecture::PPC: { static int mode = CS_MODE_32; ImGui::RadioButton("hex.disassembler.view.disassembler.32bit"_lang, &mode, CS_MODE_32); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.64bit"_lang, &mode, CS_MODE_64); static bool qpx = false; ImGui::Checkbox("hex.disassembler.view.disassembler.ppc.qpx"_lang, &qpx); #if CS_API_MAJOR >= 5 static bool spe = false; ImGui::Checkbox("hex.disassembler.view.disassembler.ppc.spe"_lang, &spe); static bool booke = false; ImGui::Checkbox("hex.disassembler.view.disassembler.ppc.booke"_lang, &booke); m_mode = cs_mode(mode | (qpx ? CS_MODE_QPX : cs_mode(0)) | (spe ? CS_MODE_SPE : cs_mode(0)) | (booke ? CS_MODE_BOOKE : cs_mode(0))); #else m_mode = cs_mode(mode | (qpx ? CS_MODE_QPX : cs_mode(0))); #endif } break; case Architecture::SPARC: { static bool v9Mode = false; ImGui::Checkbox("hex.disassembler.view.disassembler.sparc.v9"_lang, &v9Mode); m_mode = cs_mode(v9Mode ? CS_MODE_V9 : cs_mode(0)); } break; #if CS_API_MAJOR >= 5 case Architecture::RISCV: { static int mode = CS_MODE_RISCV32; ImGui::RadioButton("hex.disassembler.view.disassembler.32bit"_lang, &mode, CS_MODE_RISCV32); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.64bit"_lang, &mode, CS_MODE_RISCV64); static bool compressed = false; ImGui::Checkbox("hex.disassembler.view.disassembler.riscv.compressed"_lang, &compressed); m_mode = cs_mode(mode | (compressed ? CS_MODE_RISCVC : cs_mode(0))); } break; #endif case Architecture::M68K: { static int selectedMode = 0; std::pair modes[] = { {"hex.disassembler.view.disassembler.m68k.000"_lang, CS_MODE_M68K_000}, { "hex.disassembler.view.disassembler.m68k.010"_lang, CS_MODE_M68K_010}, { "hex.disassembler.view.disassembler.m68k.020"_lang, CS_MODE_M68K_020}, { "hex.disassembler.view.disassembler.m68k.030"_lang, CS_MODE_M68K_030}, { "hex.disassembler.view.disassembler.m68k.040"_lang, CS_MODE_M68K_040}, { "hex.disassembler.view.disassembler.m68k.060"_lang, CS_MODE_M68K_060}, }; if (ImGui::BeginCombo("hex.disassembler.view.disassembler.settings.mode"_lang, modes[selectedMode].first)) { for (u32 i = 0; i < IM_ARRAYSIZE(modes); i++) { if (ImGui::Selectable(modes[i].first)) selectedMode = i; } ImGui::EndCombo(); } m_mode = cs_mode(modes[selectedMode].second); } break; case Architecture::M680X: { static int selectedMode = 0; std::pair modes[] = { {"hex.disassembler.view.disassembler.m680x.6301"_lang, CS_MODE_M680X_6301 }, { "hex.disassembler.view.disassembler.m680x.6309"_lang, CS_MODE_M680X_6309 }, { "hex.disassembler.view.disassembler.m680x.6800"_lang, CS_MODE_M680X_6800 }, { "hex.disassembler.view.disassembler.m680x.6801"_lang, CS_MODE_M680X_6801 }, { "hex.disassembler.view.disassembler.m680x.6805"_lang, CS_MODE_M680X_6805 }, { "hex.disassembler.view.disassembler.m680x.6808"_lang, CS_MODE_M680X_6808 }, { "hex.disassembler.view.disassembler.m680x.6809"_lang, CS_MODE_M680X_6809 }, { "hex.disassembler.view.disassembler.m680x.6811"_lang, CS_MODE_M680X_6811 }, { "hex.disassembler.view.disassembler.m680x.cpu12"_lang, CS_MODE_M680X_CPU12}, { "hex.disassembler.view.disassembler.m680x.hcs08"_lang, CS_MODE_M680X_HCS08}, }; if (ImGui::BeginCombo("hex.disassembler.view.disassembler.settings.mode"_lang, modes[selectedMode].first)) { for (u32 i = 0; i < IM_ARRAYSIZE(modes); i++) { if (ImGui::Selectable(modes[i].first)) selectedMode = i; } ImGui::EndCombo(); } m_mode = cs_mode(modes[selectedMode].second); } break; #if CS_API_MAJOR >= 5 case Architecture::MOS65XX: { static int selectedMode = 0; std::pair modes[] = { {"hex.disassembler.view.disassembler.mos65xx.6502"_lang, CS_MODE_MOS65XX_6502 }, { "hex.disassembler.view.disassembler.mos65xx.65c02"_lang, CS_MODE_MOS65XX_65C02 }, { "hex.disassembler.view.disassembler.mos65xx.w65c02"_lang, CS_MODE_MOS65XX_W65C02 }, { "hex.disassembler.view.disassembler.mos65xx.65816"_lang, CS_MODE_MOS65XX_65816 }, { "hex.disassembler.view.disassembler.mos65xx.65816_long_m"_lang, CS_MODE_MOS65XX_65816_LONG_M }, { "hex.disassembler.view.disassembler.mos65xx.65816_long_x"_lang, CS_MODE_MOS65XX_65816_LONG_X }, { "hex.disassembler.view.disassembler.mos65xx.65816_long_mx"_lang, CS_MODE_MOS65XX_65816_LONG_MX}, }; if (ImGui::BeginCombo("hex.disassembler.view.disassembler.settings.mode"_lang, modes[selectedMode].first)) { for (u32 i = 0; i < IM_ARRAYSIZE(modes); i++) { if (ImGui::Selectable(modes[i].first)) selectedMode = i; } ImGui::EndCombo(); } m_mode = cs_mode(modes[selectedMode].second); } break; #endif #if CS_API_MAJOR >= 5 case Architecture::BPF: { static int mode = CS_MODE_BPF_CLASSIC; ImGui::RadioButton("hex.disassembler.view.disassembler.bpf.classic"_lang, &mode, CS_MODE_BPF_CLASSIC); ImGui::SameLine(); ImGui::RadioButton("hex.disassembler.view.disassembler.bpf.extended"_lang, &mode, CS_MODE_BPF_EXTENDED); m_mode = cs_mode(mode); } break; case Architecture::SH: { static u32 selectionMode = 0; static bool fpu = false; static bool dsp = false; std::pair modes[] = { { "hex.disassembler.view.disassembler.sh.sh2"_lang, CS_MODE_SH2 }, { "hex.disassembler.view.disassembler.sh.sh2a"_lang, CS_MODE_SH2A }, { "hex.disassembler.view.disassembler.sh.sh3"_lang, CS_MODE_SH3 }, { "hex.disassembler.view.disassembler.sh.sh4"_lang, CS_MODE_SH4 }, { "hex.disassembler.view.disassembler.sh.sh4a"_lang, CS_MODE_SH4A }, }; if (ImGui::BeginCombo("hex.disassembler.view.disassembler.settings.mode"_lang, modes[selectionMode].first)) { for (u32 i = 0; i < IM_ARRAYSIZE(modes); i++) { if (ImGui::Selectable(modes[i].first)) selectionMode = i; } ImGui::EndCombo(); } ImGui::Checkbox("hex.disassembler.view.disassembler.sh.fpu"_lang, &fpu); ImGui::SameLine(); ImGui::Checkbox("hex.disassembler.view.disassembler.sh.dsp"_lang, &dsp); m_mode = cs_mode(modes[selectionMode].second | (fpu ? CS_MODE_SHFPU : cs_mode(0)) | (dsp ? CS_MODE_SHDSP : cs_mode(0))); } break; case Architecture::TRICORE: { static u32 selectionMode = 0; std::pair modes[] = { { "hex.disassembler.view.disassembler.tricore.110"_lang, CS_MODE_TRICORE_110 }, { "hex.disassembler.view.disassembler.tricore.120"_lang, CS_MODE_TRICORE_120 }, { "hex.disassembler.view.disassembler.tricore.130"_lang, CS_MODE_TRICORE_130 }, { "hex.disassembler.view.disassembler.tricore.131"_lang, CS_MODE_TRICORE_131 }, { "hex.disassembler.view.disassembler.tricore.160"_lang, CS_MODE_TRICORE_160 }, { "hex.disassembler.view.disassembler.tricore.161"_lang, CS_MODE_TRICORE_161 }, { "hex.disassembler.view.disassembler.tricore.162"_lang, CS_MODE_TRICORE_162 }, }; if (ImGui::BeginCombo("hex.disassembler.view.disassembler.settings.mode"_lang, modes[selectionMode].first)) { for (u32 i = 0; i < IM_ARRAYSIZE(modes); i++) { if (ImGui::Selectable(modes[i].first)) selectionMode = i; } ImGui::EndCombo(); } m_mode = cs_mode(modes[selectionMode].second); } break; case Architecture::WASM: #endif case Architecture::EVM: case Architecture::TMS320C64X: case Architecture::ARM64: case Architecture::SYSZ: case Architecture::XCORE: m_mode = cs_mode(0); break; } if (littleEndian) { m_mode = cs_mode(u32(m_mode) | CS_MODE_LITTLE_ENDIAN); } else { m_mode = cs_mode(u32(m_mode) | CS_MODE_BIG_ENDIAN); } ImGuiExt::EndBox(); } } // Draw disassemble button ImGui::BeginDisabled(m_disassemblerTask.isRunning() || m_regionToDisassemble.getStartAddress() < m_imageBaseAddress); { if (ImGui::Button("hex.disassembler.view.disassembler.disassemble"_lang)) this->disassemble(); } ImGui::EndDisabled(); // Draw export to file icon button ImGui::SameLine(); ImGui::BeginDisabled(m_disassemblerTask.isRunning() || m_disassembly.empty()); { if (ImGuiExt::DimmedIconButton(ICON_VS_EXPORT, ImGui::GetStyleColorVec4(ImGuiCol_Text))) this->exportToFile(); } ImGui::EndDisabled(); ImGuiExt::InfoTooltip("hex.disassembler.view.disassembler.export"_lang); // Draw a spinner if the disassembler is running if (m_disassemblerTask.isRunning()) { ImGuiExt::TextSpinner("hex.disassembler.view.disassembler.disassembling"_lang); } ImGui::NewLine(); ImGui::TextUnformatted("hex.disassembler.view.disassembler.disassembly.title"_lang); ImGui::Separator(); // Draw disassembly table if (ImGui::BeginTable("##disassembly", 4, ImGuiTableFlags_ScrollY | ImGuiTableFlags_Borders | ImGuiTableFlags_Resizable | ImGuiTableFlags_RowBg | ImGuiTableFlags_Reorderable | ImGuiTableFlags_Hideable)) { ImGui::TableSetupScrollFreeze(0, 1); ImGui::TableSetupColumn("hex.disassembler.view.disassembler.disassembly.address"_lang); ImGui::TableSetupColumn("hex.disassembler.view.disassembler.disassembly.offset"_lang); ImGui::TableSetupColumn("hex.disassembler.view.disassembler.disassembly.bytes"_lang); ImGui::TableSetupColumn("hex.disassembler.view.disassembler.disassembly.title"_lang); if (!m_disassemblerTask.isRunning()) { ImGuiListClipper clipper; clipper.Begin(m_disassembly.size()); ImGui::TableHeadersRow(); while (clipper.Step()) { for (int i = clipper.DisplayStart; i < clipper.DisplayEnd; i++) { const auto &instruction = m_disassembly[i]; ImGui::TableNextRow(); ImGui::TableNextColumn(); // Draw a selectable label for the address ImGui::PushID(i); if (ImGui::Selectable("##DisassemblyLine", false, ImGuiSelectableFlags_SpanAllColumns)) { ImHexApi::HexEditor::setSelection(instruction.offset, instruction.size); } ImGui::PopID(); // Draw instruction address ImGui::SameLine(); ImGuiExt::TextFormatted("0x{0:X}", instruction.address); // Draw instruction offset ImGui::TableNextColumn(); ImGuiExt::TextFormatted("0x{0:X}", instruction.offset); // Draw instruction bytes ImGui::TableNextColumn(); ImGui::TextUnformatted(instruction.bytes.c_str()); // Draw instruction mnemonic and operands ImGui::TableNextColumn(); ImGuiExt::TextFormattedColored(ImColor(0xFFD69C56), "{}", instruction.mnemonic); ImGui::SameLine(); ImGui::TextUnformatted(instruction.operators.c_str()); } } clipper.End(); } ImGui::EndTable(); } } } }