/* DingusPPC - The Experimental PowerPC Macintosh emulator Copyright (C) 2018-24 divingkatae and maximum (theweirdo) spatium (Contact divingkatae#1017 or powermax#2286 on Discord for more info) This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ /** @file Logic for the disassembler. @author maximumspatium */ #define _CRT_SECURE_NO_WARNINGS /* shut up MSVC regarding the unsafe strcpy/strcat */ #include "ppcdisasm.h" #include #include #include /* without this, MSVC doesn't understand std::function */ #include #include #include using namespace std; /* debugging support API */ uint64_t get_reg(std::string reg_name); /* get content of the register reg_name */ template std::string my_sprintf(const char* format, Args... args) { char buf_small[32]; int length = std::snprintf(buf_small, sizeof(buf_small), format, args...); if (length <= 0) return {}; /* empty string in C++11 */ if (length < sizeof(buf_small)) return buf_small; char* buf = new char[length + 1]; std::snprintf(buf, length + 1, format, args...); std::string str(buf); delete[] buf; return str; } template std::string add_reg(std::vector ®s, const char* format, Args... args) { std::string str = my_sprintf(format, args...); std::vector::iterator it; it = std::find(regs.begin(), regs.end(), str); if (it != regs.end()) { return *it; } else { regs.push_back(str); return regs.back(); } } template std::string add_reg_in(PPCDisasmContext *ctx, const char* format, Args... args) { return add_reg(ctx->regs_in, format, args...); } template std::string add_reg_out(PPCDisasmContext *ctx, const char* format, Args... args) { return add_reg(ctx->regs_out, format, args...); } const char* arith_im_mnem[9] = {"mulli", "subfic", "", "", "", "addic", "addic.", "addi", "addis"}; const char* bx_mnem[4] = {"b", "bl", "ba", "bla"}; const char* bclrx_mnem[2] = {"blr", "blrl"}; const char* bcctrx_mnem[2] = {"bctr", "bctrl"}; const char* br_cond[8] = {/* simplified branch conditions */ "ge", "le", "ne", "ns", "lt", "gt", "eq", "so"}; const char* bclrx_cond[8] = {/* simplified branch conditions */ "gelr", "lelr", "nelr", "nslr", "ltlr", "gtlr", "eqlr", "solr"}; const char* bcctrx_cond[8] = {/* simplified branch conditions */ "gectr", "lectr", "nectr", "nsctr", "ltctr", "gtctr", "eqctr", "soctr"}; const char* opc_idx_ldst[32] = {/* indexed load/store opcodes */ "lwzx", "lwzux", "lbzx", "lbzux", "stwx", "stwux", "stbx", "stbux", "lhzx", "lhzux", "lhax", "lhaux", "sthx", "sthux", "", "", "lfsx", "lfsux", "lfdx", "lfdux", "stfsx", "stfsux", "stfdx", "stfdux", "", "", "", "", "", "", "stfiwx", ""}; const char* opc_bim_str[6] = {/* boolean immediate instructions */ "ori", "oris", "xori", "xoris", "andi.", "andis."}; const char* opc_logic[16] = {/* logic instructions */ "and", "andc", "", "nor", "", "", "", "", "eqv", "xor", "", "", "orc", "or", "nand", ""}; const char* opc_subs[16] = {/* subtracts & friends */ "subfc", "subf", "", "neg", "subfe", "", "subfze", "subfme", "doz", "", "", "abs", "", "", "", "nabs"}; const char* opc_adds[9] = {/* additions */ "addc", "", "", "", "adde", "", "addze", "addme", "add"}; const char* opc_muldivs[16] = {/* multiply and division instructions */ "mulhwu", "", "mulhw", "mul", "", "", "", "mullw", "", "", "div", "divs", "", "", "divwu", "divw"}; const char* opc_shft_reg[32]{ /* Regular shift instructions */ "slw", "", "", "", "slq", "sliq", "sllq", "slliq", "", "", "", "", "", "", "", "", "srw", "", "", "", "srq", "sriq", "srlq", "srliq", "sraw", "srawi", "", "", "sraq", "sraiq", "", ""}; const char* opc_shft_ext[32]{ /* Extended shift instructions (601 only) */ "", "", "", "", "sle", "", "sleq", "", "", "", "", "", "", "", "", "", "rrib", "", "", "", "sre", "", "sreq", "", "", "", "", "", "srea", "", "", ""}; const char* opc_int_ldst[16] = {/* integer load and store instructions */ "lwz", "lwzu", "lbz", "lbzu", "stw", "stwu", "stb", "stbu", "lhz", "lhzu", "lha", "lhau", "sth", "sthu", "lmw", "stmw"}; /* processor management + byte reversed load and store */ const char* proc_mgmt_str[32] = { "", "dcbst", "dcbf", "", "stwcx.", "", "", "dcbtst", "dcbt", "eciwx", "", "", "", "ecowx", "dcbi", "", "lwbrx", "tlbsync", "sync", "", "stwbrx", "", "", "dcba", "lhbrx", "", "eieio", "", "sthbrx", "", "icbi", "dcbz"}; const char* opc_flt_ldst[8] = {/* integer load and store instructions */ "lfs", "lfsu", "lfd", "lfdu", "stfs", "stfsu", "stfd", "stfdu"}; const char* opc_flt_ext_arith[32] = { /* integer load and store instructions */ "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "fsel", "", "fmul", "", "", "fmsub", "fmadd", "fnmsub", "fnmadd", }; const char* trap_cond[32] = {/*Trap conditions*/ "", "twlgt", "twllt", "", "tweq", "twlge", "twlle", "", "twgt", "", "", "", "twge", "", "", "", "twlt", "", "", "", "twle", "", "", "", "twne", "", "", "", "", "", "", ""}; const char* spr_index0[32] = {"mq", "xer", "", "", "rtcu", "rtcl", "", "", "lr", "ctr", "", "", "", "", "", "", "", "", "dsisr", "dar", "", "", "dec", "", "", "sdr1", "srr0", "srr1", "", "", "", ""}; const char* spr_index8[32] = { "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "sprg0", "sprg1", "sprg2", "sprg3", "sprg4", "sprg5", "sprg6", "sprg7", "", "", "ear", "", "tbl", "tbu", "", "pvr"}; const char* spr_index16[32] = { "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "ibat0u", "ibat0l", "ibat1u", "ibat1l", "ibat2u", "ibat2l", "ibat3u", "ibat3l", "dbat0u", "dbat0l", "dbat1u", "dbat1l", "dbat2u", "dbat2l", "dbat3u", "dbat3l", }; const char* spr_index29[32] = { "", "", "", "", "", "", "", "", "ummcr0", "upmc1", "upmc2", "usia", "ummcr1", "upmc3", "upmc4", "", "", "", "", "", "", "", "", "", "mmcr0", "pmc1", "pmc2", "sia", "mmcr1", "pmc3", "pmc4", "sda"}; const char* spr_index30[32] = { "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "dmiss", "dcmp", "hash1", "hash2", "imiss", "icmp", "rpa", "", "", "", "", "", "", "", "", ""}; const char* spr_index31[32] = { "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "hid0", "hid1", "iabr", "", "", "dabr", "", "", "", "l2cr", "", "ictc", "thrm1", "thrm2", "thrm3", "pir", }; /** various formatting helpers. */ void fmt_oneop(PPCDisasmContext* ctx, const char* opc, int src) { ctx->instr_str = my_sprintf("%-7s r%d", opc, src); add_reg_in(ctx, "r%d", src); } void fmt_oneop_out(PPCDisasmContext* ctx, const char* opc, int dst) { ctx->instr_str = my_sprintf("%-7s r%d", opc, dst); add_reg_out(ctx, "r%d", dst); } void fmt_twoop(PPCDisasmContext* ctx, const char* opc, int dst, int src) { ctx->instr_str = my_sprintf("%-7s r%d, r%d", opc, dst, src); add_reg_in(ctx, "r%d", src); add_reg_out(ctx, "r%d", dst); } void fmt_twoop_in(PPCDisasmContext* ctx, const char* opc, int dst, int src) { ctx->instr_str = my_sprintf("%-7s r%d, r%d", opc, dst, src); add_reg_in(ctx, "r%d", src); add_reg_in(ctx, "r%d", dst); } void fmt_twoop_simm(PPCDisasmContext* ctx, const char* opc, int dst, int imm) { ctx->instr_str = my_sprintf("%-7s r%d, %s0x%X", opc, dst, (imm < 0) ? "-" : "", abs(imm)); add_reg_out(ctx, "r%d", dst); } void fmt_twoop_fromspr(PPCDisasmContext* ctx, const char* opc, int dst, int src) { ctx->instr_str = my_sprintf("%-7s pr%d, spr%d", opc, dst, src); add_reg_in(ctx, "spr%d", src); add_reg_out(ctx, "r%d", dst); } void fmt_twoop_tospr(PPCDisasmContext* ctx, const char* opc, int dst, int src) { ctx->instr_str = my_sprintf("%-7s spr%d, r%d", opc, dst, src); add_reg_in(ctx, "r%d", src); add_reg_out(ctx, "spr%d", dst); } void fmt_twoop_flt(PPCDisasmContext* ctx, const char* opc, int dst, int src1) { ctx->instr_str = my_sprintf("%-7s f%d, f%d", opc, dst, src1); add_reg_in(ctx, "f%d", src1); add_reg_out(ctx, "f%d", dst); } void fmt_threeop(PPCDisasmContext* ctx, const char* opc, int dst, int src1, int src2) { ctx->instr_str = my_sprintf("%-7s r%d, r%d, r%d", opc, dst, src1, src2); add_reg_in(ctx, "r%d", src1); add_reg_in(ctx, "r%d", src2); add_reg_out(ctx, "r%d", dst); } void fmt_threeop_store(PPCDisasmContext* ctx, const char* opc, int dst, int src1, int src2) { ctx->instr_str = my_sprintf("%-7s r%d, r%d, r%d", opc, dst, src1, src2); add_reg_in(ctx, "r%d", dst); add_reg_in(ctx, "r%d", src1); add_reg_in(ctx, "r%d", src2); } void fmt_threeop_crb(PPCDisasmContext* ctx, const char* opc, int dst, int src1, int src2) { ctx->instr_str = my_sprintf("%-7s crb%d, crb%d, crb%d", opc, dst, src1, src2); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); } void fmt_threeop_uimm(PPCDisasmContext* ctx, const char* opc, int dst, int src1, int imm) { ctx->instr_str = my_sprintf("%-7s r%d, r%d, 0x%X", opc, dst, src1, imm); add_reg_in(ctx, "r%d", src1); add_reg_out(ctx, "r%d", dst); } void fmt_threeop_simm(PPCDisasmContext* ctx, const char* opc, int dst, int src1, int imm) { ctx->instr_str = my_sprintf("%-7s r%d, r%d, %s0x%X", opc, dst, src1, (imm < 0) ? "-" : "", abs(imm)); add_reg_in(ctx, "r%d", src1); add_reg_out(ctx, "r%d", dst); } void fmt_threeop_flt(PPCDisasmContext* ctx, const char* opc, int dst, int src1, int src2) { ctx->instr_str = my_sprintf("%-7s f%d, f%d, f%d", opc, dst, src1, src2); add_reg_in(ctx, "f%d", src1); add_reg_in(ctx, "f%d", src2); add_reg_out(ctx, "f%d", dst); } void fmt_fourop_flt(PPCDisasmContext* ctx, const char* opc, int dst, int src1, int src2, int src3) { ctx->instr_str = my_sprintf("%-7s f%d, f%d, f%d, f%d", opc, dst, src1, src2, src3); add_reg_in(ctx, "f%d", src1); add_reg_in(ctx, "f%d", src2); add_reg_in(ctx, "f%d", src3); add_reg_out(ctx, "f%d", dst); } void fmt_rotateop(PPCDisasmContext* ctx, const char* opc, int dst, int src, int sh, int mb, int me, bool imm) { if (imm) { ctx->instr_str = my_sprintf("%-7s r%d, r%d, %d, %d, %d", opc, dst, src, sh, mb, me); add_reg_in(ctx, "r%d", src); add_reg_out(ctx, "r%d", dst); } else { ctx->instr_str = my_sprintf("%-7s r%d, r%d, r%d, %d, %d", opc, dst, src, sh, mb, me); add_reg_in(ctx, "r%d", src); add_reg_in(ctx, "r%d", sh); add_reg_out(ctx, "r%d", dst); } } /* Opcodes */ void opc_illegal(PPCDisasmContext* ctx) { ctx->instr_str = my_sprintf("%-7s 0x%08X", "dc.l", ctx->instr_code); } void opc_twi(PPCDisasmContext* ctx) { char opcode[10] = ""; auto to = (ctx->instr_code >> 21) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; int32_t imm = SIGNEXT(ctx->instr_code & 0xFFFF, 15); if (ctx->simplified) { strcpy(opcode, trap_cond[to]); if (strlen(opcode) > 0) { strcat(opcode, "i"); ctx->instr_str = my_sprintf("%-7s r%d, 0x%X", opcode, ra, imm); add_reg_in(ctx, "r%d", ra); return; } } ctx->instr_str = my_sprintf("%-7s %d, r%d, 0x%X", "twi", to, ra, imm); add_reg_in(ctx, "r%d", ra); } void opc_group4(PPCDisasmContext* ctx) { printf("Altivec group 4 not supported yet\n"); } void opc_ar_im(PPCDisasmContext* ctx) { auto ra = (ctx->instr_code >> 16) & 0x1F; auto rd = (ctx->instr_code >> 21) & 0x1F; int32_t imm = SIGNEXT(ctx->instr_code & 0xFFFF, 15); if (ctx->simplified) { if (((ctx->instr_code >> 26) == 0xE) && !ra) { fmt_twoop_simm(ctx, "li", rd, imm); return; } else if (((ctx->instr_code >> 26) == 0xF) && !ra) { fmt_twoop_simm(ctx, "lis", rd, imm); return; } if (imm > 0x7FFF) { switch ((ctx->instr_code >> 26)) { case 0xC: fmt_threeop_simm(ctx, "subic", rd, ra, imm); return; case 0xD: fmt_threeop_simm(ctx, "subic.", rd, ra, imm); add_reg_out(ctx, "cr"); return; case 0xE: fmt_threeop_simm(ctx, "subi", rd, ra, imm); return; case 0xF: fmt_threeop_simm(ctx, "subis", rd, ra, imm); return; } } } fmt_threeop_simm(ctx, arith_im_mnem[(ctx->instr_code >> 26) - 7], rd, ra, imm); if ((ctx->instr_code >> 26) == 13) { /* addic. */ add_reg_out(ctx, "cr"); } } void power_dozi(PPCDisasmContext* ctx) { auto ra = (ctx->instr_code >> 16) & 0x1F; auto rd = (ctx->instr_code >> 21) & 0x1F; auto imm = ctx->instr_code & 0xFFFF; fmt_threeop_simm(ctx, "dozi", rd, ra, imm); } void fmt_rot_imm(PPCDisasmContext* ctx, const char* opc, int ra, int rs, int n) { char opcode[10]; strcpy(opcode, opc); if (ctx->instr_code & 1) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s r%d, r%d, %d", opcode, ra, rs, n); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "r%d", ra); } void fmt_rot_2imm(PPCDisasmContext* ctx, const char* opc, int ra, int rs, int n, int b) { char opcode[10]; strcpy(opcode, opc); if (ctx->instr_code & 1) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s r%d, r%d, %d, %d", opcode, ra, rs, n, b); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "r%d", ra); } void opc_rlwimi(PPCDisasmContext* ctx) { char opcode[10]; auto rs = (ctx->instr_code >> 21) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; auto sh = (ctx->instr_code >> 11) & 0x1F; auto mb = (ctx->instr_code >> 6) & 0x1F; auto me = (ctx->instr_code >> 1) & 0x1F; if (ctx->simplified) { if ((32 - sh) == mb) { fmt_rot_2imm(ctx, "inslwi", ra, rs, me + 1 - mb, mb); return; } else if (sh == 32 - (me + 1)) { fmt_rot_2imm(ctx, "insrwi", ra, rs, (me - mb + 1), mb); return; } } strcpy(opcode, "rlwimi"); if (ctx->instr_code & 1) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_rotateop(ctx, opcode, ra, rs, sh, mb, me, true); } void opc_rlwinm(PPCDisasmContext* ctx) { char opcode[10]; auto rs = (ctx->instr_code >> 21) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; auto sh = (ctx->instr_code >> 11) & 0x1F; auto mb = (ctx->instr_code >> 6) & 0x1F; auto me = (ctx->instr_code >> 1) & 0x1F; if (ctx->simplified) { if ((mb == 0) && (me == 31)) { if (sh < 16) { fmt_rot_imm(ctx, "rotlwi", ra, rs, sh); } else { fmt_rot_imm(ctx, "rotrwi", ra, rs, 32 - sh); } return; } else if (me == 31) { if ((32 - sh) == mb) { fmt_rot_imm(ctx, "srwi", ra, rs, mb); } else if (sh == 0) { fmt_rot_imm(ctx, "clrlwi", ra, rs, mb); } else { fmt_rot_2imm(ctx, "extrwi", ra, rs, (32 - mb), (sh - (32 - mb))); } return; } else if (mb == 0) { if ((31 - me) == sh) { fmt_rot_imm(ctx, "slwi", ra, rs, sh); } else if (sh == 0) { fmt_rot_imm(ctx, "clrrwi", ra, rs, (31 - me)); } else { fmt_rot_2imm(ctx, "extlwi", ra, rs, (me + 1), sh); } return; } else if (mb) { if ((31 - me) == sh) { fmt_rot_2imm(ctx, "clrlslwi", ra, rs, (mb + sh), sh); return; } } } strcpy(opcode, "rlwinm"); if (ctx->instr_code & 1) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_rotateop(ctx, opcode, ra, rs, sh, mb, me, true); } void opc_rlmi(PPCDisasmContext* ctx) { auto rs = (ctx->instr_code >> 21) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; auto rb = (ctx->instr_code >> 11) & 0x1F; auto mb = (ctx->instr_code >> 6) & 0x1F; auto me = (ctx->instr_code >> 1) & 0x1F; if (ctx->instr_code & 1) { fmt_rotateop(ctx, "rlmi.", ra, rs, rb, mb, me, false); add_reg_out(ctx, "cr"); } else fmt_rotateop(ctx, "rlmi", ra, rs, rb, mb, me, false); } void opc_rlwnm(PPCDisasmContext* ctx) { char opcode[10]; auto rs = (ctx->instr_code >> 21) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; auto rb = (ctx->instr_code >> 11) & 0x1F; auto mb = (ctx->instr_code >> 6) & 0x1F; auto me = (ctx->instr_code >> 1) & 0x1F; if (ctx->simplified) { if ((me == 31) && (mb == 0)) { strcpy(opcode, "rotlw"); if (ctx->instr_code & 1) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_threeop(ctx, opcode, ra, rs, rb); return; } } strcpy(opcode, "rlwnm"); if (ctx->instr_code & 1) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_rotateop(ctx, "rlwnm", ra, rs, rb, mb, me, false); } void opc_cmp_i_li(PPCDisasmContext* ctx) { auto ls = (ctx->instr_code >> 21) & 0x1; auto ra = (ctx->instr_code >> 16) & 0x1F; auto crfd = (ctx->instr_code >> 23) & 0x07; int imm = ctx->instr_code & 0xFFFF; if (ctx->simplified) { if (!ls) { if ((ctx->instr_code >> 26) & 0x1) { ctx->instr_str = my_sprintf("%-7s cr%d, r%d, 0x%X", "cmpwi", crfd, ra, imm); add_reg_in(ctx, "r%d", ra); add_reg_out(ctx, "cr"); } else { ctx->instr_str = my_sprintf( "%-7s cr%d, r%d, %s0x%X", "cmplwi", crfd, ra, (imm < 0) ? "-" : "", abs(imm)); add_reg_in(ctx, "r%d", ra); add_reg_out(ctx, "cr"); } return; } } if ((ctx->instr_code >> 26) & 0x1) { ctx->instr_str = my_sprintf("%-7s cr%d, %d, r%d, 0x%X", "cmpi", crfd, ls, ra, imm); add_reg_in(ctx, "r%d", ra); add_reg_out(ctx, "cr"); } else { ctx->instr_str = my_sprintf( "%-7s cr%d, %d, r%d, %s0x%X", "cmpli", crfd, ls, ra, (imm < 0) ? "-" : "", abs(imm)); add_reg_in(ctx, "r%d", ra); add_reg_out(ctx, "cr"); } } void opc_bool_im(PPCDisasmContext* ctx) { char opcode[10]; auto ra = (ctx->instr_code >> 16) & 0x1F; auto rs = (ctx->instr_code >> 21) & 0x1F; auto index = ((ctx->instr_code >> 26) & 0x1F) - 24; auto imm = ctx->instr_code & 0xFFFF; if (ctx->simplified) { if (index == 0) { if (imm == 0 && !ra && !rs && !imm) { /* unofficial, produced by IDA */ ctx->instr_str = my_sprintf("%-7s ", "nop"); return; } } } strcpy(opcode, opc_bim_str[index]); fmt_threeop_uimm(ctx, opcode, ra, rs, imm); if (index >= 4) { /* andi. or andis. */ add_reg_out(ctx, "cr"); } } void generic_bcx(PPCDisasmContext* ctx, uint32_t bo, uint32_t bi, uint32_t dst) { char opcode[10] = "bc"; uint32_t cr; if (!(bo & 4)) { add_reg_in(ctx, "ctr"); add_reg_out(ctx, "ctr"); } if (!(bo & 0x10)) { add_reg_in(ctx, "cr"); } if (ctx->instr_code & 1) { strcat(opcode, "l"); /* add suffix "l" if the LK bit is set */ add_reg_out(ctx, "lr"); } if (ctx->instr_code & 2) { strcat(opcode, "a"); /* add suffix "a" if the AA bit is set */ } ctx->instr_str = my_sprintf("%-7s %d, %d, 0x%08X", opcode, bo, bi, dst); } void generic_bcctrx(PPCDisasmContext* ctx, uint32_t bo, uint32_t bi) { char opcode[10] = "bcctr"; add_reg_in(ctx, "ctr"); if (!(bo & 0x10)) { add_reg_in(ctx, "cr"); } if (ctx->instr_code & 1) { strcat(opcode, "l"); /* add suffix "l" if the LK bit is set */ add_reg_out(ctx, "lr"); } ctx->instr_str = my_sprintf("%-7s %d, %d, 0x%08X", opcode, bo, bi); } void generic_bclrx(PPCDisasmContext* ctx, uint32_t bo, uint32_t bi) { char opcode[10] = "bclr"; add_reg_in(ctx, "lr"); if (!(bo & 4)) { add_reg_in(ctx, "ctr"); add_reg_out(ctx, "ctr"); } if (!(bo & 0x10)) { add_reg_in(ctx, "cr"); } if (ctx->instr_code & 1) { strcat(opcode, "l"); /* add suffix "l" if the LK bit is set */ add_reg_out(ctx, "lr"); } ctx->instr_str = my_sprintf("%-7s %d, %d, 0x%08X", opcode, bo, bi); } void opc_bcx(PPCDisasmContext* ctx) { uint32_t bo, bi, dst, cr; char opcode[10] = "b"; char operands[12] = ""; bo = (ctx->instr_code >> 21) & 0x1F; bi = (ctx->instr_code >> 16) & 0x1F; cr = bi >> 2; dst = ((ctx->instr_code & 2) ? 0 : ctx->instr_addr) + SIGNEXT(ctx->instr_code & 0xFFFC, 15); if (!ctx->simplified || ((bo & 0x10) && bi) || (((bo & 0x14) == 0x14) && (bo & 0xB) && bi)) { generic_bcx(ctx, bo, bi, dst); return; } if ((bo & 0x14) == 0x14) { ctx->instr_str = my_sprintf("%-7s 0x%08X", bx_mnem[0], dst); return; } if (!(bo & 4)) { strcat(opcode, "d"); add_reg_in(ctx, "ctr"); strcat(opcode, (bo & 2) ? "z" : "nz"); if (!(bo & 0x10)) { strcat(opcode, (bo & 8) ? "t" : "f"); if (cr) { strcat(operands, "4*cr0+"); operands[4] = cr + '0'; } strcat(operands, br_cond[4 + (bi & 3)]); strcat(operands, ", "); add_reg_in(ctx, "cr"); } add_reg_out(ctx, "ctr"); } else { /* CTR ignored */ strcat(opcode, br_cond[((bo >> 1) & 4) | (bi & 3)]); if (cr) { strcat(operands, "cr0, "); operands[2] = cr + '0'; } if (!(bo & 0x10)) { add_reg_in(ctx, "cr"); } } if (ctx->instr_code & 1) { strcat(opcode, "l"); /* add suffix "l" if the LK bit is set */ add_reg_out(ctx, "lr"); } if (ctx->instr_code & 2) { strcat(opcode, "a"); /* add suffix "a" if the AA bit is set */ } if (bo & 1) { /* incorporate prediction bit if set */ strcat(opcode, (ctx->instr_code & 0x8000) ? "-" : "+"); } ctx->instr_str = my_sprintf("%-7s %s0x%08X", opcode, operands, dst); } void opc_bcctrx(PPCDisasmContext* ctx) { uint32_t bo, bi, cr; char opcode[10] = "b"; char operands[4] = ""; bo = (ctx->instr_code >> 21) & 0x1F; bi = (ctx->instr_code >> 16) & 0x1F; cr = bi >> 2; if (!(bo & 4)) { /* bcctr with BO[2] = 0 is invalid */ opc_illegal(ctx); } if (!ctx->simplified || ((bo & 0x10) && bi) || (((bo & 0x14) == 0x14) && (bo & 0xB) && bi)) { generic_bcctrx(ctx, bo, bi); return; } if ((bo & 0x14) == 0x14) { ctx->instr_str = my_sprintf("%-7s ", bcctrx_mnem[ctx->instr_code & 1]); add_reg_in(ctx, "ctr"); if (ctx->instr_code & 1) { add_reg_out(ctx, "lr"); } return; } strcat(opcode, br_cond[((bo >> 1) & 4) | (bi & 3)]); strcat(opcode, "ctr"); if (cr) { strcat(operands, "cr0"); operands[2] = cr + '0'; } add_reg_in(ctx, "ctr"); if (!(bo & 0x10)) { add_reg_in(ctx, "cr"); } if (ctx->instr_code & 1) { strcat(opcode, "l"); /* add suffix "l" if the LK bit is set */ add_reg_out(ctx, "lr"); } if (bo & 1) { /* incorporate prediction bit if set */ strcat(opcode, "+"); } ctx->instr_str = my_sprintf("%-7s %s", opcode, operands); } void opc_bclrx(PPCDisasmContext* ctx) { uint32_t bo, bi, cr; char opcode[10] = "b"; char operands[12] = ""; bo = (ctx->instr_code >> 21) & 0x1F; bi = (ctx->instr_code >> 16) & 0x1F; cr = bi >> 2; if (!ctx->simplified || ((bo & 0x10) && bi) || (((bo & 0x14) == 0x14) && (bo & 0xB) && bi)) { generic_bclrx(ctx, bo, bi); return; } if ((bo & 0x14) == 0x14) { ctx->instr_str = my_sprintf("%-7s ", bclrx_mnem[ctx->instr_code & 1]); add_reg_in(ctx, "lr"); if (ctx->instr_code & 1) { add_reg_out(ctx, "lr"); } return; } if (!(bo & 4)) { strcat(opcode, "d"); add_reg_in(ctx, "ctr"); strcat(opcode, (bo & 2) ? "z" : "nz"); if (!(bo & 0x10)) { strcat(opcode, (bo & 8) ? "t" : "f"); if (cr) { strcat(operands, "4*cr0+"); operands[4] = cr + '0'; } strcat(operands, br_cond[4 + (bi & 3)]); } add_reg_out(ctx, "ctr"); } else { /* CTR ignored */ strcat(opcode, br_cond[((bo >> 1) & 4) | (bi & 3)]); if (cr) { strcat(operands, "cr0"); operands[2] = cr + '0'; } } if (!(bo & 0x10)) { add_reg_in(ctx, "cr"); } strcat(opcode, "lr"); add_reg_in(ctx, "lr"); if (ctx->instr_code & 1) { strcat(opcode, "l"); /* add suffix "l" if the LK bit is set */ add_reg_out(ctx, "lr"); } if (bo & 1) { /* incorporate prediction bit if set */ strcat(opcode, "+"); } ctx->instr_str = my_sprintf("%-7s %s", opcode, operands); } void opc_bx(PPCDisasmContext* ctx) { uint32_t dst = ((ctx->instr_code & 2) ? 0 : ctx->instr_addr) + SIGNEXT(ctx->instr_code & 0x3FFFFFC, 25); ctx->instr_str = my_sprintf("%-7s 0x%08X", bx_mnem[ctx->instr_code & 3], dst); if (ctx->instr_code & 1) { add_reg_in(ctx, "lr"); } } void opc_sc(PPCDisasmContext* ctx) { ctx->instr_str = my_sprintf("%-7s ", "sc"); } void opc_group19(PPCDisasmContext* ctx) { auto rb = (ctx->instr_code >> 11) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; auto rs = (ctx->instr_code >> 21) & 0x1F; int ext_opc = (ctx->instr_code >> 1) & 0x3FF; /* extract extended opcode */ char operand1[12] = ""; char operand2[12] = ""; char operand3[12] = ""; if (rs > 3) { strcat(operand1, "4*cr0+"); operand1[4] = (rs >> 2) + '0'; } strcat(operand1, br_cond[((rs % 4) + 4)]); if (ra > 3) { strcat(operand2, "4*cr0+"); operand2[4] = (ra >> 2) + '0'; } strcat(operand2, br_cond[((ra % 4) + 4)]); if (rb > 3) { strcat(operand3, "4*cr0+"); operand3[4] = (rb >> 2) + '0'; } strcat(operand3, br_cond[((rb % 4) + 4)]); switch (ext_opc) { case 0: ctx->instr_str = my_sprintf("%-7s cr%d, cr%d", "mcrf", (rs >> 2), (ra >> 2)); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); return; case 16: opc_bclrx(ctx); return; case 33: if (ctx->simplified && (ra == rb)) { ctx->instr_str = my_sprintf("%-7s %s, %s, %s", "crnor", operand1, operand2, operand3); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); } else { fmt_threeop_crb(ctx, "crnor", rs, ra, rb); } return; case 50: ctx->instr_str = my_sprintf("%-7s ", "rfi"); add_reg_in(ctx, "msr"); add_reg_in(ctx, "srr0"); add_reg_in(ctx, "srr1"); add_reg_out(ctx, "msr"); return; case 129: if (ctx->simplified) { ctx->instr_str = my_sprintf("%-7s %s, %s, %s", "crandc", operand1, operand2, operand3); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); } else { fmt_threeop_crb(ctx, "crandc", rs, ra, rb); } return; case 150: ctx->instr_str = my_sprintf("%-7s ", "isync"); return; case 193: if (ctx->simplified && (rs == ra) && (rs == rb)) { ctx->instr_str = my_sprintf("%-7s crb%d", "crclr", rs); add_reg_out(ctx, "cr"); return; } else if (ctx->simplified && ((rs != ra) || (rs != rb))) { ctx->instr_str = my_sprintf("%-7s %s, %s, %s", "crxor", operand1, operand2, operand3); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); return; } fmt_threeop_crb(ctx, "crxor", rs, ra, rb); return; case 225: if (ctx->simplified) { ctx->instr_str = my_sprintf("%-7s %s, %s, %s", "crnand", operand1, operand2, operand3); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); } else { fmt_threeop_crb(ctx, "crnand", rs, ra, rb); } return; case 257: if (ctx->simplified) { ctx->instr_str = my_sprintf("%-7s %s, %s, %s", "crand", operand1, operand2, operand3); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); } else { fmt_threeop_crb(ctx, "crand", rs, ra, rb); } return; case 289: if (ctx->simplified && (rs == ra) && (rs == rb)) { ctx->instr_str = my_sprintf("%-7s crb%d", "crset", rs); add_reg_out(ctx, "cr"); return; } else if (ctx->simplified && ((rs != ra) || (rs != rb))) { ctx->instr_str = my_sprintf("%-7s %s, %s, %s", "creqv", operand1, operand2, operand3); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); return; } fmt_threeop_crb(ctx, "creqv", rs, ra, rb); return; case 417: if (ctx->simplified) { ctx->instr_str = my_sprintf("%-7s %s, %s, %s", "crorc", operand1, operand2, operand3); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); } else { fmt_threeop_crb(ctx, "crorc", rs, ra, rb); } return; case 449: if (ctx->simplified && (ra == rb)) { ctx->instr_str = my_sprintf("%-7s crb%d, crb%d", "crmove", rs, ra); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); return; } else if (ctx->simplified && (ra != rb)) { ctx->instr_str = my_sprintf("%-7s %s, %s, %s", "cror", operand1, operand2, operand3); add_reg_in(ctx, "cr"); add_reg_out(ctx, "cr"); return; } fmt_threeop_crb(ctx, "cror", rs, ra, rb); return; case 528: opc_bcctrx(ctx); return; } } void opc_group31(PPCDisasmContext* ctx) { char opcode[10] = ""; auto rb = (ctx->instr_code >> 11) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; auto rs = (ctx->instr_code >> 21) & 0x1F; int ext_opc = (ctx->instr_code >> 1) & 0x3FF; /* extract extended opcode */ int index = ext_opc >> 5; bool rc_set = ctx->instr_code & 1; switch (ext_opc & 0x1F) { case 8: /* subtracts & friends */ index &= 0xF; /* strip OE bit */ if (!strlen(opc_subs[index])) { opc_illegal(ctx); } else { strcpy(opcode, opc_subs[index]); if (ext_opc & 0x200) /* check OE bit */ strcat(opcode, "o"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (index == 3 || index == 6 || index == 7 || index == 11 || index == 15) { /* ugly check for two-operands instructions */ if (rb != 0) opc_illegal(ctx); else fmt_twoop(ctx, opcode, rs, ra); } else fmt_threeop(ctx, opcode, rs, ra, rb); } return; case 10: /* additions */ index &= 0xF; /* strip OE bit */ if (index > 8 || !strlen(opc_adds[index])) { opc_illegal(ctx); } else { strcpy(opcode, opc_adds[index]); if (ext_opc & 0x200) /* check OE bit */ strcat(opcode, "o"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (index == 6 || index == 7) { if (rb != 0) opc_illegal(ctx); else fmt_twoop(ctx, opcode, rs, ra); } else fmt_threeop(ctx, opcode, rs, ra, rb); } return; case 11: /* integer multiplications and divisions */ index &= 0xF; /* strip OE bit */ if (!strlen(opc_muldivs[index])) { opc_illegal(ctx); } else { strcpy(opcode, opc_muldivs[index]); if (ext_opc & 0x200) /* check OE bit */ strcat(opcode, "o"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if ((!index || index == 2) && (ext_opc & 0x200)) opc_illegal(ctx); else fmt_threeop(ctx, opcode, rs, ra, rb); } return; case 0x12: /* tlb & segment register instructions */ if (index == 4) { /* mtmsr */ if ((ra != 0) || (rb != 0) || rc_set) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s r%d", "mtmsr", rs); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "msr"); } } else if (index == 6) { /* mtsr */ if (ra & 16) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s %d, r%d", "mtsr", ra, rs); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "sr%d", ra); } } else if (index == 7) { /* mtsrin */ ctx->instr_str = my_sprintf("%-7s r%d, r%d", "mtsrin", rs, rb); add_reg_in(ctx, "r%d", rs); std::string reg_name = add_reg_in(ctx, "r%d", rb); uint64_t reg_val = get_reg(reg_name); add_reg_out(ctx, "sr%d", reg_val >> 28); } else if (index == 9) { /* tlbie */ ctx->instr_str = my_sprintf("%-7s r%d", "tlbie", rb); add_reg_in(ctx, "r%d", rb); } else if (index == 11) { /* tlbia */ ctx->instr_str = my_sprintf("%-7s ", "tlbia"); } else if (index == 30) { /* tlbld - 603 only */ if (!rs && !ra) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s r%s", "tlbld", rb); add_reg_in(ctx, "r%d", rb); } } else if (index == 30) { /* tlbli - 603 only */ if (!rs && !ra) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s r%s", "tlbli", rb); add_reg_in(ctx, "r%d", rb); } } return; case 0x18: /* Shifting instructions */ strcpy(opcode, opc_shft_reg[index]); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if ((index == 0) || (index == 4) || (index == 6) || (index == 16) || (index == 20) || (index == 22) || (index == 24) || (index == 28)) { fmt_threeop(ctx, opcode, ra, rs, rb); } else if ((index == 5) || (index == 7) || (index == 21) || (index == 23) || (index == 25) || (index == 29)) { fmt_threeop_simm(ctx, opcode, ra, rs, rb); } else { opc_illegal(ctx); } return; case 0x19: /* (Extended) Shifting instructions - 601 only*/ strcpy(opcode, opc_shft_ext[index]); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if ((index == 4) || (index == 6) || (index == 16) || (index == 20) || (index == 22) || (index == 28)) { fmt_threeop(ctx, opcode, ra, rs, rb); } else { opc_illegal(ctx); } return; case 0x1A: /* Byte sign extend instructions (and cntlzw) */ if (index == 0) strcpy(opcode, "cntlzw"); else if (index == 28) strcpy(opcode, "extsh"); else if (index == 29) strcpy(opcode, "extsb"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_twoop(ctx, opcode, ra, rs); return; case 0x1C: /* logical instructions */ if (index == 13 && rs == rb && ctx->simplified) { fmt_twoop(ctx, rc_set ? "mr." : "mr", ra, rs); if (rc_set) { add_reg_out(ctx, "cr"); } } else { strcpy(opcode, opc_logic[index]); if (!strlen(opcode)) { opc_illegal(ctx); } else { if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_threeop(ctx, opcode, ra, rs, rb); } } return; case 0x17: /* indexed load/store instructions */ if (index == 30) { /* stfiwx sneaks in here */ if (rc_set) { opc_illegal(ctx); return; } else { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s f%d, 0, r%d", opc_idx_ldst[index], rs, rb); add_reg_in(ctx, "f%d", rs); add_reg_in(ctx, "r%d", rb); } else { ctx->instr_str = my_sprintf("%-7s f%d, r%d, r%d", opc_idx_ldst[index], rs, ra, rb); add_reg_in(ctx, "f%d", rs); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); } return; } } if (index > 23 || rc_set || strlen(opc_idx_ldst[index]) == 0) { opc_illegal(ctx); return; } if (index < 16) { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, r%d", opc_idx_ldst[index], rs, rb); if (opc_idx_ldst[index][0] == 'l') add_reg_out(ctx, "r%d", rs); else add_reg_in(ctx, "r%d", rs); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); } else fmt_threeop(ctx, opc_idx_ldst[index], rs, ra, rb); return; } else { ctx->instr_str = my_sprintf("%-7s f%d, r%d, r%d", opc_idx_ldst[index], rs, ra, rb); if (opc_idx_ldst[index][0] == 'l') add_reg_out(ctx, "r%d", rs); else add_reg_in(ctx, "r%d", rs); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); } return; case 0x16: /* processor mgmt + byte reversed load and store instructions */ strcpy(opcode, proc_mgmt_str[index]); if (index == 4) { /* stwcx.*/ if (!rc_set) { opc_illegal(ctx); return; } else { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, r%d", opcode, rs, rb); add_reg_in(ctx, "r%d", rs); add_reg_in(ctx, "r%d", rb); } else fmt_threeop_store(ctx, opcode, rs, ra, rb); add_reg_out(ctx, "cr"); return; } } /* eciwx, lhbrx, lwbrx */ else if ((index == 9) || (index == 16) || (index == 24)) { if (rc_set) { opc_illegal(ctx); return; } else { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, r%d", opcode, rs, rb); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "r%d", rs); } else fmt_threeop(ctx, opcode, rs, ra, rb); return; } } /* ecowx, stwbrx, sthbrx */ else if ((index == 13) || (index == 20) || (index == 28)) { if (rc_set) { opc_illegal(ctx); return; } else { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, r%d", opcode, rs, rb); add_reg_in(ctx, "r%d", rs); add_reg_in(ctx, "r%d", rb); } else fmt_threeop_store(ctx, opcode, rs, ra, rb); return; } } else if ((index == 18) || (index == 26)) { /* sync, eieio */ ctx->instr_str = my_sprintf("%-7s ", opcode); return; } /* dcba, dcbf, dcbi, dcbst, dcbt, dcbz, icbi */ else if ( (index == 1) || (index == 2) || (index == 7) || (index == 8) || (index == 14) || (index == 23) || (index == 30) || (index == 31)) { if (rc_set || (rs != 0)) { opc_illegal(ctx); return; } else { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s 0, r%d", opcode, rb); add_reg_in(ctx, "r%d", rb); } else fmt_twoop_in(ctx, opcode, ra, rb); return; } } else if (index == 17) { /* tlbsync */ ctx->instr_str = my_sprintf("%-7s ", opcode); return; } else { opc_illegal(ctx); } return; break; } auto ref_spr = (((ctx->instr_code >> 11) & 31) << 5) | ((ctx->instr_code >> 16) & 31); auto spr_high = (ctx->instr_code >> 11) & 31; auto spr_low = (ctx->instr_code >> 16) & 31; switch (ext_opc) { case 0: /* cmp */ if (rc_set) { opc_illegal(ctx); return; } if (ctx->simplified) { if (!(rs & 1)) { if ((rs >> 2) == 0) { ctx->instr_str = my_sprintf("%-7s r%d, r%d", "cmpw", ra, rb); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "cr"); return; } else { ctx->instr_str = my_sprintf("%-7s cr%d, r%d, r%d", "cmpw", (rs >> 2), ra, rb); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "cr"); return; } } } ctx->instr_str = my_sprintf("%-7s cr%d, r%d, r%d", "cmp", (rs >> 2), ra, rb); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "cr"); break; case 4: /* tw */ if (rc_set) { opc_illegal(ctx); } else { if (ctx->simplified) { strcpy(opcode, trap_cond[rs]); if (strlen(opcode) != 0) { ctx->instr_str = my_sprintf("%-7s r%d, r%d", opcode, ra, rb); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); break; } } ctx->instr_str = my_sprintf("%-7s %d, r%d, r%d", "tw", rs, ra, rb); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); } break; case 19: /* mfcr */ fmt_oneop_out(ctx, "mfcr", rs); add_reg_in(ctx, "cr"); break; case 20: /* lwarx */ if (rc_set) { opc_illegal(ctx); } else { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, r%d", "lwarx", rs, rb); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "r%d", rs); } else fmt_threeop(ctx, "lwarx", rs, ra, rb); } break; case 29: /* maskg */ strcpy(opcode, "maskg"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_threeop(ctx, opcode, rs, ra, rb); break; case 32: /* cmpl */ if (rc_set) { opc_illegal(ctx); return; } if (ctx->simplified) { if (!(rs & 1)) { if ((rs >> 2) == 0) ctx->instr_str = my_sprintf("%-7s r%d, r%d", "cmplw", ra, rb); else ctx->instr_str = my_sprintf("%-7s cr%d, r%d, r%d", "cmplw", (rs >> 2), ra, rb); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "cr"); return; } } else { ctx->instr_str = my_sprintf( "%-7s cr%d, %d, r%d, r%d", "cmpl", (rs >> 2), (rs & 1), ra, rb); add_reg_in(ctx, "r%d", ra); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "cr"); } break; case 83: /* mfmsr */ ctx->instr_str = my_sprintf("%-7s r%d", "mfmsr", rs); add_reg_in(ctx, "msr"); add_reg_out(ctx, "r%d", rs); break; case 144: /* mtcrf */ if (ctx->instr_code & 0x100801) opc_illegal(ctx); else { if (((ctx->instr_code >> 12) & 0xFF) == 0xFF) ctx->instr_str = my_sprintf("%-7s r%d", "mtcr", rs); else ctx->instr_str = my_sprintf( "%-7s 0x%02X, r%d", "mtcrf", (ctx->instr_code >> 12) & 0xFF, rs); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "cr"); } break; case 277: /* lscbx */ strcpy(opcode, "lscbx"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } add_reg_in(ctx, "xer"); add_reg_out(ctx, "xer"); if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, r%d", opcode, rs, rb); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "r%d", rs); /* string operation: additional registers are affected */ } else fmt_threeop(ctx, opcode, rs, ra, rb); break; case 339: /* mfspr */ if (ctx->simplified) { switch (spr_high) { case 0: strcpy(opcode, "mf"); strcat(opcode, spr_index0[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "%s", spr_index0[spr_low]); add_reg_out(ctx, "r%d", rs); return; case 8: strcpy(opcode, "mf"); strcat(opcode, spr_index8[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "%s", spr_index8[spr_low]); add_reg_out(ctx, "r%d", rs); return; case 16: strcpy(opcode, "mf"); strcat(opcode, spr_index16[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "%s", spr_index16[spr_low]); add_reg_out(ctx, "r%d", rs); return; case 29: strcpy(opcode, "mf"); strcat(opcode, spr_index29[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "%s", spr_index29[spr_low]); add_reg_out(ctx, "r%d", rs); return; case 30: strcpy(opcode, "mf"); strcat(opcode, spr_index30[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "%s", spr_index30[spr_low]); add_reg_out(ctx, "r%d", rs); return; case 31: strcpy(opcode, "mf"); strcat(opcode, spr_index31[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "%s", spr_index31[spr_low]); add_reg_out(ctx, "r%d", rs); return; } } fmt_twoop_fromspr(ctx, "mfspr", rs, ref_spr); break; case 371: /* mftb */ if (ctx->simplified) { if (ref_spr == 268) { fmt_oneop_out(ctx, "mftbl", rs); add_reg_in(ctx, "tbl"); return; } if (ref_spr == 269) { fmt_oneop_out(ctx, "mftbu", rs); add_reg_in(ctx, "tbu"); return; } } ctx->instr_str = my_sprintf("%-7s r%d, %d", "mftb", rs, ref_spr); if (ref_spr == 268) { add_reg_in(ctx, "tbl"); } else if (ref_spr == 269) { add_reg_in(ctx, "tbu"); } else { add_reg_in(ctx, "spr", ref_spr); } add_reg_out(ctx, "r%d", rs); break; case 467: /* mtspr */ if (ctx->simplified) { switch (spr_high) { case 0: strcpy(opcode, "mt"); strcat(opcode, spr_index0[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "r%d", rs); if (spr_low == 0x16) { /* decrementer */ add_reg_in(ctx, "msr"); } add_reg_out(ctx, "%s", spr_index0[spr_low]); return; case 8: strcpy(opcode, "mt"); strcat(opcode, spr_index8[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "%s", spr_index8[spr_low]); return; case 16: strcpy(opcode, "mt"); strcat(opcode, spr_index16[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "%s", spr_index16[spr_low]); return; case 29: strcpy(opcode, "mt"); strcat(opcode, spr_index29[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "%s", spr_index29[spr_low]); return; case 30: strcpy(opcode, "mt"); strcat(opcode, spr_index30[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "%s", spr_index30[spr_low]); return; case 31: strcpy(opcode, "mt"); strcat(opcode, spr_index31[spr_low]); ctx->instr_str = my_sprintf("%-7s r%d", opcode, rs); add_reg_in(ctx, "r%d", rs); add_reg_out(ctx, "%s", spr_index31[spr_low]); return; } } fmt_twoop_tospr(ctx, "mtspr", ref_spr, rs); break; case 512: /* mcrxr */ ctx->instr_str = my_sprintf("%-7s cr%d", "mcrxr", (rs >> 2)); add_reg_in(ctx, "xer"); add_reg_out(ctx, "cr"); add_reg_out(ctx, "xer"); break; case 531: /* clcs */ strcpy(opcode, "clcs"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_twoop(ctx, opcode, rs, ra); break; case 533: /* lswx */ if (rc_set) { opc_illegal(ctx); } else { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, r%d", "lswx", rs, rb); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "r%d", rs); /* string operation: additional registers are affected */ } else fmt_threeop(ctx, "lswx", rs, ra, rb); } break; case 541: /* maskir */ strcpy(opcode, "maskir"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_threeop(ctx, opcode, ra, rs, rb); return; case 595: /* mfsr */ if (ra & 16) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s r%d, %d", "mfsr", rs, ra); add_reg_in(ctx, "sr%d", ra); add_reg_out(ctx, "r%d", rs); } break; case 597: /* lswi */ if (rc_set) { opc_illegal(ctx); } else { if (rb == 0) rb = 32; if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, %x", "lswi", rs, rb); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "r%d", rs); /* string operation: additional registers are affected */ } else fmt_threeop_simm(ctx, "lswi", rs, ra, rb); } break; case 659: /* mfsrin */ { ctx->instr_str = my_sprintf("%-7s r%d, r%d", "mfsrin", rs, rb); add_reg_out(ctx, "r%d", rs); std::string reg_name = add_reg_in(ctx, "r%d", rb); uint64_t reg_val = get_reg(reg_name); add_reg_in(ctx, "sr%d", reg_val >> 28); } break; case 661: /* stswx */ if (rc_set) { opc_illegal(ctx); return; } else { if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, r%d", "stswx", rs, rb); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "r%d", rs); /* string operation: additional registers are affected */ } else fmt_threeop(ctx, "stswx", rs, ra, rb); return; } break; case 725: /* stswi */ if (rc_set) { opc_illegal(ctx); return; } else { if (rb == 0) rb = 32; if (ra == 0) { ctx->instr_str = my_sprintf("%-7s r%d, 0, %d", "stswi", rs, rb); add_reg_in(ctx, "r%d", rb); add_reg_out(ctx, "r%d", rs); /* string operation: additional registers are affected */ } else fmt_threeop_simm(ctx, "stswi", rs, ra, rb); return; } break; default: opc_illegal(ctx); } } void opc_group59(PPCDisasmContext* ctx) { char opcode[10] = ""; auto rc = (ctx->instr_code >> 6) & 0x1F; auto rb = (ctx->instr_code >> 11) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; auto rs = (ctx->instr_code >> 21) & 0x1F; int ext_opc = (ctx->instr_code >> 1) & 0x3FF; /* extract extended opcode */ bool rc_set = ctx->instr_code & 1; switch (ext_opc & 0x1F) { case 18: /* floating point division */ strcpy(opcode, "fdivs"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (rc != 0) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rb); add_reg_out(ctx, "fpscr"); } return; case 20: /* floating point subtract */ strcpy(opcode, "fsubs"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (rc != 0) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rb); add_reg_out(ctx, "fpscr"); } return; case 21: /* floating point addition */ strcpy(opcode, "fadds"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (rc != 0) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rb); add_reg_out(ctx, "fpscr"); } return; case 22: /* floating point square root */ strcpy(opcode, "fsqrts"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if ((rc != 0) || (ra != 0)) opc_illegal(ctx); else { fmt_twoop_flt(ctx, "fsqrts", rs, rb); add_reg_out(ctx, "fpscr"); } return; case 24: /* fres */ strcpy(opcode, "fres"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if ((rc != 0) || (ra != 0)) opc_illegal(ctx); else { fmt_twoop_flt(ctx, opcode, rs, rb); add_reg_out(ctx, "fpscr"); } return; case 25: /* fmuls */ strcpy(opcode, opc_flt_ext_arith[25]); strcat(opcode, "s"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (rb != 0) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rc); add_reg_out(ctx, "fpscr"); } return; case 28: /* fmsubs */ strcpy(opcode, opc_flt_ext_arith[28]); strcat(opcode, "s"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); add_reg_out(ctx, "fpscr"); return; case 29: /* fmadds */ strcpy(opcode, opc_flt_ext_arith[29]); strcat(opcode, "s"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); add_reg_out(ctx, "fpscr"); return; case 30: /* fnmsubs */ strcpy(opcode, opc_flt_ext_arith[30]); strcat(opcode, "s"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); add_reg_out(ctx, "fpscr"); return; case 31: /* fnmadds */ strcpy(opcode, opc_flt_ext_arith[31]); strcat(opcode, "s"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); add_reg_out(ctx, "fpscr"); return; } } void opc_group63(PPCDisasmContext* ctx) { char opcode[10] = ""; auto rc = (ctx->instr_code >> 6) & 0x1F; auto rb = (ctx->instr_code >> 11) & 0x1F; auto ra = (ctx->instr_code >> 16) & 0x1F; auto rs = (ctx->instr_code >> 21) & 0x1F; int ext_opc = (ctx->instr_code >> 1) & 0x3FF; /* extract extended opcode */ bool rc_set = ctx->instr_code & 1; switch (ext_opc & 0x1F) { case 18: /* floating point division */ strcpy(opcode, "fdiv"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (rc != 0) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rb); add_reg_out(ctx, "fpscr"); } return; case 20: /* floating point subtract */ strcpy(opcode, "fsub"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (rc != 0) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rb); add_reg_out(ctx, "fpscr"); } return; case 21: /* floating point addition */ strcpy(opcode, "fadd"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (rc != 0) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rb); add_reg_out(ctx, "fpscr"); } return; case 22: /* floating point square root */ strcpy(opcode, "fsqrt"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if ((rc != 0) || (ra != 0)) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rb); add_reg_out(ctx, "fpscr"); } return; case 23: /* fsel */ strcpy(opcode, opc_flt_ext_arith[23]); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); return; case 25: /* fmul */ strcpy(opcode, opc_flt_ext_arith[25]); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (rb != 0) opc_illegal(ctx); else { fmt_threeop_flt(ctx, opcode, rs, ra, rc); add_reg_out(ctx, "fpscr"); } return; case 26: /* frsqrte */ strcpy(opcode, "frsqrte"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if ((rc != 0) || (ra != 0)) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s f%d, f%d", opcode, rs, rb); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "f%d", rs); add_reg_out(ctx, "fpscr"); } return; case 28: /* fmsub */ strcpy(opcode, opc_flt_ext_arith[28]); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); return; case 29: /* fmadd */ strcpy(opcode, opc_flt_ext_arith[29]); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); add_reg_out(ctx, "fpscr"); return; case 30: /* fnmsub */ strcpy(opcode, opc_flt_ext_arith[30]); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); add_reg_out(ctx, "fpscr"); return; case 31: /* fnmadd */ strcpy(opcode, opc_flt_ext_arith[31]); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } fmt_fourop_flt(ctx, opcode, rs, ra, rc, rb); add_reg_out(ctx, "fpscr"); return; } auto fm = (ctx->instr_code >> 17) & 0xFF; switch (ext_opc) { case 0: /* fcmpu */ if (rs & 3) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s cr%d, f%d, f%d", "fcmpu", (rs >> 2), ra, rb); add_reg_in(ctx, "f%d", ra); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "cr"); } break; case 12: /* frsp */ if (ra != 0) opc_illegal(ctx); else { strcpy(opcode, "frsp"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s f%d, f%d", opcode, rs, rb); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "f%d", rs); add_reg_out(ctx, "fpscr"); } break; case 14: /* fctiw */ if (ra != 0) opc_illegal(ctx); else { strcpy(opcode, "fctiw"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s f%d, f%d", opcode, rs, rb); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "f%d", rs); add_reg_out(ctx, "fpscr"); } break; case 15: /* fctiwz */ if (ra != 0) opc_illegal(ctx); else { strcpy(opcode, "fctiwz"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s f%d, f%d", opcode, rs, rb); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "f%d", rs); add_reg_out(ctx, "fpscr"); } break; case 32: /* fcmpo */ if (rs & 3) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s cr%d, f%d, f%d", "fcmpo", (rs >> 2), ra, rb); add_reg_in(ctx, "f%d", rb); add_reg_in(ctx, "f%d", rs); add_reg_out(ctx, "cr"); } break; case 38: /* mtfsb1 */ strcpy(opcode, "mtfsb1"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s %d", opcode, rs); add_reg_out(ctx, "fpscr"); break; case 40: /* fneg */ strcpy(opcode, "fneg"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if (ra != 0) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s f%d, f%d", opcode, rs, rb); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "f%d", rs); } break; case 64: strcpy(opcode, "mcrfs"); ctx->instr_str = my_sprintf("%-7s cr%d, cr%d", opcode, (rs >> 2), (ra >> 2)); add_reg_in(ctx, "fpscr"); add_reg_out(ctx, "cr"); break; case 70: /* mtfsb0 */ strcpy(opcode, "mtfsb0"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s %d", opcode, rs); add_reg_out(ctx, "fpscr"); break; case 72: /* fmr */ if (ra != 0) opc_illegal(ctx); else { strcpy(opcode, "fmr"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s f%d, f%d", opcode, rs, rb); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "f%d", rs); } break; case 134: /* mtfsfi */ if (ra != 0) opc_illegal(ctx); else { strcpy(opcode, "mtfsfi"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s cr%d, %d", opcode, (rs >> 2), (rb >> 1)); add_reg_out(ctx, "fpscr"); } break; case 136: /* fnabs */ if (ra != 0) opc_illegal(ctx); else { strcpy(opcode, "fnabs"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s f%d, f%d", opcode, rs, rb); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "f%d", rs); } break; case 264: /* fabs */ if (ra != 0) opc_illegal(ctx); else { strcpy(opcode, "fabs"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s f%d, f%d", opcode, rs, rb); add_reg_in(ctx, "f%d", rb); add_reg_out(ctx, "f%d", rs); } break; case 583: /* mffs */ strcpy(opcode, "mffs"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } if ((ra != 0) || (rb != 0)) opc_illegal(ctx); else { ctx->instr_str = my_sprintf("%-7s f%d", opcode, rs); add_reg_in(ctx, "fpscr"); add_reg_out(ctx, "f%d", rs); } break; case 711: /* mtfsf */ strcpy(opcode, "mtfsf"); if (rc_set) { strcat(opcode, "."); add_reg_out(ctx, "cr"); } ctx->instr_str = my_sprintf("%-7s %d, f%d", opcode, fm, rb); add_reg_in(ctx, "f%d", rs); add_reg_out(ctx, "fpscr"); break; default: opc_illegal(ctx); } } void opc_intldst(PPCDisasmContext* ctx) { int32_t opcode = (ctx->instr_code >> 26) - 32; int32_t ra = (ctx->instr_code >> 16) & 0x1F; int32_t rd = (ctx->instr_code >> 21) & 0x1F; int32_t imm = SIGNEXT(ctx->instr_code & 0xFFFF, 15); /* ra = 0 is forbidden for loads and stores with update */ /* ra = rd is forbidden for loads with update */ if (((opcode < 14) && (opcode & 5) == 1 && ra == rd) || ((opcode & 1) && !ra)) { opc_illegal(ctx); return; } if (opc_int_ldst[opcode][0] == 'l') add_reg_out(ctx, "r%d", rd); else add_reg_in(ctx, "r%d", rd); if (ra) { ctx->instr_str = my_sprintf( "%-7s r%d, %s0x%X(r%d)", opc_int_ldst[opcode], rd, ((imm < 0) ? "-" : ""), abs(imm), ra); add_reg_in(ctx, "r%d", ra); } else { ctx->instr_str = my_sprintf( "%-7s r%d, %s0x%X", opc_int_ldst[opcode], rd, ((imm < 0) ? "-" : ""), abs(imm)); } } void opc_fltldst(PPCDisasmContext* ctx) { int32_t opcode = (ctx->instr_code >> 26) - 48; int32_t ra = (ctx->instr_code >> 16) & 0x1F; int32_t rd = (ctx->instr_code >> 21) & 0x1F; int32_t imm = SIGNEXT(ctx->instr_code & 0xFFFF, 15); /* ra = 0 is forbidden for loads and stores with update */ /* ra = rd is forbidden for loads with update */ if ((((opcode == 1) || (opcode == 3)) && ra == rd) || ((opcode & 1) && !ra)) { opc_illegal(ctx); return; } if (opc_flt_ldst[opcode][0] == 'l') add_reg_out(ctx, "f%d", rd); else add_reg_in(ctx, "f%d", rd); if (ra) { ctx->instr_str = my_sprintf( "%-7s f%d, %s0x%X(r%d)", opc_flt_ldst[opcode], rd, ((imm < 0) ? "-" : ""), abs(imm), ra); add_reg_in(ctx, "r%d", ra); } else { ctx->instr_str = my_sprintf( "%-7s f%d, %s0x%X", opc_flt_ldst[opcode], rd, ((imm < 0) ? "-" : ""), abs(imm)); } } /** main dispatch table. */ static std::function OpcodeDispatchTable[64] = { opc_illegal, opc_illegal, opc_illegal, opc_twi, opc_group4, opc_illegal, opc_illegal, opc_ar_im, opc_ar_im, power_dozi, opc_cmp_i_li, opc_cmp_i_li, opc_ar_im, opc_ar_im, opc_ar_im, opc_ar_im, opc_bcx, opc_sc, opc_bx, opc_group19, opc_rlwimi, opc_rlwinm, opc_rlmi, opc_rlwnm, opc_bool_im, opc_bool_im, opc_bool_im, opc_bool_im, opc_bool_im, opc_bool_im, opc_illegal, opc_group31, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_intldst, opc_fltldst, opc_fltldst, opc_fltldst, opc_fltldst, opc_fltldst, opc_fltldst, opc_fltldst, opc_fltldst, opc_illegal, opc_illegal, opc_illegal, opc_group59, opc_illegal, opc_illegal, opc_illegal, opc_group63}; string disassemble_single(PPCDisasmContext* ctx) { if (ctx->instr_addr & 3) { throw std::invalid_argument(string("PPC instruction address must be a multiply of 4!")); } OpcodeDispatchTable[ctx->instr_code >> 26](ctx); ctx->instr_addr += 4; return ctx->instr_str; }