2013-12-13 19:52:32 +08:00
|
|
|
/* NDS32-specific support for 32-bit ELF.
|
|
|
|
Copyright (C) 2012-2013 Free Software Foundation, Inc.
|
|
|
|
Contributed by Andes Technology Corporation.
|
|
|
|
|
|
|
|
This file is part of BFD, the Binary File Descriptor library.
|
|
|
|
|
|
|
|
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, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA
|
|
|
|
02110-1301, USA.*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "sysdep.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "ansidecl.h"
|
|
|
|
#include "dis-asm.h"
|
|
|
|
#include "bfd.h"
|
|
|
|
#include "symcat.h"
|
|
|
|
#include "libiberty.h"
|
|
|
|
#include "opintl.h"
|
|
|
|
#include "bfd_stdint.h"
|
|
|
|
|
|
|
|
#define __MF(v, off, bs) ((v & ((1 << (bs)) - 1)) << (off))
|
|
|
|
#define __GF(v, off, bs) ((v >> (off)) & ((1 << (bs)) - 1))
|
|
|
|
#define __PF(v, off, bs, val) do { v = __put_field (v, off, bs, val); } while (0)
|
|
|
|
/* #define __SEXT(v, bs) ((v ^ (1 << (bs - 1))) - (1 << (bs - 1))) */
|
|
|
|
#define __SEXT(v, bs) (((v & ((1 << bs) - 1)) ^ (1 << (bs - 1))) - (1 << (bs - 1)))
|
|
|
|
#define __BIT(n) (1 << n)
|
|
|
|
|
|
|
|
/* Get fields */
|
|
|
|
#define OP6(insn) ((insn >> 25) & 0x3F)
|
|
|
|
#define RT5(insn) ((insn >> 20) & 0x1F)
|
|
|
|
#define RA5(insn) ((insn >> 15) & 0x1F)
|
|
|
|
#define RB5(insn) ((insn >> 10) & 0x1F)
|
|
|
|
#define RD5(insn) ((insn >> 5) & 0x1F)
|
|
|
|
#define SUB5(insn) ((insn >> 0) & 0x1F)
|
|
|
|
#define SUB10(insn) ((insn >> 0) & 0x3FF)
|
|
|
|
#define IMMU(insn, bs) (insn & ((1 << bs) - 1))
|
|
|
|
#define IMMS(insn, bs) __SEXT ((insn & ((1 << bs) - 1)), bs)
|
|
|
|
#define IMM1U(insn) IMMU ((insn >> 10), 5)
|
|
|
|
#define IMM1S(insn) IMMS ((insn >> 10), 5)
|
|
|
|
#define IMM2U(insn) IMMU ((insn >> 5), 5)
|
|
|
|
#define IMM2S(insn) IMMS ((insn >> 5), 5)
|
|
|
|
|
|
|
|
/* Default text to print if an instruction isn't recognized. */
|
|
|
|
#define UNKNOWN_INSN_MSG _("*unknown*")
|
|
|
|
|
|
|
|
static const char *mnemonic_op6[] =
|
|
|
|
{
|
|
|
|
"lbi", "lhi", "lwi", "ldi", "lbi.bi", "lhi.bi", "lwi.bi", "ldi.bi",
|
|
|
|
"sbi", "shi", "swi", "sdi", "sbi.bi", "shi.bi", "swi.bi", "sdi.bi",
|
|
|
|
"lbsi", "lhsi", "lwsi", "dprefi", "lbsi.bi", "lhsi.bi", "lwsi.bi", "lbgp",
|
|
|
|
"lwc", "swc", "ldc", "sdc", "mem", "lsmw", "hwgp", "sbgp",
|
|
|
|
"alu1", "alu2", "movi", "sethi", "ji", "jreg", "br1", "br2",
|
|
|
|
"addi", "subri", "andi", "xori", "ori", "br3", "slti", "sltsi",
|
|
|
|
"aext", "cext", "misc", "bitci", "op_64", "cop"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_mem[] =
|
|
|
|
{
|
|
|
|
"lb", "lh", "lw", "ld", "lb.bi", "lh.bi", "lw.bi", "ld.bi",
|
|
|
|
"sb", "sh", "sw", "sd", "sb.bi", "sh.bi", "sw.bi", "sd.bi",
|
|
|
|
"lbs", "lhs", "lws", "dpref", "lbs.bi", "lhs.bi", "lws.bi", "27",
|
|
|
|
"llw", "scw", "32", "33", "34", "35", "36", "37",
|
|
|
|
"lbup", "41", "lwup", "43", "44", "45", "46", "47",
|
|
|
|
"sbup", "51", "swup"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_alu1[] =
|
|
|
|
{
|
|
|
|
"add", "sub", "and", "xor", "or", "nor", "slt", "slts",
|
|
|
|
"slli", "srli", "srai", "rotri", "sll", "srl", "sra", "rotr",
|
|
|
|
"seb", "seh", "bitc", "zeh", "wsbh", "or_srli", "divsr", "divr",
|
|
|
|
"sva", "svs", "cmovz", "cmovn", "add_srli", "sub_srli", "and_srli", "xor_srli"
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static const char *mnemonic_alu20[] =
|
|
|
|
{
|
|
|
|
"max", "min", "ave", "abs", "clips", "clip", "clo", "clz",
|
|
|
|
"bset", "bclr", "btgl", "btst", "bse", "bsp", "ffb", "ffmism",
|
|
|
|
"add.sc", "sub.sc", "add.wc", "sub.wc", "24", "25", "26", "ffzmism",
|
|
|
|
"qadd", "qsub", "32", "33", "34", "35", "36", "37",
|
|
|
|
"mfusr", "mtusr", "42", "43", "mul", "45", "46", "47",
|
|
|
|
"mults64", "mult64", "madds64", "madd64", "msubs64", "msub64", "divs", "div",
|
|
|
|
"60", "mult32", "62", "madd32", "64", "msub32", "65", "66",
|
|
|
|
"dmadd", "dmaddc", "dmsub", "dmsubc", "rmfhi", "qmflo"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_alu21[] =
|
|
|
|
{
|
|
|
|
"00", "01", "02", "03", "04", "05", "06", "07",
|
|
|
|
"10", "11", "12", "13", "14", "15", "ffbi", "flmism",
|
|
|
|
"20", "21", "22", "23", "24", "25", "26", "27",
|
|
|
|
"30", "31", "32", "33", "34", "35", "36", "37",
|
|
|
|
"40", "41", "42", "43", "44", "45", "46", "47",
|
|
|
|
"mulsr64", "mulr64", "52", "53", "54", "55", "56", "57",
|
|
|
|
"60", "61", "62", "maddr32", "64", "msubr32", "66", "67",
|
|
|
|
"70", "71", "72", "73", "74", "75", "76", "77"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_br2[] =
|
|
|
|
{
|
|
|
|
"ifcall", "01", "beqz", "bnez", "bgez", "bltz", "bgtz", "blez",
|
|
|
|
"10", "11", "12", "13", "bgezal", "bltzal"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_misc[] =
|
|
|
|
{
|
|
|
|
"standby", "cctl", "mfsr", "mtsr", "iret", "trap", "teqz", "tnez",
|
|
|
|
"dsb", "isb", "break", "syscall", "msync", "isync", "tlbop"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_hwgp[] =
|
|
|
|
{
|
|
|
|
"lhi.gp", "lhi.gp", "lhsi.gp", "lhsi.gp",
|
|
|
|
"shi.gp", "shi.gp", "lwi.gp", "swi.gp"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *keyword_dpref[] =
|
|
|
|
{
|
|
|
|
"SRD", "MRD", "SWR", "MWR", "PTE", "CLWR", "6", "7",
|
|
|
|
"8", "9", "10", "11", "12", "13", "14", "15"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_alu[] =
|
|
|
|
{
|
|
|
|
"fadds", "fsubs", "fcpynss", "fcpyss", "fmadds",
|
|
|
|
"fmsubs", "fcmovns", "fcmovzs", "fnmadds", "fnmsubs",
|
|
|
|
"10", "11", "fmuls", "fdivs", "faddd",
|
|
|
|
"fsubd", "fcpynsd", "fcpysd", "fmaddd", "fmsubd",
|
|
|
|
"fcmovnd", "fcmovzd", "fnmaddd", "fnmsubd", "24",
|
|
|
|
"25", "fmuld", "fdivd"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_fpu_2op[] =
|
|
|
|
{
|
|
|
|
"fs2d", "fsqrts", "2", "3", "4", "fabss", "6", "7",
|
|
|
|
"fui2s", "9", "10", "11", "fsi2s", "13", "14", "15",
|
|
|
|
"fs2ui", "17", "18", "19", "fs2ui.z", "21", "22", "23",
|
|
|
|
"fs2si", "25", "26", "27", "fs2si.z", "fd2s", "fsqrtd", "31",
|
|
|
|
"32", "33", "fabsd", "35", "36", "fui2d", "38", "39",
|
|
|
|
"40", "fsi2d", "42", "43", "44", "fd2ui", "46", "47",
|
|
|
|
"48", "fd2ui.z", "50", "51", "52", "fd2si", "54", "55",
|
|
|
|
"56", "fd2si.z"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_fs2_cmp[] =
|
|
|
|
{
|
|
|
|
"fcmpeqs", "fcmpeqs.e", "fcmplts", "fcmplts.e",
|
|
|
|
"fcmples", "fcmples.e", "fcmpuns", "fcmpuns.e"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_fd2_cmp[] =
|
|
|
|
{
|
|
|
|
"fcmpeqd", "fcmpeqd.e", "fcmpltd", "fcmpltd.e",
|
|
|
|
"fcmpled", "fcmpled.e", "fcmpund", "fcmpund.e"
|
|
|
|
};
|
|
|
|
|
2013-12-12 13:43:51 +08:00
|
|
|
/* Register name table. */
|
|
|
|
/* General purpose register. */
|
|
|
|
|
2013-12-13 19:52:32 +08:00
|
|
|
static const char *gpr_map[] =
|
|
|
|
{
|
|
|
|
"$r0", "$r1", "$r2", "$r3", "$r4", "$r5", "$r6", "$r7",
|
|
|
|
"$r8", "$r9", "$r10", "$r11", "$r12", "$r13", "$r14", "$r15",
|
|
|
|
"$r16", "$r17", "$r18", "$r19", "$r20", "$r21", "$r22", "$r23",
|
|
|
|
"$r24", "$r25", "$r26", "$r27", "$fp", "$gp", "$lp", "$sp"
|
|
|
|
};
|
|
|
|
|
2013-12-12 13:43:51 +08:00
|
|
|
/* User special register. */
|
|
|
|
|
2013-12-13 19:52:32 +08:00
|
|
|
static const char *usr_map[][32] =
|
|
|
|
{
|
|
|
|
{
|
2013-12-12 13:43:51 +08:00
|
|
|
"d0.lo", "d0.hi", "d1.lo", "d1.hi", "4", "5", "6", "7",
|
2013-12-13 19:52:32 +08:00
|
|
|
"8", "9", "10", "11", "12", "13", "14", "15",
|
|
|
|
"16", "17", "18", "19", "20", "21", "22", "23",
|
|
|
|
"24", "25", "26", "27", "28", "29", "30", "pc"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"DMA_CFG", "DMA_GCSW", "DMA_CHNSEL", "DMA_ACT", "DMA_SETUP",
|
|
|
|
"DMA_ISADDR", "DMA_ESADDR", "DMA_TCNT", "DMA_STATUS", "DMA_2DSET",
|
|
|
|
"10", "11", "12", "13", "14",
|
2013-12-12 13:43:51 +08:00
|
|
|
"15", "16", "17", "18", "19",
|
|
|
|
"20", "21", "22", "23", "24",
|
2013-12-13 19:52:32 +08:00
|
|
|
"DMA_2DSCTL"
|
|
|
|
},
|
|
|
|
{
|
2013-12-12 13:43:51 +08:00
|
|
|
"PFMC0", "PFMC1", "PFMC2", "3", "PFM_CTL"
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/* System register. */
|
|
|
|
/* Major Minor Extension. */
|
|
|
|
static const char *sr_map[8][16][8] =
|
|
|
|
{
|
|
|
|
{
|
|
|
|
{"CPU_VER", "CORE_ID"},
|
|
|
|
{"ICM_CFG"},
|
|
|
|
{"DCM_CFG"},
|
|
|
|
{"MMU_CFG"},
|
|
|
|
{"MSC_CFG"}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{"PSW", "IPSW", "P_IPSW"},
|
|
|
|
{"0", "IVB", "INT_CTRL"},
|
|
|
|
{"0", "EVA", "P_EVA"},
|
|
|
|
{"0", "ITYPE", "P_ITYPE"},
|
|
|
|
{"0", "MERR"},
|
|
|
|
{"0", "IPC", "P_IPC", "OIPC"},
|
|
|
|
{"0", "1", "P_P0"},
|
|
|
|
{"0", "1", "P_P1"},
|
|
|
|
{"INT_MASK", "INT_MASK2"},
|
|
|
|
{"INT_PEND", "INT_PEND2", "2", "3", "INT_TRIGGER"},
|
|
|
|
{"SP_USR", "SP_PRIV"},
|
|
|
|
{"INT_PRI", "INT_PRI2"}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{"MMU_CTL"},
|
|
|
|
{"L1_PPTB"},
|
|
|
|
{"TLB_VPN"},
|
|
|
|
{"TLB_DATA"},
|
|
|
|
{"TLB_MISC"},
|
|
|
|
{"VLPT_IDX"},
|
|
|
|
{"ILMB"},
|
|
|
|
{"DLMB"},
|
|
|
|
{"CACHE_CTL"},
|
|
|
|
{"HSMP_SADDR", "HSMP_EADDR"},
|
|
|
|
{"0"},
|
|
|
|
{"0"},
|
|
|
|
{"0"},
|
|
|
|
{"0"},
|
|
|
|
{"0"},
|
|
|
|
{"SDZ_CTL", "MISC_CTL"}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{"BPC0", "BPC1", "BPC2", "BPC3", "BPC4", "BPC5", "BPC6", "BPC7"},
|
|
|
|
{"BPA0", "BPA1", "BPA2", "BPA3", "BPA4", "BPA5", "BPA6", "BPA7"},
|
|
|
|
{"BPAM0", "BPAM1", "BPAM2", "BPAM3", "BPAM4", "BPAM5", "BPAM6", "BPAM7"},
|
|
|
|
{"BPV0", "BPV1", "BPV2", "BPV3", "BPV4", "BPV5", "BPV6", "BPV7"},
|
|
|
|
{"BPCID0", "BPCID1", "BPCID2", "BPCID3", "BPCID4", "BPCID5", "BPCID6", "BPCID7"},
|
|
|
|
{"EDM_CFG"},
|
|
|
|
{"EDMSW"},
|
|
|
|
{"EDM_CTL"},
|
|
|
|
{"EDM_DTR"},
|
|
|
|
{"BPMTC"},
|
|
|
|
{"DIMBR"},
|
|
|
|
{"EDM_PROBE"},
|
|
|
|
{"0"},
|
|
|
|
{"0"},
|
|
|
|
{"TECR0", "TECR1"}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{"PFMC0", "PFMC1", "PFMC2"},
|
|
|
|
{"PFM_CTL"},
|
|
|
|
{"0"},
|
|
|
|
{"0"},
|
|
|
|
{"PRUSR_ACC_CTL"},
|
|
|
|
{"FUCOP_CTL"}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{"DMA_CFG"},
|
|
|
|
{"DMA_GCSW"},
|
|
|
|
{"DMA_CHNSEL"},
|
|
|
|
{"DMA_ACT"},
|
|
|
|
{"DMA_SETUP"},
|
|
|
|
{"DMA_ISADDR"},
|
|
|
|
{"DMA_ESADDR"},
|
|
|
|
{"DMA_TCNT"},
|
|
|
|
{"DMA_STATUS"},
|
|
|
|
{"DMA_2DSET", "DMA_2DSCTL"}
|
2013-12-13 19:52:32 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_insn16 (bfd_vma pc, disassemble_info *info, uint32_t insn)
|
|
|
|
{
|
|
|
|
static char r4map[] =
|
|
|
|
{
|
|
|
|
0, 1, 2, 3, 4, 5, 6, 7,
|
|
|
|
8, 9, 10, 11, 16, 17, 18, 19
|
|
|
|
};
|
|
|
|
const int rt5 = __GF (insn, 5, 5);
|
|
|
|
const int ra5 = __GF (insn, 0, 5);
|
|
|
|
const int rt4 = r4map[__GF (insn, 5, 4)];
|
|
|
|
const int imm5u = IMMU (insn, 5);
|
|
|
|
const int imm9u = IMMU (insn, 9);
|
|
|
|
const int rt3 = __GF (insn, 6, 3);
|
|
|
|
const int ra3 = __GF (insn, 3, 3);
|
|
|
|
const int rb3 = __GF (insn, 0, 3);
|
|
|
|
const int rt38 = __GF (insn, 8, 3);
|
|
|
|
const int imm3u = rb3;
|
|
|
|
fprintf_ftype func = info->fprintf_func;
|
|
|
|
void *stream = info->stream;
|
|
|
|
|
|
|
|
static const char *mnemonic_96[] =
|
|
|
|
{
|
|
|
|
"0x1", "0x1", "0x2", "0x3",
|
|
|
|
"add45", "sub45", "addi45", "subi45",
|
|
|
|
"srai45", "srli45", "slli333", "0xb",
|
|
|
|
"add333", "sub333", "addi333", "subi333",
|
|
|
|
"lwi333", "lwi333.bi", "lhi333", "lbi333",
|
|
|
|
"swi333", "swi333.bi", "shi333", "sbi333",
|
|
|
|
"addri36.sp", "lwi45.fe", "lwi450", "swi450",
|
|
|
|
"0x1c", "0x1d", "0x1e", "0x1f",
|
|
|
|
"0x20", "0x21", "0x22", "0x23",
|
2013-12-14 00:43:59 +08:00
|
|
|
"0x24", "0x25", "0x26", "0x27",
|
2013-12-13 19:52:32 +08:00
|
|
|
"0x28", "0x29", "0x2a", "0x2b",
|
2013-12-14 00:43:59 +08:00
|
|
|
"0x2c", "0x2d", "0x2e", "0x2f",
|
2013-12-13 19:52:32 +08:00
|
|
|
"slts45", "slt45", "sltsi45", "slti45",
|
|
|
|
"0x34", "0x35", "0x36", "0x37",
|
|
|
|
"0x38", "0x39", "0x3a", "0x3b",
|
|
|
|
"ifcall9", "movpi45"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_misc33[] =
|
|
|
|
{
|
|
|
|
"misc33_0", "misc33_1", "neg33", "not33", "mul33", "xor33", "and33", "or33",
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_0xb[] =
|
|
|
|
{
|
|
|
|
"zeb33", "zeh33", "seb33", "seh33", "xlsb33", "x11b33", "bmski33", "fexti33"
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *mnemonic_bnes38[] =
|
|
|
|
{
|
|
|
|
"jr5", "jral5", "ex9.it", "?", "ret5", "add5.pc"
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (__GF (insn, 7, 8))
|
|
|
|
{
|
|
|
|
case 0xf8: /* push25 */
|
|
|
|
case 0xf9: /* pop25 */
|
|
|
|
{
|
|
|
|
uint32_t res[] = { 6, 8, 10, 14 };
|
|
|
|
uint32_t re = res[__GF (insn, 5, 2)];
|
|
|
|
|
|
|
|
func (stream, "%s\t%s, %d", (insn & __BIT (7)) ? "pop25" : "push25",
|
|
|
|
gpr_map[re], imm5u << 3);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (__GF (insn, 8, 7) == 0x7d) /* movd44 */
|
|
|
|
{
|
|
|
|
int rt5e = __GF (insn, 4, 4) << 1;
|
|
|
|
int ra5e = IMMU (insn, 4) << 1;
|
|
|
|
|
|
|
|
func (stream, "movd44\t%s, %d", gpr_map[rt5e], ra5e);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (__GF (insn, 9, 6))
|
|
|
|
{
|
|
|
|
case 0x4: /* add45 */
|
|
|
|
case 0x5: /* sub45 */
|
|
|
|
case 0x30: /* slts45 */
|
|
|
|
case 0x31: /* slt45 */
|
|
|
|
func (stream, "%s\t%s, %s", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt4], gpr_map[ra5]);
|
|
|
|
return;
|
|
|
|
case 0x6: /* addi45 */
|
|
|
|
case 0x7: /* subi45 */
|
|
|
|
case 0x8: /* srai45 */
|
|
|
|
case 0x9: /* srli45 */
|
|
|
|
case 0x32: /* sltsi45 */
|
|
|
|
case 0x33: /* slti45 */
|
|
|
|
func (stream, "%s\t%s, %d", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt4], ra5);
|
|
|
|
return;
|
|
|
|
case 0xc: /* add333 */
|
|
|
|
case 0xd: /* sub333 */
|
|
|
|
func (stream, "%s\t%s, %s, %s", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt3], gpr_map[ra3], gpr_map[rb3]);
|
|
|
|
return;
|
|
|
|
case 0xa: /* slli333 */
|
|
|
|
case 0xe: /* addi333 */
|
|
|
|
case 0xf: /* subi333 */
|
|
|
|
func (stream, "%s\t%s, %s, %d", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt3], gpr_map[ra3], imm3u);
|
|
|
|
return;
|
|
|
|
case 0x10: /* lwi333 */
|
|
|
|
case 0x14: /* swi333 */
|
|
|
|
func (stream, "%s\t%s, [%s + %d]", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt3], gpr_map[ra3], imm3u << 2);
|
|
|
|
return;
|
|
|
|
case 0x12: /* lhi333 */
|
|
|
|
case 0x16: /* shi333 */
|
|
|
|
func (stream, "%s\t%s, [%s + %d]", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt3], gpr_map[ra3], imm3u << 1);
|
|
|
|
return;
|
|
|
|
case 0x13: /* lbi333 */
|
|
|
|
case 0x17: /* sbi333 */
|
|
|
|
func (stream, "%s\t%s, [%s + %d]", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt3], gpr_map[ra3], imm3u);
|
|
|
|
return;
|
|
|
|
case 0x11: /* lwi333.bi */
|
|
|
|
case 0x15: /* swi333.bi */
|
|
|
|
func (stream, "%s\t%s, [%s], %d", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt3], gpr_map[ra3], imm3u << 2);
|
|
|
|
return;
|
|
|
|
case 0x18: /* addri36.sp */
|
|
|
|
func (stream, "%s\t%s, %d", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt3], IMMU (insn, 6) << 2);
|
|
|
|
return;
|
|
|
|
case 0x19: /* lwi45.fe */
|
|
|
|
func (stream, "%s\t%s, %d", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt4], -((32 - imm5u) << 2));
|
|
|
|
return;
|
|
|
|
case 0x1a: /* lwi450 */
|
|
|
|
case 0x1b: /* swi450 */
|
|
|
|
func (stream, "%s\t%s, [%s]", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt4], gpr_map[ra5]);
|
|
|
|
return;
|
|
|
|
case 0x34: /* beqzs8, bnezs8 */
|
|
|
|
func (stream, "%s\t", ((insn & __BIT (8)) ? "bnezs8" : "beqzs8"));
|
|
|
|
info->print_address_func ((IMMS (insn, 8) << 1) + pc, info);
|
|
|
|
return;
|
|
|
|
case 0x35: /* break16, ex9.it */
|
|
|
|
/* FIXME: Check bfd_mach. */
|
|
|
|
if (imm9u < 32) /* break16 */
|
|
|
|
func (stream, "break16\t%d", imm9u);
|
|
|
|
else
|
|
|
|
func (stream, "ex9.it\t%d", imm9u);
|
|
|
|
return;
|
|
|
|
case 0x3c: /* ifcall9 */
|
|
|
|
func (stream, "%s\t", mnemonic_96[__GF (insn, 9, 6)]);
|
|
|
|
info->print_address_func ((IMMU (insn, 9) << 1) + pc, info);
|
|
|
|
return;
|
|
|
|
case 0x3d: /* movpi45 */
|
|
|
|
func (stream, "%s\t%s, %d", mnemonic_96[__GF (insn, 9, 6)],
|
|
|
|
gpr_map[rt4], ra5 + 16);
|
|
|
|
return;
|
|
|
|
case 0x3f: /* MISC33 */
|
|
|
|
func (stream, "%s\t%s, %s", mnemonic_misc33[rb3],
|
|
|
|
gpr_map[rt3], gpr_map[ra3]);
|
|
|
|
return;
|
|
|
|
case 0xb: /* ... */
|
|
|
|
func (stream, "%s\t%s, %s", mnemonic_0xb[rb3],
|
|
|
|
gpr_map[rt3], gpr_map[ra3]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (__GF (insn, 10, 5))
|
|
|
|
{
|
|
|
|
case 0x0: /* mov55 or ifret16 */
|
|
|
|
/* FIXME: Check bfd_mach. */
|
|
|
|
if (rt5 == ra5 && rt5 == 31)
|
|
|
|
func (stream, "ifret16");
|
|
|
|
else
|
|
|
|
func (stream, "mov55\t%s, %s", gpr_map[rt5], gpr_map[ra5]);
|
|
|
|
return;
|
|
|
|
case 0x1: /* movi55 */
|
|
|
|
func (stream, "movi55\t%s, %d", gpr_map[rt5], IMMS (insn, 5));
|
|
|
|
return;
|
|
|
|
case 0x1b: /* addi10s (V2) */
|
|
|
|
func (stream, "addi10s\t%d", IMMS (insn, 10));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (__GF (insn, 11, 4))
|
|
|
|
{
|
|
|
|
case 0x7: /* lwi37.fp/swi37.fp */
|
|
|
|
func (stream, "%s\t%s, [$fp + 0x%x]",
|
|
|
|
((insn & __BIT (7)) ? "swi37" : "lwi37"),
|
|
|
|
gpr_map[rt38], IMMU (insn, 7) << 2);
|
|
|
|
return;
|
|
|
|
case 0x8: /* beqz38 */
|
|
|
|
case 0x9: /* bnez38 */
|
|
|
|
func (stream, "%s\t%s, ",
|
|
|
|
((__GF (insn, 11, 4) & 1) ? "bnez38" : "beqz38"), gpr_map[rt38]);
|
|
|
|
info->print_address_func ((IMMS (insn, 8) << 1) + pc, info);
|
|
|
|
return;
|
|
|
|
case 0xa: /* beqs38/j8, implied r5 */
|
|
|
|
if (rt38 == 5)
|
|
|
|
{
|
|
|
|
func (stream, "j8\t");
|
|
|
|
info->print_address_func ((IMMS (insn, 8) << 1) + pc, info);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
func (stream, "beqs38\t%s, ", gpr_map[rt38]);
|
|
|
|
info->print_address_func ((IMMS (insn, 8) << 1) + pc, info);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case 0xb: /* bnes38 and others */
|
|
|
|
if (rt38 == 5)
|
|
|
|
{
|
|
|
|
switch (__GF (insn, 5, 3))
|
|
|
|
{
|
|
|
|
case 0: /* jr5 */
|
|
|
|
case 1: /* jral5 */
|
|
|
|
case 4: /* ret5 */
|
|
|
|
func (stream, "%s\t%s", mnemonic_bnes38[__GF (insn, 5, 3)],
|
|
|
|
gpr_map[ra5]);
|
|
|
|
return;
|
|
|
|
case 2: /* ex9.it imm5 */
|
|
|
|
case 5: /* add5.pc */
|
|
|
|
func (stream, "%s\t%d", mnemonic_bnes38[__GF (insn, 5, 3)], ra5);
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
func (stream, UNKNOWN_INSN_MSG);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
func (stream, "bnes38\t%s", gpr_map[rt3]);
|
|
|
|
info->print_address_func ((IMMS (insn, 8) << 1) + pc, info);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case 0xe: /* lwi37/swi37 */
|
|
|
|
func (stream, "%s\t%s, [+ 0x%x]",
|
|
|
|
((insn & __BIT (7)) ? "swi37.sp" : "lwi37.sp"),
|
|
|
|
gpr_map[rt38], IMMU (insn, 7) << 2);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_insn32_mem (bfd_vma pc ATTRIBUTE_UNUSED, disassemble_info *info,
|
|
|
|
uint32_t insn)
|
|
|
|
{
|
|
|
|
const int rt = RT5 (insn);
|
|
|
|
const int ra = RA5 (insn);
|
|
|
|
const int rb = RB5 (insn);
|
|
|
|
const int sv = __GF (insn, 8, 2);
|
|
|
|
const int op = insn & 0xFF;
|
|
|
|
fprintf_ftype func = info->fprintf_func;
|
|
|
|
void *stream = info->stream;
|
|
|
|
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case 0x0: /* lb */
|
|
|
|
case 0x1: /* lh */
|
|
|
|
case 0x2: /* lw */
|
|
|
|
case 0x3: /* ld */
|
|
|
|
case 0x8: /* sb */
|
|
|
|
case 0x9: /* sh */
|
|
|
|
case 0xa: /* sw */
|
|
|
|
case 0xb: /* sd */
|
|
|
|
case 0x10: /* lbs */
|
|
|
|
case 0x11: /* lhs */
|
|
|
|
case 0x12: /* lws */
|
|
|
|
case 0x18: /* llw */
|
|
|
|
case 0x19: /* scw */
|
|
|
|
case 0x20: /* lbup */
|
|
|
|
case 0x22: /* lwup */
|
|
|
|
case 0x28: /* sbup */
|
|
|
|
case 0x2a: /* swup */
|
|
|
|
func (stream, "%s\t%s, [%s + (%s << %d)]",
|
|
|
|
mnemonic_mem[op], gpr_map[rt], gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
break;
|
|
|
|
case 0x4: /* lb.bi */
|
|
|
|
case 0x5: /* lh.bi */
|
|
|
|
case 0x6: /* lw.bi */
|
|
|
|
case 0x7: /* ld.bi */
|
|
|
|
case 0xc: /* sb.bi */
|
|
|
|
case 0xd: /* sh.bi */
|
|
|
|
case 0xe: /* sw.bi */
|
|
|
|
case 0xf: /* sd.bi */
|
|
|
|
case 0x14: /* lbs.bi */
|
|
|
|
case 0x15: /* lhs.bi */
|
|
|
|
case 0x16: /* lws.bi */
|
|
|
|
func (stream, "%s\t%s, [%s], (%s << %d)",
|
|
|
|
mnemonic_mem[op], gpr_map[rt], gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
break;
|
|
|
|
case 0x13: /* dpref */
|
|
|
|
{
|
|
|
|
const char *subtype = "???";
|
|
|
|
|
|
|
|
if ((rt & 0xf) < ARRAY_SIZE (keyword_dpref))
|
|
|
|
subtype = keyword_dpref[rt & 0xf];
|
|
|
|
|
|
|
|
func (stream, "%s\t%s, [%s + (%s << %d)]",
|
|
|
|
"dpref", subtype, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
func (stream, UNKNOWN_INSN_MSG);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_insn32_alu1 (bfd_vma pc ATTRIBUTE_UNUSED, disassemble_info *info, uint32_t insn)
|
|
|
|
{
|
|
|
|
int op = insn & 0x1f;
|
|
|
|
const int rt = RT5 (insn);
|
|
|
|
const int ra = RA5 (insn);
|
|
|
|
const int rb = RB5 (insn);
|
|
|
|
const int rd = RD5 (insn);
|
|
|
|
fprintf_ftype func = info->fprintf_func;
|
|
|
|
void *stream = info->stream;
|
|
|
|
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case 0x0: /* add, add_slli */
|
|
|
|
case 0x1: /* sub, sub_slli */
|
|
|
|
case 0x2: /* and, add_slli */
|
|
|
|
case 0x3: /* xor, xor_slli */
|
|
|
|
case 0x4: /* or, or_slli */
|
|
|
|
if (rd != 0)
|
|
|
|
{
|
|
|
|
func (stream, "%s_slli\t%s, %s, %s, #%d",
|
|
|
|
mnemonic_alu1[op], gpr_map[rt], gpr_map[ra], gpr_map[rb], rd);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
func (stream, "%s\t%s, %s, %s",
|
|
|
|
mnemonic_alu1[op], gpr_map[rt], gpr_map[ra], gpr_map[rb]);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case 0x1c: /* add_srli */
|
|
|
|
case 0x1d: /* sub_srli */
|
|
|
|
case 0x1e: /* and_srli */
|
|
|
|
case 0x1f: /* xor_srli */
|
|
|
|
case 0x15: /* or_srli */
|
|
|
|
func (stream, "%s\t%s, %s, %s, #%d",
|
|
|
|
mnemonic_alu1[op], gpr_map[rt], gpr_map[ra], gpr_map[rb], rd);
|
|
|
|
return;
|
|
|
|
case 0x5: /* nor */
|
|
|
|
case 0x6: /* slt */
|
|
|
|
case 0x7: /* slts */
|
|
|
|
case 0xc: /* sll */
|
|
|
|
case 0xd: /* srl */
|
|
|
|
case 0xe: /* sra */
|
|
|
|
case 0xf: /* rotr */
|
|
|
|
case 0x12: /* bitc */
|
|
|
|
case 0x18: /* sva */
|
|
|
|
case 0x19: /* svs */
|
|
|
|
case 0x1a: /* cmovz */
|
|
|
|
case 0x1b: /* cmovn */
|
|
|
|
func (stream, "%s\t%s, %s, %s",
|
|
|
|
mnemonic_alu1[op], gpr_map[rt], gpr_map[ra], gpr_map[rb]);
|
|
|
|
return;
|
|
|
|
case 0x9: /* srli */
|
|
|
|
if (ra ==0 && rb == 0 && rb==0)
|
|
|
|
{
|
|
|
|
func (stream, "nop");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case 0x8: /* slli */
|
|
|
|
case 0xa: /* srai */
|
|
|
|
case 0xb: /* rotri */
|
|
|
|
func (stream, "%s\t%s, %s, #%d",
|
|
|
|
mnemonic_alu1[op], gpr_map[rt], gpr_map[ra], rb);
|
|
|
|
return;
|
|
|
|
case 0x10: /* seb */
|
|
|
|
case 0x11: /* seh */
|
|
|
|
case 0x13: /* zeh */
|
|
|
|
case 0x14: /* wsbh */
|
|
|
|
func (stream, "%s\t%s, %s",
|
|
|
|
mnemonic_alu1[op], gpr_map[rt], gpr_map[ra]);
|
|
|
|
return;
|
|
|
|
case 0x16: /* divsr */
|
|
|
|
case 0x17: /* divr */
|
|
|
|
func (stream, "%s\t%s, %s, %s, %s",
|
|
|
|
mnemonic_alu1[op], gpr_map[rt], gpr_map[rd], gpr_map[ra], gpr_map[rb]);
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
func (stream, UNKNOWN_INSN_MSG);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_insn32_alu2 (bfd_vma pc ATTRIBUTE_UNUSED,
|
|
|
|
disassemble_info *info,
|
|
|
|
uint32_t insn)
|
|
|
|
{
|
|
|
|
int op = insn & 0x3ff;
|
|
|
|
const int rt = RT5 (insn);
|
|
|
|
const int ra = RA5 (insn);
|
|
|
|
const int rb = RB5 (insn);
|
|
|
|
fprintf_ftype func = info->fprintf_func;
|
|
|
|
void *stream = info->stream;
|
|
|
|
|
|
|
|
if ((insn & 0x7f) == 0x4e) /* ffbi */
|
|
|
|
{
|
|
|
|
func (stream, "ffbi\t%s, %s, #0x%x",
|
|
|
|
gpr_map[rt], gpr_map[ra], __GF (insn, 7, 8));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case 0x0: /* max */
|
|
|
|
case 0x1: /* min */
|
|
|
|
case 0x2: /* ave */
|
|
|
|
case 0xc: /* bse */
|
|
|
|
case 0xd: /* bsp */
|
|
|
|
case 0xe: /* ffb */
|
|
|
|
case 0xf: /* ffmism */
|
|
|
|
case 0x17: /* ffzmism */
|
|
|
|
case 0x24: /* mul */
|
|
|
|
func (stream, "%s\t%s, %s, %s", mnemonic_alu20[op],
|
|
|
|
gpr_map[rt], gpr_map[ra], gpr_map[rb]);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x3: /* abs */
|
|
|
|
case 0x6: /* clo */
|
|
|
|
case 0x7: /* clz */
|
|
|
|
func (stream, "%s\t%s, %s", mnemonic_alu20[op], gpr_map[rt], gpr_map[ra]);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x4: /* clips */
|
|
|
|
case 0x5: /* clip */
|
|
|
|
case 0x8: /* bset */
|
|
|
|
case 0x9: /* bclr */
|
|
|
|
case 0xa: /* btgl */
|
|
|
|
case 0xb: /* btst */
|
|
|
|
func (stream, "%s\t%s, %s, #%d", mnemonic_alu20[op],
|
|
|
|
gpr_map[rt], gpr_map[ra], IMM1U (insn));
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x20: /* mfusr */
|
|
|
|
case 0x21: /* mtusr */
|
|
|
|
func (stream, "%s\t%s, $%s", mnemonic_alu20[op],
|
|
|
|
gpr_map[rt], usr_map[__GF (insn, 10, 5)][__GF (insn, 15, 5)]);
|
|
|
|
return;
|
|
|
|
case 0x28: /* mults64 */
|
|
|
|
case 0x29: /* mult64 */
|
|
|
|
case 0x2a: /* madds64 */
|
|
|
|
case 0x2b: /* madd64 */
|
|
|
|
case 0x2c: /* msubs64 */
|
|
|
|
case 0x2d: /* msub64 */
|
|
|
|
case 0x2e: /* divs */
|
|
|
|
case 0x2f: /* div */
|
|
|
|
case 0x31: /* mult32 */
|
|
|
|
case 0x33: /* madd32 */
|
|
|
|
case 0x35: /* msub32 */
|
|
|
|
func (stream, "%s\t$d%d, %s, %s", mnemonic_alu20[op],
|
|
|
|
rt >> 1, gpr_map[ra], gpr_map[rb]);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x4f: /* flmism */
|
|
|
|
case 0x68: /* mulsr64 */
|
|
|
|
case 0x69: /* mulr64 */
|
|
|
|
case 0x73: /* maddr32 */
|
|
|
|
case 0x75: /* msubr32 */
|
|
|
|
op = insn & 0x3f;
|
|
|
|
func (stream, "%s\t%s, %s, %s", mnemonic_alu21[op],
|
|
|
|
gpr_map[rt], gpr_map[ra], gpr_map[rb]);
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
func (stream, UNKNOWN_INSN_MSG);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_insn32_jreg (bfd_vma pc ATTRIBUTE_UNUSED, disassemble_info *info, uint32_t insn)
|
|
|
|
{
|
|
|
|
int op = insn & 0xff;
|
|
|
|
const int rt = RT5 (insn);
|
|
|
|
const int rb = RB5 (insn);
|
|
|
|
const char *dtit_on[] = { "", ".iton", ".dton", ".ton" };
|
|
|
|
const char *dtit_off[] = { "", ".itoff", ".dtoff", ".toff" };
|
|
|
|
const char *mnemonic_jreg[] = { "jr", "jral", "jrnez", "jralnez" };
|
|
|
|
const char *mnemonic_ret[] = { "jr", "ret", NULL, "ifret" };
|
|
|
|
const int dtit = __GF (insn, 8, 2);
|
|
|
|
fprintf_ftype func = info->fprintf_func;
|
|
|
|
void *stream = info->stream;
|
|
|
|
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case 0: /* jr */
|
|
|
|
func (stream, "%s%s\t%s", mnemonic_ret[op >> 5],
|
|
|
|
dtit_on[dtit], gpr_map[rb]);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x20: /* ret */
|
|
|
|
func (stream, "%s%s\t%s", mnemonic_ret[op >> 5],
|
|
|
|
dtit_off[dtit], gpr_map[rb]);
|
|
|
|
return;
|
|
|
|
case 0x60: /* ifret */
|
|
|
|
break;
|
|
|
|
case 1: /* jral */
|
|
|
|
case 2: /* jrnez */
|
|
|
|
case 3: /* jralnez */
|
|
|
|
func (stream, "%s%s\t%s, %s", mnemonic_jreg[op],
|
|
|
|
dtit_on[dtit], gpr_map[rt], gpr_map[rb]);
|
|
|
|
return;
|
|
|
|
default: /* unknown */
|
|
|
|
func (stream, UNKNOWN_INSN_MSG);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_insn32_misc (bfd_vma pc ATTRIBUTE_UNUSED, disassemble_info *info,
|
|
|
|
uint32_t insn)
|
|
|
|
{
|
|
|
|
int op = insn & 0x1f;
|
|
|
|
int rt = RT5 (insn);
|
|
|
|
unsigned int id;
|
|
|
|
fprintf_ftype func = info->fprintf_func;
|
|
|
|
void *stream = info->stream;
|
|
|
|
|
|
|
|
static const char *keyword_standby[] =
|
|
|
|
{
|
|
|
|
"no_wake_grant", "wake_grant", "wait_done",
|
|
|
|
};
|
|
|
|
static const char *keyword_tlbop[] =
|
|
|
|
{
|
|
|
|
"TRD", "TWR", "RWR", "RWLK", "UNLK", "PB", "INV", "FLUA"
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case 0x0: /* standby */
|
|
|
|
id = __GF (insn, 5, 20);
|
|
|
|
if (id < ARRAY_SIZE (keyword_standby))
|
|
|
|
func (stream, "standby\t%s", keyword_standby[id]);
|
|
|
|
else
|
|
|
|
func (stream, "standby\t%d", id);
|
|
|
|
return;
|
|
|
|
case 0x1: /* cctl */
|
|
|
|
func (stream, "cctl\t!FIXME");
|
|
|
|
return;
|
|
|
|
case 0x8: /* dsb */
|
|
|
|
case 0x9: /* isb */
|
|
|
|
case 0xd: /* isync */
|
|
|
|
case 0xc: /* msync */
|
|
|
|
case 0x4: /* iret */
|
|
|
|
func (stream, "%s", mnemonic_misc[op]);
|
|
|
|
return;
|
|
|
|
case 0x5: /* trap */
|
|
|
|
case 0xa: /* break */
|
|
|
|
case 0xb: /* syscall */
|
|
|
|
id = __GF (insn, 5, 15);
|
|
|
|
func (stream, "%s\t%d", mnemonic_misc[op], id);
|
|
|
|
return;
|
|
|
|
case 0x2: /* mfsr */
|
|
|
|
case 0x3: /* mtsr */
|
|
|
|
/* FIXME: setend, setgie. */
|
2013-12-12 13:43:51 +08:00
|
|
|
func (stream, "%s\t%s, $%s", mnemonic_misc[op], gpr_map[rt],
|
|
|
|
sr_map[__GF(insn, 17, 3)][__GF(insn, 13, 4)][__GF(insn, 10, 3)]);
|
2013-12-13 19:52:32 +08:00
|
|
|
return;
|
|
|
|
case 0x6: /* teqz */
|
|
|
|
case 0x7: /* tnez */
|
|
|
|
id = __GF (insn, 5, 15);
|
|
|
|
func (stream, "%s\t%s, %d", mnemonic_misc[op], gpr_map[rt], id);
|
|
|
|
return;
|
|
|
|
case 0xe: /* tlbop */
|
|
|
|
id = __GF (insn, 5, 5);
|
|
|
|
if (id < ARRAY_SIZE (keyword_tlbop))
|
|
|
|
func (stream, "tlbop\t%s", keyword_tlbop[id]);
|
|
|
|
else
|
|
|
|
func (stream, "tlbop\t%d", id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_insn32_fpu (bfd_vma pc ATTRIBUTE_UNUSED, disassemble_info *info,
|
|
|
|
uint32_t insn)
|
|
|
|
{
|
|
|
|
int op = insn & 0xf;
|
|
|
|
int mask_sub_op = (insn & 0x3c0) >> 6;
|
|
|
|
int mask_bi = (insn & 0x80) >> 7;
|
|
|
|
int mask_cfg = (insn & 0x7c00) >> 10;
|
|
|
|
int mask_f2op = (insn & 0x7c00) >> 10;
|
|
|
|
int dp = 0;
|
|
|
|
int dp_insn = 0;
|
|
|
|
char wd = 's';
|
|
|
|
const int rt = RT5 (insn);
|
|
|
|
const int ra = RA5 (insn);
|
|
|
|
const int rb = RB5 (insn);
|
|
|
|
const int sv = __GF (insn, 8, 2);
|
|
|
|
fprintf_ftype func = info->fprintf_func;
|
|
|
|
void *stream = info->stream;
|
|
|
|
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case 0x0: /* fs1 */
|
|
|
|
case 0x8: /* fd1 */
|
|
|
|
dp = (op & 0x8) ? 1 : 0;
|
|
|
|
if (dp)
|
|
|
|
{
|
|
|
|
wd = 'd';
|
|
|
|
dp_insn = 14;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
wd = 's';
|
|
|
|
dp_insn = 0;
|
|
|
|
}
|
|
|
|
switch (mask_sub_op)
|
|
|
|
{
|
|
|
|
case 0x0:
|
|
|
|
case 0x1:
|
|
|
|
case 0x2:
|
|
|
|
case 0x3:
|
|
|
|
case 0x4:
|
|
|
|
case 0x5:
|
|
|
|
case 0x8:
|
|
|
|
case 0x9:
|
|
|
|
case 0xc:
|
|
|
|
case 0xd:
|
|
|
|
func (stream, "%s\t$f%c%d, $f%c%d, $f%c%d",
|
|
|
|
mnemonic_alu[mask_sub_op + dp_insn],
|
|
|
|
wd, rt, wd, ra, wd, rb);
|
|
|
|
return;
|
|
|
|
case 0x6:
|
|
|
|
case 0x7:
|
|
|
|
func (stream, "%s\t$f%c%d, $f%c%d, $fs%d",
|
|
|
|
mnemonic_alu[mask_sub_op + dp_insn],
|
|
|
|
wd, rt, wd, ra, rb);
|
|
|
|
return;
|
|
|
|
case 0xf:
|
|
|
|
if (dp)
|
|
|
|
{
|
|
|
|
wd = 'd';
|
|
|
|
dp_insn = 0x1d;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
wd = 's';
|
|
|
|
dp_insn = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (mask_f2op)
|
|
|
|
{
|
|
|
|
case 0x0:
|
|
|
|
if (dp)
|
|
|
|
func (stream, "%s\t$fs%d, $fd%d",
|
|
|
|
mnemonic_fpu_2op[mask_f2op + dp_insn], rt, ra);
|
|
|
|
else
|
|
|
|
func (stream, "%s\t$fd%d, $fs%d",
|
|
|
|
mnemonic_fpu_2op[mask_f2op + dp_insn], rt, ra);
|
|
|
|
return;
|
|
|
|
case 0x1:
|
|
|
|
case 0x5:
|
|
|
|
func (stream, "%s\t$f%c%d, $f%c%d",
|
|
|
|
mnemonic_fpu_2op[mask_f2op + dp_insn], wd, rt, wd, ra);
|
|
|
|
return;
|
|
|
|
case 0x8:
|
|
|
|
case 0xc:
|
|
|
|
func (stream, "%s\t$f%c%d, $fs%d",
|
|
|
|
mnemonic_fpu_2op[mask_f2op + dp_insn], wd, rt, ra);
|
|
|
|
return;
|
|
|
|
case 0x10:
|
|
|
|
case 0x14:
|
|
|
|
case 0x18:
|
|
|
|
case 0x1c:
|
|
|
|
func (stream, "%s\t$fs%d, $f%c%d",
|
|
|
|
mnemonic_fpu_2op[mask_f2op + dp_insn], rt, wd, ra);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case 0x1: /* mfcp */
|
|
|
|
switch (mask_sub_op)
|
|
|
|
{
|
|
|
|
case 0x0:
|
|
|
|
func (stream, "fmfsr\t%s, $fs%d", gpr_map[rt], ra);
|
|
|
|
return;
|
|
|
|
case 0x1:
|
|
|
|
func (stream, "fmfdr\t%s, $fd%d", gpr_map[rt], ra);
|
|
|
|
return;
|
|
|
|
case 0xc:
|
|
|
|
if (mask_cfg)
|
|
|
|
func (stream, "fmfcsr\t%s", gpr_map[rt]);
|
|
|
|
else
|
|
|
|
func (stream, "fmfcfg\t%s", gpr_map[rt]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case 0x2: /* fls */
|
|
|
|
if (mask_bi)
|
|
|
|
func (stream, "fls.bi\t$fs%d, [%s], (%s << %d)",
|
|
|
|
rt, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
else
|
|
|
|
func (stream, "fls\t$fs%d, [%s + (%s << %d)]",
|
|
|
|
rt, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
return;
|
|
|
|
case 0x3: /* fld */
|
|
|
|
if (mask_bi)
|
|
|
|
func (stream, "fld.bi\t$fd%d, [%s], (%s << %d)",
|
|
|
|
rt, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
else
|
|
|
|
func (stream, "fld\t$fd%d, [%s + (%s << %d)]",
|
|
|
|
rt, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
return;
|
|
|
|
case 0x4: /* fs2 */
|
|
|
|
func (stream, "%s\t$fs%d, $fs%d, $fs%d",
|
|
|
|
mnemonic_fs2_cmp[mask_sub_op], rt, ra, rb);
|
|
|
|
return;
|
|
|
|
case 0x9: /* mtcp */
|
|
|
|
switch (mask_sub_op)
|
|
|
|
{
|
|
|
|
case 0x0:
|
|
|
|
func (stream, "fmtsr\t%s, $fs%d", gpr_map[rt], ra);
|
|
|
|
return;
|
|
|
|
case 0x1:
|
|
|
|
func (stream, "fmtdr\t%s, $fd%d", gpr_map[rt], ra);
|
|
|
|
return;
|
|
|
|
case 0xc:
|
|
|
|
func (stream, "fmtcsr\t%s", gpr_map[rt]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case 0xa: /* fss */
|
|
|
|
if (mask_bi)
|
|
|
|
func (stream, "fss.bi\t$fs%d, [%s], (%s << %d)",
|
|
|
|
rt, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
else
|
|
|
|
func (stream, "fss\t$fs%d, [%s + (%s << %d)]",
|
|
|
|
rt, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
return;
|
|
|
|
case 0xb: /* fsd */
|
|
|
|
if (mask_bi)
|
|
|
|
func (stream, "fsd.bi\t$fd%d, [%s], (%s << %d)",
|
|
|
|
rt, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
else
|
|
|
|
func (stream, "fsd\t$fd%d, [%s + (%s << %d)]",
|
|
|
|
rt, gpr_map[ra], gpr_map[rb], sv);
|
|
|
|
return;
|
|
|
|
case 0xc: /* fd2 */
|
|
|
|
func (stream, "%s\t$fs%d, $fd%d, $fd%d",
|
|
|
|
mnemonic_fd2_cmp[mask_sub_op], rt, ra, rb);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_insn32 (bfd_vma pc, disassemble_info *info, uint32_t insn)
|
|
|
|
{
|
|
|
|
int op = OP6 (insn);
|
|
|
|
const int rt = RT5 (insn);
|
|
|
|
const int ra = RA5 (insn);
|
|
|
|
const int rb = RB5 (insn);
|
|
|
|
const int imm15s = IMMS (insn, 15);
|
|
|
|
const int imm15u = IMMU (insn, 15);
|
|
|
|
uint32_t shift;
|
|
|
|
fprintf_ftype func = info->fprintf_func;
|
|
|
|
void *stream = info->stream;
|
|
|
|
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case 0x0: /* lbi */
|
|
|
|
case 0x1: /* lhi */
|
|
|
|
case 0x2: /* lwi */
|
|
|
|
case 0x3: /* ldi */
|
|
|
|
case 0x8: /* sbi */
|
|
|
|
case 0x9: /* shi */
|
|
|
|
case 0xa: /* swi */
|
|
|
|
case 0xb: /* sdi */
|
|
|
|
case 0x10: /* lbsi */
|
|
|
|
case 0x11: /* lhsi */
|
|
|
|
case 0x12: /* lwsi */
|
|
|
|
shift = op & 0x3;
|
|
|
|
func (stream, "%s\t%s, [%s + #%d]",
|
|
|
|
mnemonic_op6[op], gpr_map[rt], gpr_map[ra], imm15s << shift);
|
|
|
|
return;
|
|
|
|
case 0x4: /* lbi.bi */
|
|
|
|
case 0x5: /* lhi.bi */
|
|
|
|
case 0x6: /* lwi.bi */
|
|
|
|
case 0x7: /* ldi.bi */
|
|
|
|
case 0xc: /* sbi.bi */
|
|
|
|
case 0xd: /* shi.bi */
|
|
|
|
case 0xe: /* swi.bi */
|
|
|
|
case 0xf: /* sdi.bi */
|
|
|
|
case 0x14: /* lbsi.bi */
|
|
|
|
case 0x15: /* lhsi.bi */
|
|
|
|
case 0x16: /* lwsi.bi */
|
|
|
|
shift = op & 0x3;
|
|
|
|
func (stream, "%s\t%s, [%s], #%d",
|
|
|
|
mnemonic_op6[op], gpr_map[rt], gpr_map[ra], imm15s << shift);
|
|
|
|
return;
|
|
|
|
case 0x13: /* dprefi */
|
|
|
|
{
|
|
|
|
const char *subtype = "???";
|
|
|
|
char wd = 'w';
|
|
|
|
|
|
|
|
shift = 2;
|
|
|
|
|
|
|
|
/* d-bit */
|
|
|
|
if (rt & 0x10)
|
|
|
|
{
|
|
|
|
wd = 'd';
|
|
|
|
shift = 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((rt & 0xf) < ARRAY_SIZE (keyword_dpref))
|
|
|
|
subtype = keyword_dpref[rt & 0xf];
|
|
|
|
|
|
|
|
func (stream, "%s.%c\t%s, [%s + #%d]",
|
|
|
|
mnemonic_op6[op], wd, subtype, gpr_map[ra], imm15s << shift);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case 0x17: /* LBGP */
|
|
|
|
func (stream, "%s\t%s, [+ %d]",
|
|
|
|
((insn & __BIT (19)) ? "lbsi.gp" : "lbi.gp"),
|
|
|
|
gpr_map[rt], IMMS (insn, 19));
|
|
|
|
return;
|
|
|
|
case 0x18: /* LWC */
|
|
|
|
case 0x19: /* SWC */
|
|
|
|
case 0x1a: /* LDC */
|
|
|
|
case 0x1b: /* SDC */
|
|
|
|
if (__GF (insn, 13, 2) == 0)
|
|
|
|
{
|
|
|
|
char ls = (op & 1) ? 's' : 'l';
|
|
|
|
char wd = (op & 2) ? 'd' : 's';
|
|
|
|
|
|
|
|
if (insn & __BIT (12))
|
|
|
|
{
|
|
|
|
func (stream, "f%c%ci.bi\t$f%c%d, [%s], %d", ls, wd,
|
|
|
|
wd, rt, gpr_map[ra], IMMS (insn, 12) << 2);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
func (stream, "f%c%ci\t$f%c%d, [%s + %d]", ls, wd,
|
|
|
|
wd, rt, gpr_map[ra], IMMS (insn, 12) << 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
char ls = (op & 1) ? 's' : 'l';
|
|
|
|
char wd = (op & 2) ? 'd' : 'w';
|
|
|
|
int cp = __GF (insn, 13, 2);
|
|
|
|
|
|
|
|
if (insn & __BIT (12))
|
|
|
|
{
|
|
|
|
func (stream, "cp%c%ci\tcp%d, $cpr%d, [%s], %d", ls, wd,
|
|
|
|
cp, rt, gpr_map[ra], IMMS (insn, 12) << 2);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
func (stream, "cp%c%ci\tcp%d, $cpr%d, [%s + %d]", ls, wd,
|
|
|
|
cp, rt, gpr_map[ra], IMMS (insn, 12) << 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case 0x1c: /* MEM */
|
|
|
|
print_insn32_mem (pc, info, insn);
|
|
|
|
return;
|
|
|
|
case 0x1d: /* LSMW */
|
|
|
|
{
|
|
|
|
int enb4 = __GF (insn, 6, 4);
|
|
|
|
char ls = (insn & __BIT (5)) ? 's' : 'l';
|
|
|
|
char ab = (insn & __BIT (4)) ? 'a' : 'b';
|
|
|
|
char *di = (insn & __BIT (3)) ? "d" : "i";
|
|
|
|
char *m = (insn & __BIT (2)) ? "m" : "";
|
|
|
|
static const char *s[] = {"", "a", "zb", "?"};
|
|
|
|
|
|
|
|
/* lsmwzb only always increase. */
|
|
|
|
if ((insn & 0x3) == 2)
|
|
|
|
di = "";
|
|
|
|
|
|
|
|
func (stream, "%cmw%s.%c%s%s\t%s, [%s], %s, 0x%x",
|
|
|
|
ls, s[insn & 0x3], ab, di, m, gpr_map[rt],
|
|
|
|
gpr_map[ra], gpr_map[rb], enb4);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case 0x1e: /* HWGP */
|
|
|
|
op = __GF (insn, 17, 3);
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case 0: case 1: /* lhi.gp */
|
|
|
|
case 2: case 3: /* lhsi.gp */
|
|
|
|
case 4: case 5: /* shi.gp */
|
|
|
|
func (stream, "%s\t%s, [+ %d]",
|
|
|
|
mnemonic_hwgp[op], gpr_map[rt], IMMS (insn, 18) << 1);
|
|
|
|
return;
|
|
|
|
case 6: /* lwi.gp */
|
|
|
|
case 7: /* swi.gp */
|
|
|
|
func (stream, "%s\t%s, [+ %d]",
|
|
|
|
mnemonic_hwgp[op], gpr_map[rt], IMMS (insn, 17) << 2);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case 0x1f: /* SBGP */
|
|
|
|
if (insn & __BIT (19))
|
|
|
|
func (stream, "addi.gp\t%s, %d",
|
|
|
|
gpr_map[rt], IMMS (insn, 19));
|
|
|
|
else
|
|
|
|
func (stream, "sbi.gp\t%s, [+ %d]",
|
|
|
|
gpr_map[rt], IMMS (insn, 19));
|
|
|
|
return;
|
|
|
|
case 0x20: /* ALU_1 */
|
|
|
|
print_insn32_alu1 (pc, info, insn);
|
|
|
|
return;
|
|
|
|
case 0x21: /* ALU_2 */
|
|
|
|
print_insn32_alu2 (pc, info, insn);
|
|
|
|
return;
|
|
|
|
case 0x22: /* movi */
|
|
|
|
func (stream, "movi\t%s, %d", gpr_map[rt], IMMS (insn, 20));
|
|
|
|
return;
|
|
|
|
case 0x23: /* sethi */
|
|
|
|
func (stream, "sethi\t%s, 0x%x", gpr_map[rt], IMMU (insn, 20));
|
|
|
|
return;
|
|
|
|
case 0x24: /* ji, jal */
|
|
|
|
/* FIXME: Handle relocation. */
|
|
|
|
if (info->flags & INSN_HAS_RELOC)
|
|
|
|
pc = 0;
|
|
|
|
func (stream, "%s\t", ((insn & __BIT (24)) ? "jal" : "j"));
|
|
|
|
info->print_address_func ((IMMS (insn, 24) << 1) + pc, info);
|
|
|
|
return;
|
|
|
|
case 0x25: /* jreg */
|
|
|
|
print_insn32_jreg (pc, info, insn);
|
|
|
|
return;
|
|
|
|
case 0x26: /* br1 */
|
|
|
|
func (stream, "%s\t%s, %s, ", ((insn & __BIT (14)) ? "bne" : "beq"),
|
|
|
|
gpr_map[rt], gpr_map[ra]);
|
|
|
|
info->print_address_func ((IMMS (insn, 14) << 1) + pc, info);
|
|
|
|
return;
|
|
|
|
case 0x27: /* br2 */
|
|
|
|
func (stream, "%s\t%s, ", mnemonic_br2[__GF (insn, 16, 4)],
|
|
|
|
gpr_map[rt]);
|
|
|
|
info->print_address_func ((IMMS (insn, 16) << 1) + pc, info);
|
|
|
|
return;
|
|
|
|
case 0x28: /* addi */
|
|
|
|
case 0x2e: /* slti */
|
|
|
|
case 0x2f: /* sltsi */
|
|
|
|
case 0x29: /* subri */
|
|
|
|
func (stream, "%s\t%s, %s, %d",
|
|
|
|
mnemonic_op6[op], gpr_map[rt], gpr_map[ra], imm15s);
|
|
|
|
return;
|
|
|
|
case 0x2a: /* andi */
|
|
|
|
case 0x2b: /* xori */
|
|
|
|
case 0x2c: /* ori */
|
|
|
|
case 0x33: /* bitci */
|
|
|
|
func (stream, "%s\t%s, %s, %d",
|
|
|
|
mnemonic_op6[op], gpr_map[rt], gpr_map[ra], imm15u);
|
|
|
|
return;
|
|
|
|
case 0x2d: /* br3, beqc, bnec */
|
|
|
|
func (stream, "%s\t%s, %d, ", ((insn & __BIT (19)) ? "bnec" : "beqc"),
|
|
|
|
gpr_map[rt], __SEXT (__GF (insn, 8, 11), 11));
|
|
|
|
info->print_address_func ((IMMS (insn, 8) << 1) + pc, info);
|
|
|
|
return;
|
|
|
|
case 0x32: /* misc */
|
|
|
|
print_insn32_misc (pc, info, insn);
|
|
|
|
return;
|
|
|
|
case 0x35: /* FPU */
|
|
|
|
print_insn32_fpu (pc, info, insn);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
print_insn_nds32 (bfd_vma pc, disassemble_info *info)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
bfd_byte buf[4];
|
|
|
|
uint32_t insn;
|
|
|
|
|
|
|
|
status = info->read_memory_func (pc, (bfd_byte *) buf, 2, info);
|
|
|
|
if (status)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* 16-bit instruction. */
|
|
|
|
if (buf[0] & 0x80)
|
|
|
|
{
|
|
|
|
insn = bfd_getb16 (buf);
|
|
|
|
print_insn16 (pc, info, insn);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 32-bit instructions. */
|
|
|
|
status = info->read_memory_func (pc + 2, (bfd_byte *) buf + 2, 2, info);
|
|
|
|
if (status)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
insn = bfd_getb32 (buf);
|
|
|
|
print_insn32 (pc, info, insn);
|
|
|
|
|
|
|
|
return 4;
|
|
|
|
}
|