mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-10 15:33:04 +00:00
221 lines
5.7 KiB
C
221 lines
5.7 KiB
C
/* radare2 - LGPL - Copyright 2015-2018 - oddcoder, thestr4ng3r */
|
|
|
|
#include <r_types.h>
|
|
|
|
#include "pic_pic18.h"
|
|
|
|
//PIC18CXXX instruction set
|
|
|
|
//instruction classification according to the argument types
|
|
#define NO_ARG 0
|
|
#define DAF_T 1
|
|
#define F32_T 2
|
|
#define BAF_T 3
|
|
#define K_T 4
|
|
#define N_T 5
|
|
#define CALL_T 6
|
|
#define NEX_T 7
|
|
#define AF_T 8
|
|
#define GOTO_T 9
|
|
#define SHK_T 10
|
|
#define S_T 11
|
|
#define LFSR_T 12
|
|
|
|
static char *fsr[] = {"fsr0", "fsr1", "fsr2", "reserved"};
|
|
|
|
static struct {
|
|
ut16 opmin;
|
|
ut16 opmax;
|
|
char *name;
|
|
ut8 optype;
|
|
//and some magical hocus pocus ;)
|
|
} ops[] = {
|
|
{0xf000, 0xffff, "nop", NO_ARG},
|
|
{0xef00, 0xefff, "goto", GOTO_T},
|
|
{0xee00, 0xee3f, "lfsr", LFSR_T},
|
|
{0xec00, 0xedff, "call", CALL_T},
|
|
{0xe700, 0xe7ff, "bnn", N_T},
|
|
{0xe600, 0xe6ff, "bn", N_T},
|
|
{0xe500, 0xe5ff, "bnov", N_T},
|
|
{0xe400, 0xe4ff, "bov", N_T},
|
|
{0xe300, 0xe3ff, "bnc", N_T},
|
|
{0xe200, 0xe2ff, "bc", N_T},
|
|
{0xe100, 0xe1ff, "bnz", N_T},
|
|
{0xe000, 0xe0ff, "bz", N_T},
|
|
{0xd800, 0xdfff, "rcall", NEX_T},
|
|
{0xd000, 0xd7ff, "bra", NEX_T},
|
|
{0xc000, 0xcfff, "movff", F32_T},
|
|
{0xb000, 0xbfff, "btfsc", BAF_T},
|
|
{0xa000, 0xafff, "btfss", BAF_T},
|
|
{0x9000, 0x9fff, "bcf", BAF_T},
|
|
{0x8000, 0x8fff, "bsf", BAF_T},
|
|
{0x7000, 0x7fff, "btg", BAF_T},
|
|
{0x6e00, 0x6fff, "movwf", AF_T},
|
|
{0x6c00, 0x6dff, "negf", AF_T},
|
|
{0x6a00, 0x6bff, "clrf", AF_T},
|
|
{0x6800, 0x69ff, "setf", AF_T},
|
|
{0x6600, 0x67ff, "tstfsz", AF_T},
|
|
{0x6400, 0x65ff, "cpfsgt", AF_T},
|
|
{0x6200, 0x63ff, "cpfseq", AF_T},
|
|
{0x6000, 0x61ff, "cpfslt", AF_T},
|
|
{0x5c00, 0x5fff, "subwf", DAF_T},
|
|
{0x5800, 0x5bff, "subwfb", DAF_T},
|
|
{0x5400, 0x57ff, "subfwb", DAF_T},
|
|
{0x5000, 0x53ff, "movf", DAF_T},
|
|
{0x4c00, 0x4fff, "dcfsnz", DAF_T},
|
|
{0x4800, 0x4bff, "infsnz", DAF_T},
|
|
{0x4400, 0x47ff, "rlncf", DAF_T},
|
|
{0x4000, 0x43ff, "rrncf", DAF_T},
|
|
{0x3c00, 0x3fff, "incfsz", DAF_T},
|
|
{0x3800, 0x3bff, "swapf", DAF_T},
|
|
{0x3400, 0x37ff, "rlcf", DAF_T},
|
|
{0x3000, 0x33ff, "rrcf", DAF_T},
|
|
{0x2c00, 0x2fff, "decfsz", DAF_T},
|
|
{0x2800, 0x2bff, "incf", DAF_T},
|
|
{0x2400, 0x27ff, "addwf", DAF_T},
|
|
{0x2000, 0x23ff, "addwfc", DAF_T},
|
|
{0x1c00, 0x1fff, "comf", DAF_T},
|
|
{0x1800, 0x1bff, "xorwf", DAF_T},
|
|
{0x1400, 0x17ff, "andwf", DAF_T},
|
|
{0x1000, 0x13ff, "iorwf", DAF_T},
|
|
{0xf00, 0xfff, "addlw", K_T},
|
|
{0xe00, 0xeff, "movlw", K_T},
|
|
{0xd00, 0xdff, "mullw", K_T},
|
|
{0xc00, 0xcff, "retlw", K_T},
|
|
{0xb00, 0xbff, "andlw", K_T},
|
|
{0xa00, 0xaff, "xorlw", K_T},
|
|
{0x900, 0x9ff, "iorlw", K_T},
|
|
{0x800, 0x8ff, "sublw", K_T},
|
|
{0x400, 0x7ff, "decf", DAF_T},
|
|
{0x200, 0x3ff, "mulwf", AF_T},
|
|
{0x100, 0x10f, "movlb", SHK_T},
|
|
{0xff, 0xff, "reset", NO_ARG},
|
|
{0x12, 0x13, "return", S_T},
|
|
{0x10, 0x11, "retfie", S_T},
|
|
{0xf, 0xf, "tblwt+*", NO_ARG},
|
|
{0xe, 0xe, "tblwt*-", NO_ARG},
|
|
{0xd, 0xd, "tblwt*+", NO_ARG},
|
|
{0xc, 0xc, "tblwt*", NO_ARG},
|
|
{0xb, 0xb, "tblrd+*", NO_ARG},
|
|
{0xa, 0xa, "tblrd*-", NO_ARG},
|
|
{0x9, 0x9, "tblrd*+", NO_ARG},
|
|
{0x8, 0x8, "tblrd*", NO_ARG},
|
|
{0x7, 0x7, "daw", NO_ARG},
|
|
{0x6, 0x6, "pop", NO_ARG},
|
|
{0x5, 0x5, "push", NO_ARG},
|
|
{0x4, 0x4, "clrwdt", NO_ARG},
|
|
{0x3, 0x3, "sleep", NO_ARG},
|
|
{0x0, 0x0, "nop", NO_ARG},
|
|
{0x0, 0xffff, "invalid", NO_ARG},
|
|
};
|
|
|
|
int pic_pic18_disassemble(RAsmOp *op, char *opbuf, const ut8 *b, int blen) {
|
|
int i;
|
|
if (blen < 2) { //well noone loves reading bitstream of size zero or 1 !!
|
|
strcpy (opbuf, "invalid");
|
|
op->size = blen;
|
|
return -1;
|
|
}
|
|
ut16 instr = 0 ; // instruction
|
|
memcpy (&instr, b, sizeof (instr));
|
|
// if still redundan code is reported think of this of instr=0x2
|
|
const char *buf_asm = "invalid";
|
|
strcpy (opbuf, buf_asm);
|
|
|
|
for (i = 0; ops[i].opmin != (ops[i].opmin & instr) || ops[i].opmax != (ops[i].opmax | instr); i++) {
|
|
;
|
|
}
|
|
if (ops[i].opmin == 0 && ops[i].opmax == 0xffff) {
|
|
strcpy (opbuf, ops[i].name);
|
|
op->size = 2;
|
|
return -1;
|
|
}
|
|
op->size = 2;
|
|
switch (ops[i].optype) {
|
|
case NO_ARG:
|
|
buf_asm = ops[i].name;
|
|
break;
|
|
case N_T:
|
|
case K_T:
|
|
buf_asm = sdb_fmt ("%s 0x%x", ops[i].name, instr & 0xff);
|
|
break;
|
|
case DAF_T:
|
|
buf_asm = sdb_fmt ("%s 0x%x, %d, %d", ops[i].name, instr & 0xff, (instr >> 9) & 1, (instr >> 8) & 1);
|
|
break;
|
|
case AF_T:
|
|
buf_asm = sdb_fmt ("%s 0x%x, %d", ops[i].name, instr & 0xff, (instr >> 8) & 1);
|
|
break;
|
|
case BAF_T:
|
|
buf_asm = sdb_fmt ("%s 0x%x, %d, %d", ops[i].name, instr & 0xff, (instr >> 9) & 0x7, (instr >> 8) & 0x1);
|
|
break;
|
|
case NEX_T:
|
|
buf_asm = sdb_fmt ("%s 0x%x", ops[i].name, instr & 0x7ff);
|
|
break;
|
|
case CALL_T:
|
|
if (blen < 4) {
|
|
return -1;
|
|
}
|
|
op->size = 4;
|
|
{
|
|
ut32 dword_instr = *(ut32 *)b;
|
|
//I dont even know how the bits are arranged but it works !!!
|
|
//`the wierdness of little endianess`
|
|
if (dword_instr >> 28 != 0xf) {
|
|
return -1;
|
|
}
|
|
buf_asm = sdb_fmt ("%s 0x%x, %d", ops[i].name,
|
|
(dword_instr & 0xff) | (dword_instr >> 8 & 0xfff00), (dword_instr >> 8) & 0x1);
|
|
}
|
|
break;
|
|
case GOTO_T:
|
|
if (blen < 4) {
|
|
return -1;
|
|
}
|
|
{
|
|
op->size = 4;
|
|
ut32 dword_instr = *(ut32 *)b;
|
|
if (dword_instr >> 28 != 0xf) {
|
|
return -1;
|
|
}
|
|
buf_asm = sdb_fmt ("%s 0x%x", ops[i].name,
|
|
((dword_instr & 0xff) | ((dword_instr & 0xfff0000) >>8) )*2);
|
|
}
|
|
break;
|
|
case F32_T:
|
|
if (blen < 4) {
|
|
return -1;
|
|
}
|
|
op->size = 4;
|
|
{
|
|
ut32 dword_instr = *(ut32 *)b;
|
|
if (dword_instr >> 28 != 0xf) {
|
|
return -1;
|
|
}
|
|
buf_asm = sdb_fmt ("%s 0x%x, 0x%x", ops[i].name,
|
|
dword_instr & 0xfff, (dword_instr >> 16) & 0xfff);
|
|
}
|
|
break;
|
|
case SHK_T:
|
|
buf_asm = sdb_fmt ("%s 0x%x", ops[i].name, instr & 0xf);
|
|
break;
|
|
case S_T:
|
|
buf_asm = sdb_fmt ("%s %d", ops[i].name, instr & 0x1);
|
|
break;
|
|
case LFSR_T: {
|
|
op->size = 4;
|
|
ut32 dword_instr = *(ut32 *)b;
|
|
if (dword_instr >> 28 != 0xf) {
|
|
return -1;
|
|
}
|
|
ut8 reg_n = (dword_instr >> 4) & 0x3;
|
|
buf_asm = sdb_fmt ("%s %s, %d", ops[i].name, fsr[reg_n],
|
|
(dword_instr & 0xf) << 8 | ((dword_instr >> 16) & 0xff));
|
|
break;
|
|
}
|
|
default:
|
|
buf_asm = "unknown args";
|
|
};
|
|
strcpy (opbuf, buf_asm);
|
|
return op->size;
|
|
}
|