1*9a0e4156SSadaf Ebrahimi // Capstone Java binding 2*9a0e4156SSadaf Ebrahimi /* M680X Backend by Wolfgang Schwotzer <[email protected]> 2017 */ 3*9a0e4156SSadaf Ebrahimi 4*9a0e4156SSadaf Ebrahimi import java.lang.*; 5*9a0e4156SSadaf Ebrahimi import capstone.Capstone; 6*9a0e4156SSadaf Ebrahimi import capstone.M680x; 7*9a0e4156SSadaf Ebrahimi 8*9a0e4156SSadaf Ebrahimi import static capstone.M680x_const.*; 9*9a0e4156SSadaf Ebrahimi 10*9a0e4156SSadaf Ebrahimi public class TestM680x { 11*9a0e4156SSadaf Ebrahimi 12*9a0e4156SSadaf Ebrahimi static final String sAccess[] = { 13*9a0e4156SSadaf Ebrahimi "UNCHANGED", "READ", "WRITE", "READ | WRITE", 14*9a0e4156SSadaf Ebrahimi }; 15*9a0e4156SSadaf Ebrahimi 16*9a0e4156SSadaf Ebrahimi static final String M6800_CODE = "010936647f7410009010A410b6100039"; 17*9a0e4156SSadaf Ebrahimi static final String M6801_CODE = "04053c3d389310ec10ed1039"; 18*9a0e4156SSadaf Ebrahimi static final String M6805_CODE = "047f00172228002e0040425a708e979ca015ad00c31000da1234e57ffe"; 19*9a0e4156SSadaf Ebrahimi static final String M6808_CODE = "31220035224510004b005110525e226265123472848586878a8b8c9495a710af109e607f9e6b7f009ed610009ee67f"; 20*9a0e4156SSadaf Ebrahimi static final String HD6301_CODE = "6b100071100072101039"; 21*9a0e4156SSadaf Ebrahimi static final String M6809_CODE = "0610191a551e0123e931063455a681a7897fffa69d1000a791a69f100011ac99100039A607A627A647A667A60FA610A680A681A682A683A684A685A686A6887FA68880A6897FFFA6898000A68BA68C10A68D1000A691A693A694A695A696A6987FA69880A6997FFFA6998000A69BA69C10A69D1000A69F1000"; 22*9a0e4156SSadaf Ebrahimi static final String M6811_CODE = "0203127f100013990800147f02157f011e7f20008fcf18081830183c1867188c1000188f18ce100018ff10001aa37f1aac1aee7f1aef7fcdac7f"; 23*9a0e4156SSadaf Ebrahimi static final String CPU12_CODE = "000401000c00800e008000111e100080003b4a1000044b01044f7f80008f1000b752b7b1a667a6fea6f71802e23039e21000180c30391000181118121000181900181e00183e183f00"; 24*9a0e4156SSadaf Ebrahimi static final String HD6309_CODE = "0110106210107b101000cd499602d21030231038103b1053105d1130431011372510113812113923113b34118e100011af1011ab1011f68000"; 25*9a0e4156SSadaf Ebrahimi static final String HCS08_CODE = "3210009eae9ece7f9ebe10009efe7f3e10009ef37f9610009eff7f82"; 26*9a0e4156SSadaf Ebrahimi hexString2Byte(String s)27*9a0e4156SSadaf Ebrahimi static byte[] hexString2Byte(String s) { 28*9a0e4156SSadaf Ebrahimi // from http://stackoverflow.com/questions/140131/convert-a-string-representation-of-a-hex-dump-to-a-byte-array-using-java 29*9a0e4156SSadaf Ebrahimi int len = s.length(); 30*9a0e4156SSadaf Ebrahimi byte[] data = new byte[len / 2]; 31*9a0e4156SSadaf Ebrahimi for (int i = 0; i < len; i += 2) { 32*9a0e4156SSadaf Ebrahimi data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4) 33*9a0e4156SSadaf Ebrahimi + Character.digit(s.charAt(i+1), 16)); 34*9a0e4156SSadaf Ebrahimi } 35*9a0e4156SSadaf Ebrahimi return data; 36*9a0e4156SSadaf Ebrahimi } 37*9a0e4156SSadaf Ebrahimi stringToHexUc(byte[] code)38*9a0e4156SSadaf Ebrahimi static public String stringToHexUc(byte[] code) { 39*9a0e4156SSadaf Ebrahimi StringBuilder buf = new StringBuilder(800); 40*9a0e4156SSadaf Ebrahimi for (byte ch: code) { 41*9a0e4156SSadaf Ebrahimi buf.append(String.format(" 0x%02X", ch)); 42*9a0e4156SSadaf Ebrahimi } 43*9a0e4156SSadaf Ebrahimi return buf.toString(); 44*9a0e4156SSadaf Ebrahimi } 45*9a0e4156SSadaf Ebrahimi stringToHexShortUc(byte[] code)46*9a0e4156SSadaf Ebrahimi static public String stringToHexShortUc(byte[] code) { 47*9a0e4156SSadaf Ebrahimi StringBuilder buf = new StringBuilder(800); 48*9a0e4156SSadaf Ebrahimi for (byte ch: code) { 49*9a0e4156SSadaf Ebrahimi buf.append(String.format("%02X", ch)); 50*9a0e4156SSadaf Ebrahimi } 51*9a0e4156SSadaf Ebrahimi return buf.toString(); 52*9a0e4156SSadaf Ebrahimi } 53*9a0e4156SSadaf Ebrahimi 54*9a0e4156SSadaf Ebrahimi public static Capstone cs; 55*9a0e4156SSadaf Ebrahimi /* 56*9a0e4156SSadaf Ebrahimi private static String hex(int i) { 57*9a0e4156SSadaf Ebrahimi return Integer.toString(i, 16); 58*9a0e4156SSadaf Ebrahimi } 59*9a0e4156SSadaf Ebrahimi 60*9a0e4156SSadaf Ebrahimi private static String hex(long i) { 61*9a0e4156SSadaf Ebrahimi return Long.toString(i, 16); 62*9a0e4156SSadaf Ebrahimi } 63*9a0e4156SSadaf Ebrahimi */ print_ins_detail(Capstone.CsInsn ins)64*9a0e4156SSadaf Ebrahimi public static void print_ins_detail(Capstone.CsInsn ins) { 65*9a0e4156SSadaf Ebrahimi String bytes = stringToHexShortUc(ins.bytes); 66*9a0e4156SSadaf Ebrahimi System.out.printf("0x%04X:\t%s\t%s\t%s\n", ins.address, bytes, ins.mnemonic, ins.opStr); 67*9a0e4156SSadaf Ebrahimi 68*9a0e4156SSadaf Ebrahimi M680x.OpInfo operands = (M680x.OpInfo) ins.operands; 69*9a0e4156SSadaf Ebrahimi 70*9a0e4156SSadaf Ebrahimi if (operands.op.length != 0) { 71*9a0e4156SSadaf Ebrahimi System.out.printf("\top_count: %d\n", operands.op.length); 72*9a0e4156SSadaf Ebrahimi for (int c = 0; c < operands.op.length; c++) { 73*9a0e4156SSadaf Ebrahimi M680x.Operand i = (M680x.Operand) operands.op[c]; 74*9a0e4156SSadaf Ebrahimi if (i.type == M680X_OP_REGISTER) { 75*9a0e4156SSadaf Ebrahimi String comment = ""; 76*9a0e4156SSadaf Ebrahimi if ((c == 0 && ((operands.flags & M680X_FIRST_OP_IN_MNEM) != 0)) || 77*9a0e4156SSadaf Ebrahimi (c == 1 && ((operands.flags & M680X_SECOND_OP_IN_MNEM) != 0))) 78*9a0e4156SSadaf Ebrahimi comment = " (in mnemonic)"; 79*9a0e4156SSadaf Ebrahimi System.out.printf("\t\toperands[%d].type: REGISTER = %s%s\n", c, ins.regName(i.value.reg), comment); 80*9a0e4156SSadaf Ebrahimi } 81*9a0e4156SSadaf Ebrahimi if (i.type == M680X_OP_CONSTANT) 82*9a0e4156SSadaf Ebrahimi System.out.printf("\t\toperands[%d].type: CONSTANT = %d\n", c, i.value.const_val); 83*9a0e4156SSadaf Ebrahimi if (i.type == M680X_OP_IMMEDIATE) 84*9a0e4156SSadaf Ebrahimi System.out.printf("\t\toperands[%d].type: IMMEDIATE = #%d\n", c, i.value.imm); 85*9a0e4156SSadaf Ebrahimi if (i.type == M680X_OP_DIRECT) 86*9a0e4156SSadaf Ebrahimi System.out.printf("\t\toperands[%d].type: DIRECT = 0x%02X\n", c, i.value.direct_addr); 87*9a0e4156SSadaf Ebrahimi if (i.type == M680X_OP_EXTENDED) 88*9a0e4156SSadaf Ebrahimi System.out.printf("\t\toperands[%d].type: EXTENDED %s = 0x%04X\n", c, 89*9a0e4156SSadaf Ebrahimi i.value.ext.indirect != 0 ? "INDIRECT" : "", i.value.ext.address); 90*9a0e4156SSadaf Ebrahimi if (i.type == M680X_OP_RELATIVE) 91*9a0e4156SSadaf Ebrahimi System.out.printf("\t\toperands[%d].type: RELATIVE = 0x%04X\n", c, i.value.rel.address ); 92*9a0e4156SSadaf Ebrahimi if (i.type == M680X_OP_INDEXED) { 93*9a0e4156SSadaf Ebrahimi System.out.printf("\t\toperands[%d].type: INDEXED%s\n", c, 94*9a0e4156SSadaf Ebrahimi (i.value.idx.flags & M680X_IDX_INDIRECT) != 0 ? " INDIRECT" : ""); 95*9a0e4156SSadaf Ebrahimi if (i.value.idx.base_reg != M680X_REG_INVALID) { 96*9a0e4156SSadaf Ebrahimi String regName = ins.regName(i.value.idx.base_reg); 97*9a0e4156SSadaf Ebrahimi if (regName != null) 98*9a0e4156SSadaf Ebrahimi System.out.printf("\t\t\tbase register: %s\n", regName); 99*9a0e4156SSadaf Ebrahimi } 100*9a0e4156SSadaf Ebrahimi if (i.value.idx.offset_reg != M680X_REG_INVALID) { 101*9a0e4156SSadaf Ebrahimi String regName = ins.regName(i.value.idx.offset_reg); 102*9a0e4156SSadaf Ebrahimi if (regName != null) 103*9a0e4156SSadaf Ebrahimi System.out.printf("\t\t\toffset register: %s\n", regName); 104*9a0e4156SSadaf Ebrahimi } 105*9a0e4156SSadaf Ebrahimi if ((i.value.idx.offset_bits != 0) && 106*9a0e4156SSadaf Ebrahimi (i.value.idx.offset_reg == M680X_REG_INVALID) && 107*9a0e4156SSadaf Ebrahimi (i.value.idx.inc_dec == 0)) { 108*9a0e4156SSadaf Ebrahimi System.out.printf("\t\t\toffset: %d\n", i.value.idx.offset); 109*9a0e4156SSadaf Ebrahimi if (i.value.idx.base_reg == M680X_REG_PC) 110*9a0e4156SSadaf Ebrahimi System.out.printf("\t\t\toffset address: 0x%04X\n", i.value.idx.offset_addr); 111*9a0e4156SSadaf Ebrahimi System.out.printf("\t\t\toffset bits: %d\n", i.value.idx.offset_bits); 112*9a0e4156SSadaf Ebrahimi } 113*9a0e4156SSadaf Ebrahimi if (i.value.idx.inc_dec != 0) { 114*9a0e4156SSadaf Ebrahimi String post_pre = 115*9a0e4156SSadaf Ebrahimi (i.value.idx.flags & M680X_IDX_POST_INC_DEC) != 0 ? 116*9a0e4156SSadaf Ebrahimi "post" : "pre"; 117*9a0e4156SSadaf Ebrahimi String inc_dec = 118*9a0e4156SSadaf Ebrahimi i.value.idx.inc_dec > 0 ? "increment" : "decrement"; 119*9a0e4156SSadaf Ebrahimi 120*9a0e4156SSadaf Ebrahimi System.out.printf("\t\t\t%s %s: %d\n", post_pre, inc_dec, 121*9a0e4156SSadaf Ebrahimi Math.abs(i.value.idx.inc_dec)); 122*9a0e4156SSadaf Ebrahimi } 123*9a0e4156SSadaf Ebrahimi } 124*9a0e4156SSadaf Ebrahimi if (i.size != 0) 125*9a0e4156SSadaf Ebrahimi System.out.printf("\t\t\tsize: %d\n", i.size); 126*9a0e4156SSadaf Ebrahimi if (i.access != Capstone.CS_AC_INVALID) 127*9a0e4156SSadaf Ebrahimi System.out.printf("\t\t\taccess: %s\n", sAccess[i.access]); 128*9a0e4156SSadaf Ebrahimi } 129*9a0e4156SSadaf Ebrahimi } 130*9a0e4156SSadaf Ebrahimi 131*9a0e4156SSadaf Ebrahimi if (ins.regsRead.length > 0) { 132*9a0e4156SSadaf Ebrahimi System.out.printf("\tRegisters read:"); 133*9a0e4156SSadaf Ebrahimi for (int c = 0; c < ins.regsRead.length; c++) { 134*9a0e4156SSadaf Ebrahimi System.out.printf(" %s", ins.regName(ins.regsRead[c])); 135*9a0e4156SSadaf Ebrahimi } 136*9a0e4156SSadaf Ebrahimi System.out.printf("\n"); 137*9a0e4156SSadaf Ebrahimi } 138*9a0e4156SSadaf Ebrahimi 139*9a0e4156SSadaf Ebrahimi if (ins.regsWrite.length > 0) { 140*9a0e4156SSadaf Ebrahimi System.out.printf("\tRegisters modified:"); 141*9a0e4156SSadaf Ebrahimi for (int c = 0; c < ins.regsWrite.length; c++) { 142*9a0e4156SSadaf Ebrahimi System.out.printf(" %s", ins.regName(ins.regsWrite[c])); 143*9a0e4156SSadaf Ebrahimi } 144*9a0e4156SSadaf Ebrahimi System.out.printf("\n"); 145*9a0e4156SSadaf Ebrahimi } 146*9a0e4156SSadaf Ebrahimi 147*9a0e4156SSadaf Ebrahimi if (ins.groups.length > 0) 148*9a0e4156SSadaf Ebrahimi System.out.printf("\tgroups_count: %d\n", ins.groups.length); 149*9a0e4156SSadaf Ebrahimi } 150*9a0e4156SSadaf Ebrahimi main(String argv[])151*9a0e4156SSadaf Ebrahimi public static void main(String argv[]) { 152*9a0e4156SSadaf Ebrahimi 153*9a0e4156SSadaf Ebrahimi final TestBasic.platform[] all_tests = { 154*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 155*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_6301, 156*9a0e4156SSadaf Ebrahimi hexString2Byte(HD6301_CODE), "M680X_HD6301"), 157*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 158*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_6309, 159*9a0e4156SSadaf Ebrahimi hexString2Byte(HD6309_CODE), "M680X_HD6309"), 160*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 161*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_6800, 162*9a0e4156SSadaf Ebrahimi hexString2Byte(M6800_CODE), "M680X_M6800"), 163*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 164*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_6801, 165*9a0e4156SSadaf Ebrahimi hexString2Byte(M6801_CODE), "M680X_M6801"), 166*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 167*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_6805, 168*9a0e4156SSadaf Ebrahimi hexString2Byte(M6805_CODE), "M680X_M68HC05"), 169*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 170*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_6808, 171*9a0e4156SSadaf Ebrahimi hexString2Byte(M6808_CODE), "M680X_M68HC08"), 172*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 173*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_6809, 174*9a0e4156SSadaf Ebrahimi hexString2Byte(M6809_CODE), "M680X_M6809"), 175*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 176*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_6811, 177*9a0e4156SSadaf Ebrahimi hexString2Byte(M6811_CODE), "M680X_M68HC11"), 178*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 179*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_CPU12, 180*9a0e4156SSadaf Ebrahimi hexString2Byte(CPU12_CODE), "M680X_CPU12"), 181*9a0e4156SSadaf Ebrahimi new TestBasic.platform(Capstone.CS_ARCH_M680X, 182*9a0e4156SSadaf Ebrahimi Capstone.CS_MODE_M680X_HCS08, 183*9a0e4156SSadaf Ebrahimi hexString2Byte(HCS08_CODE), "M680X_HCS08"), 184*9a0e4156SSadaf Ebrahimi }; 185*9a0e4156SSadaf Ebrahimi 186*9a0e4156SSadaf Ebrahimi for (int i=0; i<all_tests.length; i++) { 187*9a0e4156SSadaf Ebrahimi TestBasic.platform test = all_tests[i]; 188*9a0e4156SSadaf Ebrahimi System.out.println(new String(new char[20]).replace("\0", "*")); 189*9a0e4156SSadaf Ebrahimi System.out.println("Platform: " + test.comment); 190*9a0e4156SSadaf Ebrahimi System.out.println("Code: " + stringToHexUc(test.code)); 191*9a0e4156SSadaf Ebrahimi System.out.println("Disasm:"); 192*9a0e4156SSadaf Ebrahimi 193*9a0e4156SSadaf Ebrahimi cs = new Capstone(test.arch, test.mode); 194*9a0e4156SSadaf Ebrahimi cs.setDetail(Capstone.CS_OPT_ON); 195*9a0e4156SSadaf Ebrahimi Capstone.CsInsn[] all_ins = cs.disasm(test.code, 0x1000); 196*9a0e4156SSadaf Ebrahimi 197*9a0e4156SSadaf Ebrahimi for (int j = 0; j < all_ins.length; j++) { 198*9a0e4156SSadaf Ebrahimi print_ins_detail(all_ins[j]); 199*9a0e4156SSadaf Ebrahimi System.out.println(); 200*9a0e4156SSadaf Ebrahimi } 201*9a0e4156SSadaf Ebrahimi 202*9a0e4156SSadaf Ebrahimi // Close when done 203*9a0e4156SSadaf Ebrahimi cs.close(); 204*9a0e4156SSadaf Ebrahimi } 205*9a0e4156SSadaf Ebrahimi } 206*9a0e4156SSadaf Ebrahimi 207*9a0e4156SSadaf Ebrahimi } 208