From ac9098fa545d3d8bb61d3afbcaa5483b0831ec44 Mon Sep 17 00:00:00 2001 From: billow Date: Fri, 27 Dec 2024 01:02:21 +0800 Subject: [PATCH] Add PIC (MCU) architecture support (#356) --- src/ArchMap.cpp | 269 ++++++++++++++++++++++++++++-------------------- 1 file changed, 156 insertions(+), 113 deletions(-) diff --git a/src/ArchMap.cpp b/src/ArchMap.cpp index b3984c5d..5bb1d620 100644 --- a/src/ArchMap.cpp +++ b/src/ArchMap.cpp @@ -11,69 +11,89 @@ using namespace ghidra; std::string CompilerFromCore(RzCore *core); -template -class BaseMapper -{ - private: - const std::function func; - public: - BaseMapper(const std::function &func) : func(func) {} - BaseMapper(const T constant) : func([constant](RzCore *core) { return constant; }) {} - T Map(RzCore *core) const { return func(core); } +template class BaseMapper { + private: + const std::function func; + + public: + BaseMapper(const std::function &func) + : func(func) + { + } + BaseMapper(const T constant) + : func([constant](RzCore *core) { return constant; }) + { + } + T Map(RzCore *core) const + { + return func(core); + } }; -template class Mapper; -template<> class Mapper : public BaseMapper { public: using BaseMapper::BaseMapper; }; -template<> class Mapper : public BaseMapper { public: using BaseMapper::BaseMapper; }; +template class Mapper; +template <> class Mapper : public BaseMapper { + public: + using BaseMapper::BaseMapper; +}; +template <> class Mapper : public BaseMapper { + public: + using BaseMapper::BaseMapper; +}; -template<> class Mapper : public BaseMapper -{ - public: - using BaseMapper::BaseMapper; - Mapper(const char *constant) : BaseMapper([constant](RzCore *core) { return constant; }) {} +template <> class Mapper : public BaseMapper { + public: + using BaseMapper::BaseMapper; + Mapper(const char *constant) + : BaseMapper([constant](RzCore *core) { return constant; }) + { + } }; -static const Mapper big_endian_mapper_default = std::function([](RzCore *core) { return rz_config_get_i(core->config, "cfg.bigendian") != 0; }); -static const Mapper bits_mapper_default = std::function([](RzCore *core) { return rz_config_get_i(core->config, "asm.bits"); }); +static const Mapper big_endian_mapper_default = + std::function([](RzCore *core) { + return rz_config_get_i(core->config, "cfg.bigendian") != 0; + }); +static const Mapper bits_mapper_default = std::function( + [](RzCore *core) { return rz_config_get_i(core->config, "asm.bits"); }); + +class ArchMapper { + private: + const Mapper arch; + const Mapper flavor; + const Mapper big_endian; + const Mapper bits; + + public: + ArchMapper(const Mapper arch, + const Mapper flavor = "default", + const Mapper bits = bits_mapper_default, + const Mapper big_endian = big_endian_mapper_default) + : arch(arch) + , flavor(flavor) + , bits(bits) + , big_endian(big_endian) + { + } -class ArchMapper -{ - private: - const Mapper arch; - const Mapper flavor; - const Mapper big_endian; - const Mapper bits; - - public: - ArchMapper( - const Mapper arch, - const Mapper flavor = "default", - const Mapper bits = bits_mapper_default, - const Mapper big_endian = big_endian_mapper_default) - : arch(arch), flavor(flavor), bits(bits), big_endian(big_endian) {} - - std::string Map(RzCore *core) const - { - return arch.Map(core) - + ":" + (big_endian.Map(core) ? "BE" : "LE") - + ":" + to_string(bits.Map(core)) - + ":" + flavor.Map(core) - + ":" + CompilerFromCore(core); - } + std::string Map(RzCore *core) const + { + return arch.Map(core) + ":" + + (big_endian.Map(core) ? "BE" : "LE") + ":" + + to_string(bits.Map(core)) + ":" + flavor.Map(core) + + ":" + CompilerFromCore(core); + } }; #define BITS (rz_config_get_i(core->config, "asm.bits")) -#define CUSTOM_BASEID(lambda) std::function([]lambda) -#define CUSTOM_FLAVOR(lambda) std::function([]lambda) -#define CUSTOM_BITS(lambda) std::function([]lambda) +#define CUSTOM_BASEID(lambda) std::function([] lambda) +#define CUSTOM_FLAVOR(lambda) std::function([] lambda) +#define CUSTOM_BITS(lambda) std::function([] lambda) // keys = asm.arch values static const std::map arch_map = { - { "x86", { - "x86", - CUSTOM_FLAVOR((RzCore *core) { - return BITS == 16 ? "Real Mode" : "default"; - })}}, + { "x86", { "x86", CUSTOM_FLAVOR((RzCore *core) { + return BITS == 16 ? "Real Mode" : "default"; + }) } }, { "mips", { "MIPS" } }, { "dalvik", { "Dalvik" } }, @@ -85,52 +105,72 @@ static const std::map arch_map = { { "sparc", { "sparc" } }, { "sh", { "SuperH4" } }, { "msp430", { "TI_MSP430" } }, - { "m68k", { - "68000", - CUSTOM_FLAVOR((RzCore *core) { - const char *cpu = rz_config_get(core->config, "asm.cpu"); - if(!cpu) - return "default"; - if(strcmp(cpu, "68020") == 0) - return "MC68020"; - if(strcmp(cpu, "68030") == 0) - return "MC68030"; - if(strcmp(cpu, "68060") == 0) - return "Coldfire"; // may not be accurate!! - return "default"; - }), - 32 } }, - - { "arm", { - CUSTOM_BASEID((RzCore *core) { - return BITS == 64 ? "AARCH64" : "ARM"; - }), - CUSTOM_FLAVOR((RzCore *core) { - return BITS == 64 ? "v8A" : "v7"; - }), - CUSTOM_BITS((RzCore *core) { - return BITS == 64 ? 64 : 32; - })}}, - - { "avr", { - CUSTOM_BASEID((RzCore *core) { - return BITS == 32 ? "avr32a" : "avr8"; - }), - "default", - CUSTOM_BITS((RzCore *core) { - return BITS == 32 ? 32 : 16; - })}}, - - { "v850", { - CUSTOM_BASEID((RzCore *core) { - return "V850"; - }), - "default", - CUSTOM_BITS((RzCore *core) { - return 32; - })}}, + { "m68k", + { "68000", CUSTOM_FLAVOR((RzCore *core) { + const char *cpu = rz_config_get(core->config, "asm.cpu"); + if (!cpu) + return "default"; + if (strcmp(cpu, "68020") == 0) + return "MC68020"; + if (strcmp(cpu, "68030") == 0) + return "MC68030"; + if (strcmp(cpu, "68060") == 0) + return "Coldfire"; // may not be accurate!! + return "default"; + }), + 32 } }, + + { "arm", + { CUSTOM_BASEID( + (RzCore *core) { return BITS == 64 ? "AARCH64" : "ARM"; }), + CUSTOM_FLAVOR((RzCore *core) { return BITS == 64 ? "v8A" : "v7"; }), + CUSTOM_BITS((RzCore *core) { return BITS == 64 ? 64 : 32; }) } }, + + { "avr", + { CUSTOM_BASEID( + (RzCore *core) { return BITS == 32 ? "avr32a" : "avr8"; }), + "default", + CUSTOM_BITS((RzCore *core) { return BITS == 32 ? 32 : 16; }) } }, + + { "v850", + { CUSTOM_BASEID((RzCore *core) { return "V850"; }), "default", + CUSTOM_BITS((RzCore *core) { return 32; }) } }, { "tricore", { "tricore" } }, + + { "pic", + { + CUSTOM_BASEID((RzCore *core) { + const char *cpu = + rz_config_get(core->config, "analysis.cpu"); + if (RZ_STR_EQ(cpu, "pic14")) { + return "PIC-12"; + } + if (RZ_STR_EQ(cpu, "pic16")) { + return "PIC-16"; + } + return "PIC-18"; + }), + CUSTOM_BASEID((RzCore *core) { + const char *cpu = + rz_config_get(core->config, "analysis.cpu"); + if (RZ_STR_EQ(cpu, "pic14")) { + return "PIC-12C5xx"; + } + if (RZ_STR_EQ(cpu, "pic16")) { + return "PIC-16F"; + } + return "PIC-18"; + }), + CUSTOM_BITS((RzCore *core) { + const char *cpu = + rz_config_get(core->config, "analysis.cpu"); + if (RZ_STR_EQ(cpu, "pic18")) { + return 24; + } + return 16; + }), + } }, }; static const std::map compiler_map = { @@ -149,7 +189,7 @@ std::string CompilerFromCore(RzCore *core) return std::string(); auto comp_it = compiler_map.find(info->rclass); - if(comp_it == compiler_map.end()) + if (comp_it == compiler_map.end()) return std::string(); return comp_it->second; @@ -160,37 +200,40 @@ RZ_API std::string SleighIdFromCore(RzCore *core) SleighArchitecture::collectSpecFiles(std::cerr); auto langs = SleighArchitecture::getLanguageDescriptions(); const char *arch = rz_config_get(core->config, "asm.arch"); - if(!strcmp(arch, "ghidra")) - return SleighIdFromSleighAsmConfig(core->rasm->cpu, core->rasm->bits, core->rasm->big_endian, langs); + if (!strcmp(arch, "ghidra")) + return SleighIdFromSleighAsmConfig(core->rasm->cpu, + core->rasm->bits, + core->rasm->big_endian, + langs); auto arch_it = arch_map.find(arch); - if(arch_it == arch_map.end()) - throw LowlevelError("Could not match asm.arch " + std::string(arch) + " to sleigh arch."); + if (arch_it == arch_map.end()) + throw LowlevelError("Could not match asm.arch " + + std::string(arch) + " to sleigh arch."); return arch_it->second.Map(core); } std::string StrToLower(std::string s) { - std::transform(s.begin(), s.end(), s.begin(), [](unsigned char c){ return std::tolower(c); }); - return s; + std::transform(s.begin(), s.end(), s.begin(), + [](unsigned char c) { return std::tolower(c); }); + return s; } -RZ_API std::string SleighIdFromSleighAsmConfig(const char *cpu, int bits, bool bigendian, const vector &langs) +RZ_API std::string +SleighIdFromSleighAsmConfig(const char *cpu, int bits, bool bigendian, + const vector &langs) { - if(!cpu) + if (!cpu) return std::string(); - if(std::string(cpu).find(':') != string::npos) // complete id specified + if (std::string(cpu).find(':') != string::npos) // complete id specified return cpu; // short form if possible std::string low_cpu = StrToLower(cpu); - for(const auto &lang : langs) - { + for (const auto &lang : langs) { auto proc = lang.getProcessor(); - if(StrToLower(proc) == low_cpu) - { - return proc - + ":" + (bigendian ? "BE" : "LE") - + ":" + to_string(bits) - + ":" + "default"; + if (StrToLower(proc) == low_cpu) { + return proc + ":" + (bigendian ? "BE" : "LE") + ":" + + to_string(bits) + ":" + "default"; } } return cpu;