static int update_switch_op (ut64 addr, const ut8 * bytes) { int sz = 4; int ccase = SWITCH_OP.cur_val + SWITCH_OP.min_val; SWITCH_OP.cur_val++; if ( ccase+1 > SWITCH_OP.max_val) IN_SWITCH_OP = 0; IFDBG eprintf ("Addr approach: 0x%04"PFMT64x" and BYTES_CONSUMED approach: 0x%04"PFMT64x"\n", addr, BYTES_CONSUMED); return update_bytes_consumed(sz); }
static int handle_switch_op (ut64 addr, const ut8 * bytes, char *output, int outlen ) { int sz = 4; ut32 jmp = (int)(UINT (bytes, 0)) + SWITCH_OP.addr; int ccase = SWITCH_OP.cur_val + SWITCH_OP.min_val; snprintf(output, outlen, "case %d: goto 0x%04x", ccase, jmp); update_switch_op (addr, bytes); return update_bytes_consumed(sz); }
static int handle_switch_op (ut64 addr, const ut8 * bytes, char *output, int outlen ) { int sz = 4; ut32 jmp = (ut32)(UINT (bytes, 0)) + SWITCH_OP.addr; ut32 ccase = SWITCH_OP.cur_val + SWITCH_OP.min_val; snprintf(output, outlen, "case %d: goto 0x%04x", ccase, jmp); SWITCH_OP.cur_val++; if ( ccase+1 > SWITCH_OP.max_val) IN_SWITCH_OP = 0; return update_bytes_consumed(sz); }
static int enter_switch_op (ut64 addr, const ut8 * bytes, char *output, int outlen ) { ut8 idx = bytes[0]; IN_SWITCH_OP = 1; ut8 sz = (BYTES_CONSUMED+1) % 4 ? 1 + 4 - (BYTES_CONSUMED+1) % 4: 1; // + (BYTES_CONSUMED+1) % 4; ut8 sz2 = (4 - (addr+1) % 4) + (addr+1) % 4; IFDBG eprintf ("Addr approach: 0x%04x and BYTES_CONSUMED approach: 0x%04x\n", sz2, sz); init_switch_op (); IN_SWITCH_OP = 1; SWITCH_OP.addr = addr; SWITCH_OP.def_jmp = (ut32)(UINT (bytes, sz)); SWITCH_OP.min_val = (ut32)(UINT (bytes, sz + 4)); SWITCH_OP.max_val = (ut32)(UINT (bytes, sz + 8)); sz += 12; snprintf (output, outlen, "%s default: 0x%04"PFMT64x, JAVA_OPS[idx].name, SWITCH_OP.def_jmp+SWITCH_OP.addr); return update_bytes_consumed(sz); }
int java_print_opcode(RBinJavaObj *obj, ut64 addr, int idx, const ut8 *bytes, char *output, int outlen) { char *arg = NULL; //(char *) malloc (1024); ut32 val_one = 0, val_two = 0; ut8 op_byte = JAVA_OPS[idx].byte; if (IN_SWITCH_OP) { return handle_switch_op (addr, bytes, output, outlen ); } IFDBG { eprintf ("Handling the following opcode %s expects: %d bytes\n", JAVA_OPS[idx].name, JAVA_OPS[idx].size); } switch (op_byte) { case 0x10: // "bipush" snprintf (output, outlen, "%s %d", JAVA_OPS[idx].name, (char) bytes[1]); output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); case 0x11: snprintf (output, outlen, "%s %d", JAVA_OPS[idx].name, (int)USHORT (bytes, 1)); output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); case 0x15: // "iload" case 0x16: // "lload" case 0x17: // "fload" case 0x18: // "dload" case 0x19: // "aload" case 0x37: // "lstore" case 0x38: // "fstore" case 0x39: // "dstore" case 0x3a: // "astore" case 0xbc: // "newarray" case 0xa9: // ret <var-num> snprintf (output, outlen, "%s %d", JAVA_OPS[idx].name, bytes[1]); output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); case 0x12: // ldc arg = r_bin_java_resolve_without_space (obj, (ut16)bytes[1]); if (arg) { snprintf (output, outlen, "%s %s", JAVA_OPS[idx].name, arg); free (arg); }else { snprintf (output, outlen, "%s #%d", JAVA_OPS[idx].name, USHORT (bytes, 1)); } output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); case 0x13: case 0x14: arg = r_bin_java_resolve_without_space (obj, (int)USHORT (bytes, 1)); if (arg) { snprintf (output, outlen, "%s %s", JAVA_OPS[idx].name, arg); free (arg); }else { snprintf (output, outlen, "%s #%d", JAVA_OPS[idx].name, USHORT (bytes, 1)); } output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); case 0x84: // iinc val_one = (ut32)bytes[1]; val_two = (ut32) bytes[2]; snprintf (output, outlen, "%s %d %d", JAVA_OPS[idx].name, val_one, val_two); output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); case 0x99: // ifeq case 0x9a: // ifne case 0x9b: // iflt case 0x9c: // ifge case 0x9d: // ifgt case 0x9e: // ifle case 0x9f: // if_icmpeq case 0xa0: // if_icmpne case 0xa1: // if_icmplt case 0xa2: // if_icmpge case 0xa3: // if_icmpgt case 0xa4: // if_icmple case 0xa5: // if_acmpne case 0xa6: // if_acmpne case 0xa7: // goto case 0xa8: // jsr snprintf (output, outlen, "%s 0x%04"PFMT64x, JAVA_OPS[idx].name, addr+(int)(short)USHORT (bytes, 1)); output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); // XXX - Figure out what constitutes the [<high>] value case 0xab: // tableswitch case 0xaa: // tableswitch return enter_switch_op (addr, bytes, output, outlen); case 0xb6: // invokevirtual case 0xb7: // invokespecial case 0xb8: // invokestatic case 0xb9: // invokeinterface case 0xba: // invokedynamic arg = r_bin_java_resolve_without_space (obj, (int)USHORT (bytes, 1)); if (arg) { snprintf (output, outlen, "%s %s", JAVA_OPS[idx].name, arg); free (arg); }else { snprintf (output, outlen, "%s #%d", JAVA_OPS[idx].name, USHORT (bytes, 1) ); } output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); case 0xb2: // getstatic case 0xb3: // putstatic case 0xb4: // getfield case 0xb5: // putfield case 0xbb: // new case 0xbd: // anewarray case 0xc0: // checkcast case 0xc1: // instance of arg = r_bin_java_resolve_with_space (obj, (int)USHORT (bytes, 1)); if (arg) { snprintf (output, outlen, "%s %s", JAVA_OPS[idx].name, arg); free (arg); }else { snprintf (output, outlen, "%s #%d", JAVA_OPS[idx].name, USHORT (bytes, 1) ); } output[outlen-1] = 0; return update_bytes_consumed (JAVA_OPS[idx].size); } /* process arguments */ switch (JAVA_OPS[idx].size) { case 1: snprintf (output, outlen, "%s", JAVA_OPS[idx].name); break; case 2: snprintf (output, outlen, "%s %d", JAVA_OPS[idx].name, bytes[1]); break; case 3: snprintf (output, outlen, "%s 0x%04x 0x%04x", JAVA_OPS[idx].name, bytes[0], bytes[1]); break; case 5: snprintf (output, outlen, "%s %d", JAVA_OPS[idx].name, bytes[1]); break; } return update_bytes_consumed (JAVA_OPS[idx].size); }