Merge pull request #500 from wheremyfoodat/capstone

Integrate Capstone disassembler
This commit is contained in:
wheremyfoodat 2024-04-28 22:23:15 +00:00 committed by GitHub
commit 3ff16e1588
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
5 changed files with 119 additions and 10 deletions

3
.gitmodules vendored
View file

@ -67,3 +67,6 @@
[submodule "third_party/Catch2"]
path = third_party/Catch2
url = https://github.com/catchorg/Catch2.git
[submodule "third_party/capstone"]
path = third_party/capstone
url = https://github.com/capstone-engine/capstone

View file

@ -153,8 +153,15 @@ if(HOST_X64 OR HOST_ARM64)
else()
message(FATAL_ERROR "Currently unsupported CPU architecture")
endif()
add_subdirectory(third_party/teakra EXCLUDE_FROM_ALL)
set(CAPSTONE_ARCHITECTURE_DEFAULT OFF)
set(CAPSTONE_ARM_SUPPORT ON)
set(CAPSTONE_BUILD_MACOS_THIN ON)
add_subdirectory(third_party/capstone)
include_directories(third_party/capstone/include)
set(SOURCE_FILES src/emulator.cpp src/io_file.cpp src/config.cpp
src/core/CPU/cpu_dynarmic.cpp src/core/CPU/dynarmic_cycles.cpp
src/core/memory.cpp src/renderer.cpp src/core/renderer_null/renderer_null.cpp
@ -234,7 +241,7 @@ set(HEADER_FILES include/emulator.hpp include/helpers.hpp include/termcolor.hpp
include/PICA/dynapica/shader_rec_emitter_arm64.hpp include/scheduler.hpp include/applets/error_applet.hpp
include/audio/dsp_core.hpp include/audio/null_core.hpp include/audio/teakra_core.hpp
include/audio/miniaudio_device.hpp include/ring_buffer.hpp include/bitfield.hpp include/audio/dsp_shared_mem.hpp
include/audio/hle_core.hpp
include/audio/hle_core.hpp include/capstone.hpp
)
cmrc_add_resource_library(
@ -399,7 +406,7 @@ set(ALL_SOURCES ${SOURCE_FILES} ${FS_SOURCE_FILES} ${CRYPTO_SOURCE_FILES} ${KERN
target_sources(AlberCore PRIVATE ${ALL_SOURCES})
target_link_libraries(AlberCore PRIVATE dynarmic cryptopp glad resources_console_fonts teakra)
target_link_libraries(AlberCore PUBLIC glad)
target_link_libraries(AlberCore PUBLIC glad capstone)
if(ENABLE_DISCORD_RPC AND NOT ANDROID)
target_compile_definitions(AlberCore PUBLIC "PANDA3DS_ENABLE_DISCORD_RPC=1")

54
include/capstone.hpp Normal file
View file

@ -0,0 +1,54 @@
#pragma once
#include <capstone/capstone.h>
#include <cstdio>
#include <span>
#include <string>
#include "helpers.hpp"
namespace Common {
class CapstoneDisassembler {
csh handle; // Handle to our disassembler object
cs_insn* instructions = nullptr; // Pointer to instruction object
bool initialized = false;
public:
bool isInitialized() { return initialized; }
void init(cs_arch arch, cs_mode mode) { initialized = (cs_open(arch, mode, &handle) == CS_ERR_OK); }
CapstoneDisassembler() {}
CapstoneDisassembler(cs_arch arch, cs_mode mode) { init(arch, mode); }
// Returns the number of instructions successfully disassembled
// pc: program counter of the instruction to disassemble
// bytes: Byte representation of instruction
// buffer: text buffer to output the disassembly too
usize disassemble(std::string& buffer, u32 pc, std::span<u8> bytes, u64 offset = 0) {
if (!initialized) {
buffer = "Capstone was not properly initialized";
return 0;
}
usize count = cs_disasm(handle, bytes.data(), bytes.size(), pc, offset, &instructions);
if (count == 0) {
// Error in disassembly, quit early and return empty string
buffer = "Error disassembling instructions with Capstone";
return 0;
}
buffer = "";
for (usize i = 0; i < count; i++) {
buffer += std::string(instructions[i].mnemonic) + " " + std::string(instructions[i].op_str);
if (i < count - 1) {
// Append newlines between instructions, sans the final instruction
buffer += "\n";
}
}
cs_free(instructions, count);
return count;
}
};
} // namespace Common

View file

@ -1,10 +1,15 @@
#ifdef PANDA3DS_ENABLE_LUA
#include <teakra/disassembler.h>
#include <array>
#include "capstone.hpp"
#include "emulator.hpp"
#include "lua_manager.hpp"
#ifndef __ANDROID__
extern "C" {
#include "luv.h"
#include "luv.h"
}
#endif
@ -203,6 +208,40 @@ static int getButtonThunk(lua_State* L) {
return 1;
}
static int disassembleARMThunk(lua_State* L) {
static Common::CapstoneDisassembler disassembler;
// We want the disassembler to only be fully initialized when this function is first used
if (!disassembler.isInitialized()) {
disassembler.init(CS_ARCH_ARM, CS_MODE_ARM);
}
const u32 pc = u32(lua_tonumber(L, 1));
const u32 instruction = u32(lua_tonumber(L, 2));
std::string disassembly;
// Convert instruction to byte array to pass to Capstone
std::array<u8, 4> bytes = {
u8(instruction & 0xff),
u8((instruction >> 8) & 0xff),
u8((instruction >> 16) & 0xff),
u8((instruction >> 24) & 0xff),
};
disassembler.disassemble(disassembly, pc, std::span(bytes));
lua_pushstring(L, disassembly.c_str());
return 1;
}
static int disassembleTeakThunk(lua_State* L) {
const u16 instruction = u16(lua_tonumber(L, 1));
const u16 expansion = u16(lua_tonumber(L, 2));
std::string disassembly = Teakra::Disassembler::Do(instruction, expansion);
lua_pushstring(L, disassembly.c_str());
return 1;
}
// clang-format off
static constexpr luaL_Reg functions[] = {
{ "__read8", read8Thunk },
@ -214,13 +253,15 @@ static constexpr luaL_Reg functions[] = {
{ "__write32", write32Thunk },
{ "__write64", write64Thunk },
{ "__getAppID", getAppIDThunk },
{ "__pause", pauseThunk},
{ "__resume", resumeThunk},
{ "__reset", resetThunk},
{ "__loadROM", loadROMThunk},
{ "__getButtons", getButtonsThunk},
{ "__getCirclepad", getCirclepadThunk},
{ "__getButton", getButtonThunk},
{ "__pause", pauseThunk },
{ "__resume", resumeThunk },
{ "__reset", resetThunk },
{ "__loadROM", loadROMThunk },
{ "__getButtons", getButtonsThunk },
{ "__getCirclepad", getCirclepadThunk },
{ "__getButton", getButtonThunk },
{ "__disassembleARM", disassembleARMThunk },
{ "__disassembleTeak", disassembleTeakThunk },
{ nullptr, nullptr },
};
// clang-format on
@ -254,6 +295,9 @@ void LuaManager::initializeThunks() {
getButton = function(button) return GLOBALS.__getButton(button) end,
getCirclepad = function() return GLOBALS.__getCirclepad() end,
disassembleARM = function(pc, instruction) return GLOBALS.__disassembleARM(pc, instruction) end,
disassembleTeak = function(opcode, exp) return GLOBALS.__disassembleTeak(opcode, exp or 0) end,
Frame = __Frame,
ButtonA = __ButtonA,
ButtonB = __ButtonB,

1
third_party/capstone vendored Submodule

@ -0,0 +1 @@
Subproject commit eb4fc2d7612db10379adf7aeb287a7923dcc0fc7