aboutsummaryrefslogtreecommitdiff
path: root/src/target/nds32_disassembler.c
diff options
context:
space:
mode:
authorHsiangkai Wang <hsiangkai@gmail.com>2013-02-05 11:55:37 +0800
committerSpencer Oliver <spen@spen-soft.co.uk>2013-06-05 19:27:35 +0000
commitcf8a3c3d7075abad3c88cd604f8add4d06898abc (patch)
tree56f8b5794fd385ba7ba4a6617c214a9516a443b5 /src/target/nds32_disassembler.c
parentceb402dc9e903d2f3f6bc8125dfed9d82b83d2d1 (diff)
nds32: add new target type nds32_v2, nds32_v3, nds32_v3m
Add target code for Andes targets. Change-Id: Ibf0e1b61b06127ca7d9ed502d98d7e2aeebbbe82 Signed-off-by: Hsiangkai Wang <hsiangkai@gmail.com> Reviewed-on: http://openocd.zylin.com/1259 Tested-by: jenkins Reviewed-by: Spencer Oliver <spen@spen-soft.co.uk>
Diffstat (limited to 'src/target/nds32_disassembler.c')
-rw-r--r--src/target/nds32_disassembler.c3637
1 files changed, 3637 insertions, 0 deletions
diff --git a/src/target/nds32_disassembler.c b/src/target/nds32_disassembler.c
new file mode 100644
index 00000000..d859e0c6
--- /dev/null
+++ b/src/target/nds32_disassembler.c
@@ -0,0 +1,3637 @@
+/***************************************************************************
+ * Copyright (C) 2013 Andes Technology *
+ * Hsiangkai Wang <hkwang@andestech.com> *
+ * *
+ * 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 2 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. *
+ ***************************************************************************/
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <helper/log.h>
+#include <target/target.h>
+#include "nds32_disassembler.h"
+
+static const int enable4_bits[] = {0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4};
+
+int nds32_read_opcode(struct nds32 *nds32, uint32_t address, uint32_t *value)
+{
+ struct target *target = nds32->target;
+ uint8_t value_buf[4];
+
+ if (!target_was_examined(target)) {
+ LOG_ERROR("Target not examined yet");
+ return ERROR_FAIL;
+ }
+
+ int retval = target_read_buffer(target, address, 4, value_buf);
+
+ if (retval == ERROR_OK) {
+ /* instructions are always big-endian */
+ *value = be_to_h_u32(value_buf);
+
+ LOG_DEBUG("address: 0x%8.8" PRIx32 ", value: 0x%8.8" PRIx32 "",
+ address,
+ *value);
+ } else {
+ *value = 0x0;
+ LOG_DEBUG("address: 0x%8.8" PRIx32 " failed",
+ address);
+ }
+
+ return retval;
+}
+
+static int nds32_parse_type_0(uint32_t opcode, int32_t *imm)
+{
+ *imm = opcode & 0x1FFFFFF;
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_type_1(uint32_t opcode, uint8_t *rt, int32_t *imm)
+{
+ *rt = (opcode >> 20) & 0x1F;
+ *imm = opcode & 0xFFFFF;
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_type_2(uint32_t opcode, uint8_t *rt, uint8_t *ra, int32_t *imm)
+{
+ *rt = (opcode >> 20) & 0x1F;
+ *ra = (opcode >> 15) & 0x1F;
+ *imm = opcode & 0x7FFF;
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_type_3(uint32_t opcode, uint8_t *rt, uint8_t *ra,
+ uint8_t *rb, int32_t *imm)
+{
+ *rt = (opcode >> 20) & 0x1F;
+ *ra = (opcode >> 15) & 0x1F;
+ *rb = (opcode >> 10) & 0x1F;
+ *imm = opcode & 0x3FF;
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_type_4(uint32_t opcode, uint8_t *rt, uint8_t *ra,
+ uint8_t *rb, uint8_t *rd, uint8_t *sub_opc)
+{
+ *rt = (opcode >> 20) & 0x1F;
+ *ra = (opcode >> 15) & 0x1F;
+ *rb = (opcode >> 10) & 0x1F;
+ *rd = (opcode >> 5) & 0x1F;
+ *sub_opc = opcode & 0x1F;
+
+ return ERROR_OK;
+}
+
+/* LBI, LHI, LWI, LBI.bi, LHI.bi, LWI.bi */
+static int nds32_parse_group_0_insn(struct nds32 *nds32, uint32_t opcode,
+ uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ uint8_t opc_6;
+
+ opc_6 = instruction->info.opc_6;
+
+ switch (opc_6 & 0x7) {
+ case 0: /* LBI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBI\t$r%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 1: /* LHI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 16; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLHI\t$r%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 2: /* LWI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 15; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLWI\t$r%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 4: /* LBI.bi */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBI.bi\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 5: /* LHI.bi */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 16; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLHI.bi\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 6: /* LWI.bi */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 15; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLWI.bi\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_1_insn(struct nds32 *nds32, uint32_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ uint8_t opc_6;
+
+ opc_6 = instruction->info.opc_6;
+
+ switch (opc_6 & 0x7) {
+ case 0: /* SBI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSBI\t$r%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 1: /* SHI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 16; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSHI\t$r%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 2: /* SWI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 15; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSWI\t$r%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 4: /* SBI.bi */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSBI.bi\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 5: /* SHI.bi */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 16; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSHI.bi\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 6: /* SWI.bi */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 15; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSWI.bi\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_2_insn(struct nds32 *nds32, uint32_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ uint8_t opc_6;
+
+ opc_6 = instruction->info.opc_6;
+
+ switch (opc_6 & 0x7) {
+ case 0: /* LBSI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBSI\t$r%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 1: /* LHSI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 16; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLHSI\t$r%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 3: { /* DPREFI */
+ uint8_t sub_type;
+ nds32_parse_type_2(opcode, &sub_type, &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->info.sub_opc = sub_type & 0xF;
+ instruction->type = NDS32_INSN_MISC;
+ if (sub_type & 0x10) { /* DPREFI.d */
+ /* sign-extend */
+ instruction->info.imm = (instruction->info.imm << 17) >> 14;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tDPREFI.d\t%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.sub_opc,
+ instruction->info.ra, instruction->info.imm);
+ } else { /* DPREFI.w */
+ /* sign-extend */
+ instruction->info.imm = (instruction->info.imm << 17) >> 15;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tDPREFI.w\t%d,[$r%d+#%d]",
+ address,
+ opcode, instruction->info.sub_opc,
+ instruction->info.ra, instruction->info.imm);
+ }
+ }
+ break;
+ case 4: /* LBSI.bi */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBSI.bi\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 5: /* LHSI.bi */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 16; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLHSI.bi\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 6: /* LBGP */
+ nds32_parse_type_1(opcode, &(instruction->info.rt), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ if ((instruction->info.imm >> 19) & 0x1) { /* LBSI.gp */
+ instruction->info.imm = (instruction->info.imm << 13) >> 13;
+ nds32_get_mapped_reg(nds32, R29, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBSI.gp\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ } else { /* LBI.gp */
+ instruction->info.imm = (instruction->info.imm << 13) >> 13;
+ nds32_get_mapped_reg(nds32, R29, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBI.gp\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ }
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_mem(struct nds32 *nds32, uint32_t opcode, uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ uint32_t sub_opcode = opcode & 0x3F;
+ uint32_t val_ra, val_rb;
+ switch (sub_opcode >> 3) {
+ case 0:
+ switch (sub_opcode & 0x7) {
+ case 0: /* LB */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), \
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLB\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 1: /* LH */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLH\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 2: /* LW */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLW\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 4: /* LB.bi */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLB.bi\t$r%d,[$r%d],($r%d<<%d)",
+ address,
+ opcode, instruction->info.rt,
+ instruction->info.ra, instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 5: /* LH.bi */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLH.bi\t$r%d,[$r%d],($r%d<<%d)",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 6: /* LW.bi */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLW.bi\t$r%d,[$r%d],($r%d<<%d)",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ }
+ break;
+ case 1:
+ switch (sub_opcode & 0x7) {
+ case 0: /* SB */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSB\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt,
+ instruction->info.ra, instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 1: /* SH */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSH\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 2: /* SW */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSW\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt,
+ instruction->info.ra, instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 4: /* SB.bi */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSB.bi\t$r%d,[$r%d],($r%d<<%d)",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 5: /* SH.bi */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSH.bi\t$r%d,[$r%d],($r%d<<%d)",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 6: /* SW.bi */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSW.bi\t$r%d,[$r%d],($r%d<<%d)",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ }
+ break;
+ case 2:
+ switch (sub_opcode & 0x7) {
+ case 0: /* LBS */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBS\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt,
+ instruction->info.ra, instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 1: /* LHS */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLHS\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 3: /* DPREF */
+ nds32_parse_type_3(opcode, &(instruction->info.sub_opc),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tDPREF\t#%d,[$r%d+($r%d<<#%d)]",
+ address,
+ opcode, instruction->info.sub_opc,
+ instruction->info.ra, instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 4: /* LBS.bi */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBS.bi\t$r%d,[$r%d],($r%d<<%d)",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 5: /* LHS.bi */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLHS.bi\t$r%d,[$r%d],($r%d<<%d)",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ }
+ break;
+ case 3:
+ switch (sub_opcode & 0x7) {
+ case 0: /* LLW */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLLW\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 1: /* SCW */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSCW\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ }
+ break;
+ case 4:
+ switch (sub_opcode & 0x7) {
+ case 0: /* LBUP */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLBUP\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 2: /* LWUP */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLWUP\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ }
+ break;
+ case 5:
+ switch (sub_opcode & 0x7) {
+ case 0: /* SBUP */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSBUP\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ case 2: /* SWUP */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &val_ra);
+ nds32_get_mapped_reg(nds32, instruction->info.rb, &val_rb);
+ instruction->access_start = val_ra +
+ (val_rb << ((instruction->info.imm >> 8) & 0x3));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSWUP\t$r%d,[$r%d+($r%d<<%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ (instruction->info.imm >> 8) & 0x3);
+ break;
+ }
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_calculate_lsmw_access_range(struct nds32 *nds32,
+ struct nds32_instruction *instruction)
+{
+ uint8_t ba;
+ uint8_t id;
+ uint8_t enable4;
+
+ enable4 = (instruction->info.imm >> 6) & 0xF;
+ ba = (instruction->info.imm >> 4) & 0x1;
+ id = (instruction->info.imm >> 3) & 0x1;
+
+ if (ba) {
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &(instruction->access_start));
+ if (id) { /* decrease */
+ /* access_end is the (last_element+1), so no need to minus 4 */
+ /* instruction->access_end -= 4; */
+ instruction->access_end = instruction->access_start;
+ } else { /* increase */
+ instruction->access_start += 4;
+ }
+ } else {
+ nds32_get_mapped_reg(nds32, instruction->info.ra, &(instruction->access_start));
+ instruction->access_end = instruction->access_start - 4;
+ }
+
+ if (id) { /* decrease */
+ instruction->access_start = instruction->access_end -
+ 4 * (instruction->info.rd - instruction->info.rb + 1);
+ instruction->access_start -= (4 * enable4_bits[enable4]);
+ } else { /* increase */
+ instruction->access_end = instruction->access_start +
+ 4 * (instruction->info.rd - instruction->info.rb + 1);
+ instruction->access_end += (4 * enable4_bits[enable4]);
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_lsmw(struct nds32 *nds32, uint32_t opcode, uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ if (opcode & 0x20) { /* SMW, SMWA, SMWZB */
+ switch (opcode & 0x3) {
+ /* TODO */
+ case 0: /* SMW */
+ /* use rd as re */
+ nds32_parse_type_3(opcode, &(instruction->info.rb),
+ &(instruction->info.ra),
+ &(instruction->info.rd), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_calculate_lsmw_access_range(nds32, instruction);
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSMW\t$r%d,[$r%d],$r%d,%d",
+ address,
+ opcode, instruction->info.rb, instruction->info.ra,
+ instruction->info.rd,
+ (instruction->info.imm >> 6) & 0xF);
+ break;
+ case 1: /* SMWA */
+ nds32_parse_type_3(opcode, &(instruction->info.rb),
+ &(instruction->info.ra),
+ &(instruction->info.rd), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_calculate_lsmw_access_range(nds32, instruction);
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSMWA\t$r%d,[$r%d],$r%d,%d",
+ address,
+ opcode, instruction->info.rb, instruction->info.ra,
+ instruction->info.rd,
+ (instruction->info.imm >> 6) & 0xF);
+ break;
+ case 2: /* SMWZB */
+ nds32_parse_type_3(opcode, &(instruction->info.rb),
+ &(instruction->info.ra),
+ &(instruction->info.rd), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ /* TODO: calculate access_start/access_end */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSMWZB\t$r%d,[$r%d],$r%d,%d",
+ address,
+ opcode, instruction->info.rb, instruction->info.ra,
+ instruction->info.rd,
+ (instruction->info.imm >> 6) & 0xF);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+ } else { /* LMW, LMWA, LMWZB */
+ switch (opcode & 0x3) {
+ case 0: /* LMW */
+ nds32_parse_type_3(opcode, &(instruction->info.rb),
+ &(instruction->info.ra),
+ &(instruction->info.rd), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_calculate_lsmw_access_range(nds32, instruction);
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLMW\t$r%d,[$r%d],$r%d,%d",
+ address,
+ opcode, instruction->info.rb, instruction->info.ra,
+ instruction->info.rd,
+ (instruction->info.imm >> 6) & 0xF);
+ break;
+ case 1: /* LMWA */
+ nds32_parse_type_3(opcode, &(instruction->info.rb),
+ &(instruction->info.ra),
+ &(instruction->info.rd), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_calculate_lsmw_access_range(nds32, instruction);
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLMWA\t$r%d,[$r%d],$r%d,%d",
+ address,
+ opcode, instruction->info.rb, instruction->info.ra,
+ instruction->info.rd,
+ (instruction->info.imm >> 6) & 0xF);
+ break;
+ case 2: /* LMWZB */
+ nds32_parse_type_3(opcode, &(instruction->info.rb),
+ &(instruction->info.ra),
+ &(instruction->info.rd), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ /* TODO: calculate access_start/access_end */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLMWZB\t$r%d,[$r%d],$r%d,%d",
+ address,
+ opcode, instruction->info.rb, instruction->info.ra,
+ instruction->info.rd,
+ (instruction->info.imm >> 6) & 0xF);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_hwgp(struct nds32 *nds32, uint32_t opcode, uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ switch ((opcode >> 18) & 0x3) {
+ case 0: /* LHI.gp */
+ nds32_parse_type_1(opcode, &(instruction->info.rt), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 14) >> 13; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, R29, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLHI.gp\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 1: /* LHSI.gp */
+ nds32_parse_type_1(opcode, &(instruction->info.rt), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 14) >> 13; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, R29, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLHSI.gp\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 2: /* SHI.gp */
+ nds32_parse_type_1(opcode, &(instruction->info.rt), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 14) >> 13; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, R29, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSHI.gp\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 3:
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ if ((opcode >> 17) & 0x1) { /* SWI.gp */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ /* sign-extend */
+ instruction->info.imm = (instruction->info.imm << 15) >> 13;
+ nds32_get_mapped_reg(nds32, R29, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSWI.gp\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ } else { /* LWI.gp */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ /* sign-extend */
+ instruction->info.imm = (instruction->info.imm << 15) >> 13;
+ nds32_get_mapped_reg(nds32, R29, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tLWI.gp\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ }
+
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_sbgp(struct nds32 *nds32, uint32_t opcode, uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ switch ((opcode >> 19) & 0x1) {
+ case 0: /* SBI.gp */
+ nds32_parse_type_1(opcode, &(instruction->info.rt), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 13) >> 13; /* sign-extend */
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, R29, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSBI.gp\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 1: /* ADDI.gp */
+ nds32_parse_type_1(opcode, &(instruction->info.rt), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 13) >> 13; /* sign-extend */
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tADDI.gp\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_3_insn(struct nds32 *nds32, uint32_t opcode, uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ uint8_t opc_6;
+
+ opc_6 = instruction->info.opc_6;
+
+ switch (opc_6 & 0x7) {
+ case 4: /* MEM */
+ nds32_parse_mem(nds32, opcode, address, instruction);
+ break;
+ case 5: /* LSMW */
+ nds32_parse_lsmw(nds32, opcode, address, instruction);
+ break;
+ case 6: /* HWGP */
+ nds32_parse_hwgp(nds32, opcode, address, instruction);
+ break;
+ case 7: /* SBGP */
+ nds32_parse_sbgp(nds32, opcode, address, instruction);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_alu_1(uint32_t opcode, uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ switch (opcode & 0x1F) {
+ case 0: /* ADD */
+ nds32_parse_type_3(opcode, &(instruction->info.rt), &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tADD_SLLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tADD\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 1: /* SUB */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSUB_SLLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSUB\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 2: /* AND */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tAND_SLLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tAND\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 3: /* XOR */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tXOR_SLLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tXOR\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 4: /* OR */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tOR_SLLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tOR\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 5: /* NOR */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tNOR\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 6: /* SLT */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSLT\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 7: /* SLTS */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSLTS\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 8: { /* SLLI */
+ uint8_t imm;
+ int32_t sub_op;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &sub_op);
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSLLI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 9: { /* SRLI */
+ uint8_t imm;
+ int32_t sub_op;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &sub_op);
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSRLI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 10: { /* SRAI */
+ uint8_t imm;
+ int32_t sub_op;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &sub_op);
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSRAI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 11: { /* ROTRI */
+ uint8_t imm;
+ int32_t sub_op;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &sub_op);
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tROTRI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 12: { /* SLL */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSLL\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 13: { /* SRL */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSRL\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 14: { /* SRA */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSRA\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 15: { /* ROTR */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tROTR\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 16: { /* SEB */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSEB\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ }
+ break;
+ case 17: { /* SEH */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSEH\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ }
+ break;
+ case 18: /* BITC */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBITC\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 19: { /* ZEH */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tZEH\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ }
+ break;
+ case 20: { /* WSBH */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tWSBH\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ }
+ break;
+ case 21: /* OR_SRLI */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tOR_SRLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tOR\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 22: { /* DIVSR */
+ nds32_parse_type_4(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.rd),
+ &(instruction->info.sub_opc));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tDIVSR\t$r%d,$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.rd);
+ }
+ break;
+ case 23: { /* DIVR */
+ nds32_parse_type_4(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.rd),
+ &(instruction->info.sub_opc));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tDIVR\t$r%d,$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.rd);
+ }
+ break;
+ case 24: { /* SVA */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSVA\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 25: { /* SVS */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSVS\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 26: { /* CMOVZ */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tCMOVZ\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 27: { /* CMOVN */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tCMOVN\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 28: /* ADD_SRLI */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tADD_SRLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tADD\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 29: /* SUB_SRLI */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSUB_SRLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSUB\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 30: /* AND_SRLI */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tAND_SRLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tAND\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 31: /* XOR_SRLI */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ instruction->info.imm = (instruction->info.imm >> 5) & 0x1F;
+ if (instruction->info.imm)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tXOR_SRLI\t$r%d,$r%d,$r%d,%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb,
+ instruction->info.imm);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tXOR\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_alu_2(uint32_t opcode, uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ switch (opcode & 0x3F) {
+ case 0: /* MAX */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMAX\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 1: /* MIN */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMIN\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 2: /* AVE */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tAVE\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 3: /* ABS */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tAVE\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 4: { /* CLIPS */
+ uint8_t imm;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &(instruction->info.imm));
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tCLIPS\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 5: { /* CLIP */
+ uint8_t imm;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &(instruction->info.imm));
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tCLIP\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 6: /* CLO */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tCLO\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 7: /* CLZ */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tCLZ\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 8: { /* BSET */
+ uint8_t imm;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &(instruction->info.imm));
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBSET\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 9: { /* BCLR */
+ uint8_t imm;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &(instruction->info.imm));
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBCLR\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 10: { /* BTGL */
+ uint8_t imm;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &(instruction->info.imm));
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBTGL\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 11: { /* BTST */
+ uint8_t imm;
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &imm, &(instruction->info.imm));
+ instruction->info.imm = imm;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBTST\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ case 12: /* BSE */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBSE\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 13: /* BSP */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBSP\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 14: /* FFB */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tFFB\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 15: /* FFMISM */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tFFMISM\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 23: /* FFZMISM */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tFFZMISM\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 32: /* MFUSR */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_RESOURCE_ACCESS;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMFUSR\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt,
+ (instruction->info.imm >> 10) & 0x3FF);
+ break;
+ case 33: /* MTUSR */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_RESOURCE_ACCESS;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMTUSR\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt,
+ (instruction->info.imm >> 10) & 0x3FF);
+ break;
+ case 36: /* MUL */
+ nds32_parse_type_3(opcode, &(instruction->info.rt),
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMUL\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ break;
+ case 40: { /* MULTS64 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val,
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMULTS64\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 41: { /* MULT64 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val,
+ &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMULT64\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 42: { /* MADDS64 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMADDS64\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 43: { /* MADD64 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMADD64\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 44: { /* MSUBS64 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMSUBS64\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 45: { /* MSUB64 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMSUB64\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 46: { /* DIVS */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tDIVS\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 47: { /* DIV */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tDIV\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 49: { /* MULT32 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMULT32\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 51: { /* MADD32 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMADD32\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 53: { /* MSUB32 */
+ uint8_t dt_val;
+ nds32_parse_type_3(opcode, &dt_val, &(instruction->info.ra),
+ &(instruction->info.rb), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMSUB32\t$D%d,$r%d,$r%d",
+ address,
+ opcode, (dt_val >> 1) & 0x1, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_4_insn(struct nds32 *nds32, uint32_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ uint8_t opc_6;
+
+ opc_6 = instruction->info.opc_6;
+
+ switch (opc_6 & 0x7) {
+ case 0: /* ALU_1 */
+ nds32_parse_alu_1(opcode, address, instruction);
+ break;
+ case 1: /* ALU_2 */
+ nds32_parse_alu_2(opcode, address, instruction);
+ break;
+ case 2: /* MOVI */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ /* sign-extend */
+ instruction->info.imm = (instruction->info.imm << 12) >> 12;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMOVI\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 3: /* SETHI */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSETHI\t$r%d,0x%8.8" PRIx32,
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 4: /* JI */
+ nds32_parse_type_0(opcode, &(instruction->info.imm));
+ /* sign-extend */
+ instruction->info.imm = (instruction->info.imm << 8) >> 8;
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ if ((instruction->info.imm >> 24) & 0x1) { /* JAL */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tJAL\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ } else { /* J */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tJ\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ }
+ break;
+ case 5: { /* JREG */
+ int32_t imm;
+ nds32_parse_type_0(opcode, &imm);
+ instruction->info.rb = (imm >> 10) & 0x1F;
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ switch (imm & 0x1F) {
+ /* TODO */
+ case 0: /* JR */
+ if (imm & 0x20) { /* RET */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tRET\t$r%d",
+ address,
+ opcode, instruction->info.rb);
+ } else { /* JR */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tJR\t$r%d",
+ address,
+ opcode, instruction->info.rb);
+ }
+ break;
+ case 1: /* JRAL */
+ instruction->info.rt = (imm >> 20) & 0x1F;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tJRAL\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.rb);
+ break;
+ case 2: /* JRNEZ */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tJRNEZ\t$r%d",
+ address,
+ opcode, instruction->info.rb);
+ break;
+ case 3: /* JRALNEZ */
+ instruction->info.rt = (imm >> 20) & 0x1F;
+ if (instruction->info.rt == R30)
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tJRALNEZ\t$r%d",
+ address,
+ opcode, instruction->info.rb);
+ else
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32
+ "\tJRALNEZ\t$r%d,$r%d",
+ address,
+ opcode,
+ instruction->info.rt,
+ instruction->info.rb);
+ break;
+ }
+ }
+ break;
+ case 6: { /* BR1 */
+ int32_t imm;
+
+ nds32_parse_type_0(opcode, &imm);
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ if ((imm >> 14) & 0x1) { /* BNE */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ /* sign-extend */
+ instruction->info.imm = (instruction->info.imm << 18) >> 18;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBNE\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ } else { /* BEQ */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ /* sign-extend */
+ instruction->info.imm = (instruction->info.imm << 18) >> 18;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBEQ\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt,
+ instruction->info.ra,
+ instruction->info.imm);
+ }
+ }
+ break;
+ case 7: { /* BR2 */
+ int32_t imm;
+
+ nds32_parse_type_0(opcode, &imm);
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ switch ((imm >> 16) & 0xF) {
+ case 2: /* BEQZ */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 16) >> 16;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBEQZ\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 3: /* BNEZ */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 16) >> 16;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBNEZ\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 4: /* BGEZ */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 16) >> 16;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBGEZ\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 5: /* BLTZ */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 16) >> 16;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBLTZ\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 6: /* BGTZ */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 16) >> 16;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBGTZ\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 7: /* BLEZ */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 16) >> 16;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBLEZ\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 12: /* BGEZAL */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 16) >> 16;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBGEZAL\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 13: /* BLTZAL */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 16) >> 16;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBLTZAL\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ }
+ }
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_5_insn(struct nds32 *nds32, uint32_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ uint8_t opc_6;
+
+ opc_6 = instruction->info.opc_6;
+
+ switch (opc_6 & 0x7) {
+ case 0: /* ADDI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tADDI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 1: /* SUBRI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSUBRI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 2: /* ANDI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tANDI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 3: /* XORI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tXORI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 4: /* ORI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tORI\t$r%d,$r%d,0x%8.8" PRIx32,
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 6: /* SLTI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSLTI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 7: /* SLTSI */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->info.imm = (instruction->info.imm << 17) >> 17; /* sign-extend */
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSLTSI\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_6_insn(struct nds32 *nds32, uint32_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ uint8_t opc_6;
+
+ opc_6 = instruction->info.opc_6;
+
+ switch (opc_6 & 0x7) {
+ case 2: { /* MISC */
+ int32_t imm;
+ uint8_t sub_opc;
+
+ nds32_parse_type_0(opcode, &imm);
+
+ sub_opc = imm & 0x1F;
+ switch (sub_opc) {
+ case 0: /* STANDBY */
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSTANDBY\t#%d",
+ address,
+ opcode, (opcode >> 5) & 0x3);
+ break;
+ case 1: /* CCTL */
+ /* TODO */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tCCTL",
+ address,
+ opcode);
+ break;
+ case 2: /* MFSR */
+ nds32_parse_type_1(opcode, &(instruction->info.rt),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_RESOURCE_ACCESS;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMFSR\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt,
+ (instruction->info.imm >> 10) & 0x3FF);
+ break;
+ case 3: /* MTSR */
+ nds32_parse_type_1(opcode, &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_RESOURCE_ACCESS;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMTSR\t$r%d,#%d",
+ address,
+ opcode, instruction->info.ra,
+ (instruction->info.imm >> 10) & 0x3FF);
+ break;
+ case 4: /* IRET */
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tIRET",
+ address,
+ opcode);
+ break;
+ case 5: /* TRAP */
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tTRAP\t#%d",
+ address,
+ opcode, (imm >> 5) & 0x7FFF);
+ break;
+ case 6: /* TEQZ */
+ nds32_parse_type_1(opcode, &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tTEQZ\t$r%d,#%d",
+ address,
+ opcode, instruction->info.ra,
+ (instruction->info.imm >> 5) & 0x7FFF);
+ break;
+ case 7: /* TNEZ */
+ nds32_parse_type_1(opcode, &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tTNEZ\t$r%d,#%d",
+ address,
+ opcode, instruction->info.ra,
+ (instruction->info.imm >> 5) & 0x7FFF);
+ break;
+ case 8: /* DSB */
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tDSB",
+ address,
+ opcode);
+ break;
+ case 9: /* ISB */
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tISB",
+ address,
+ opcode);
+ break;
+ case 10: /* BREAK */
+ instruction->type = NDS32_INSN_MISC;
+ instruction->info.sub_opc = imm & 0x1F;
+ instruction->info.imm = (imm >> 5) & 0x7FFF;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tBREAK\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ break;
+ case 11: /* SYSCALL */
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tSYSCALL\t#%d",
+ address,
+ opcode, (imm >> 5) & 0x7FFF);
+ break;
+ case 12: /* MSYNC */
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tMSYNC\t#%d",
+ address,
+ opcode, (imm >> 5) & 0x7);
+ break;
+ case 13: /* ISYNC */
+ nds32_parse_type_1(opcode, &(instruction->info.ra),
+ &(instruction->info.imm));
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tISYNC\t$r%d",
+ address,
+ opcode, instruction->info.ra);
+ break;
+ case 14: /* TLBOP */
+ /* TODO */
+ nds32_parse_type_2(opcode, &(instruction->info.rt),
+ &(instruction->info.ra), &(instruction->info.imm));
+ instruction->type = NDS32_INSN_RESOURCE_ACCESS;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tTLBOP",
+ address,
+ opcode);
+ break;
+ }
+
+ break;
+ }
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static uint32_t field_mask[9] = {
+ 0x0,
+ 0x1,
+ 0x3,
+ 0x7,
+ 0xF,
+ 0x1F,
+ 0x3F,
+ 0x7F,
+ 0xFF,
+};
+
+static uint8_t nds32_extract_field_8u(uint16_t opcode, uint32_t start, uint32_t length)
+{
+ if (0 < length && length < 9)
+ return (opcode >> start) & field_mask[length];
+
+ return 0;
+}
+
+static int nds32_parse_group_0_insn_16(struct nds32 *nds32, uint16_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ switch ((opcode >> 10) & 0x7) {
+ case 0: /* MOV55 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 5);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tMOV55\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 1: /* MOVI55 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 5);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->info.imm = (instruction->info.imm << 27) >> 27;
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tMOVI55\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 2: /* ADD45, SUB45 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.rb = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ if (nds32_extract_field_8u(opcode, 9, 1) == 0) { /* ADD45 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tADD45\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.rb);
+ } else { /* SUB45 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSUB45\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.rb);
+ }
+
+ break;
+ case 3: /* ADDI45, SUBI45 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ if (nds32_extract_field_8u(opcode, 9, 1) == 0) { /* ADDI45 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tADDI45\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ } else { /* SUBI45 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSUBI45\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ }
+ break;
+ case 4: /* SRAI45, SRLI45 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ if (nds32_extract_field_8u(opcode, 9, 1) == 0) { /* SRAI45 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSRAI45\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ } else { /* SRLI45 */
+ if ((instruction->info.rt == 0) && (instruction->info.imm == 0)) {
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tNOP",
+ address,
+ opcode);
+ } else {
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSRLI45\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ }
+ }
+ break;
+ case 5:
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ if (nds32_extract_field_8u(opcode, 9, 1) == 0) { /* SLLI333 */
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3);
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSLLI333\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ } else {
+ instruction->info.sub_opc = nds32_extract_field_8u(opcode, 0, 3);
+ switch (instruction->info.sub_opc) {
+ case 0: /* ZEB33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tZEB33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 1: /* ZEH33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tZEH33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 2: /* SEB33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSEB33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 3: /* SEH33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSEH33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 4: /* XLSB33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tXLSB33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 5: /* XLLB33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tXLLB33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 6: /* BMSKI33 */
+ instruction->info.ra = 0;
+ instruction->info.imm = nds32_extract_field_8u(opcode, 3, 3);
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tBMSKI33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 7: /* FEXTI33 */
+ instruction->info.ra = 0;
+ instruction->info.imm = nds32_extract_field_8u(opcode, 3, 3);
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tFEXTI33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32
+ "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+ }
+ break;
+ case 6: /* ADD333, SUB333 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.rb = nds32_extract_field_8u(opcode, 0, 3);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ if (nds32_extract_field_8u(opcode, 9, 1) == 0) { /* ADD333 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tADD333\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ } else { /* SUB333 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSUB333\t$r%d,$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.rb);
+ }
+ break;
+ case 7: /* ADDI333, SUBI333 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ if (nds32_extract_field_8u(opcode, 9, 1) == 0) { /* ADDI333 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tADDI333\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ } else { /* SUBI333 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSUBI333\t$r%d,$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ }
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_1_insn_16(struct nds32 *nds32, uint16_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ switch ((opcode >> 9) & 0xF) {
+ case 0: /* LWI333 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3) << 2;
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tLWI333\t$r%d,[$r%d+(#%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 1: /* LWI333.BI */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3);
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tLWI333.BI\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm << 2);
+ break;
+ case 2: /* LHI333 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3) << 1;
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tLHI333\t$r%d,[$r%d+(#%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 3: /* LBI333 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3);
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tLBI333\t$r%d,[$r%d+(#%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 4: /* SWI333 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3) << 2;
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSWI333\t$r%d,[$r%d+(#%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 5: /* SWI333.BI */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3) << 2;
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSWI333.BI\t$r%d,[$r%d],#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 6: /* SHI333 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3) << 1;
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 2;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSHI333\t$r%d,[$r%d+(#%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 7: /* SBI333 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 3);
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 1;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSHI333\t$r%d,[$r%d+(#%d)]",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra,
+ instruction->info.imm);
+ break;
+ case 8: /* ADDRI36.SP */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 6) << 2;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tADDRI36.SP\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 9: /* LWI45.FE */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->info.imm -= 32;
+ instruction->info.imm <<= 2;
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, R8, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tLWI45.FE\t$r%d,[#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 10: /* LWI450 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tLWI450\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 11: /* SWI450 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.ra = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, instruction->info.ra,
+ &(instruction->access_start));
+ instruction->access_end = instruction->access_start + 4;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSWI450\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 12:
+ case 13:
+ case 14:
+ case 15: /* LWI37, SWI37 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 8, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 7) << 2;
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, R28, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ if (nds32_extract_field_8u(opcode, 7, 1) == 0) { /* LWI37 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tLWI37\t$r%d,[fp+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ } else { /* SWI37 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSWI37\t$r%d,[fp+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ }
+ break;
+ default: /* ERROR */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_2_insn_16(struct nds32 *nds32, uint16_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ switch ((opcode >> 11) & 0x3) {
+ case 0: /* BEQZ38 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 8, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 8);
+ instruction->info.imm = (instruction->info.imm << 24) >> 24;
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tBEQZ38\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 1: /* BNEZ38 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 8, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 8);
+ instruction->info.imm = (instruction->info.imm << 24) >> 24;
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tBNEZ38\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 2: /* BEQS38,J8 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 8, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 8);
+ instruction->info.imm = (instruction->info.imm << 24) >> 24;
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ if (instruction->info.rt == 5) { /* J8 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tJ8\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ } else { /* BEQS38 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tBEQS38\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ }
+ break;
+ case 3: /* BNES38, JR5, RET5, JRAL5 */
+ instruction->info.rt = nds32_extract_field_8u(opcode, 8, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 8);
+ instruction->info.imm = (instruction->info.imm << 24) >> 24;
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ if (instruction->info.rt == 5) {
+ instruction->info.imm = 0;
+ instruction->info.rb = nds32_extract_field_8u(opcode, 0, 5);
+ switch (nds32_extract_field_8u(opcode, 5, 3)) {
+ case 0: /* JR5 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tJR5\t$r%d",
+ address,
+ opcode, instruction->info.rb);
+ break;
+ case 1: /* JRAL5 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tJRAL5\t$r%d",
+ address,
+ opcode, instruction->info.rb);
+ break;
+ case 2: /* EX9.IT */
+ instruction->info.rb = 0;
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 5);
+ /* TODO: implement real instruction semantics */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tEX9.IT\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ break;
+ case 4: /* RET5 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tRET5\t$r%d",
+ address,
+ opcode, instruction->info.rb);
+ break;
+ case 5: /* ADD5.PC */
+ instruction->info.rt = 0;
+ instruction->info.rt = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tADD5.PC\t$r%d",
+ address,
+ opcode, instruction->info.rt);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32
+ "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+ } else { /* BNES38 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tBNES38\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ }
+ break;
+ }
+
+ return ERROR_OK;
+}
+
+static int nds32_parse_group_3_insn_16(struct nds32 *nds32, uint16_t opcode,
+ uint32_t address, struct nds32_instruction *instruction)
+{
+ switch ((opcode >> 11) & 0x3) {
+ case 0:
+ switch ((opcode >> 9) & 0x3) {
+ case 0: /* SLTS45 */
+ instruction->info.ra = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.rb = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSLTS45\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.ra, instruction->info.rb);
+ break;
+ case 1: /* SLT45 */
+ instruction->info.ra = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.rb = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSLT45\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.ra, instruction->info.rb);
+ break;
+ case 2: /* SLTSI45 */
+ instruction->info.ra = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSLTSI45\t$r%d,#%d",
+ address,
+ opcode, instruction->info.ra, instruction->info.imm);
+ break;
+ case 3: /* SLTI45 */
+ instruction->info.ra = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 5);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSLTI45\t$r%d,#%d",
+ address,
+ opcode, instruction->info.ra, instruction->info.imm);
+ break;
+ }
+ break;
+ case 1:
+ switch ((opcode >> 9) & 0x3) {
+ case 0:
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 8);
+ instruction->info.imm = (instruction->info.imm << 24) >> 24;
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ if (nds32_extract_field_8u(opcode, 8, 1) == 0) { /* BEQZS8 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tBEQZS8\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ } else { /* BNEZS8 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tBNEZS8\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ }
+ break;
+ case 1: /* BREAK16 */
+ if (((opcode >> 5) & 0xF) == 0) {
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tBREAK16\t#%d",
+ address,
+ opcode, opcode & 0x1F);
+ } else { /* EX9.IT */
+ instruction->type = NDS32_INSN_MISC;
+ /* TODO: implement real instruction semantics */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tEX9.IT\t#%d",
+ address,
+ opcode, opcode & 0x1FF);
+ }
+ break;
+ case 2: /* ADDI10S */
+ case 3:
+ instruction->info.imm = opcode & 0x3FF;
+ instruction->info.imm = (instruction->info.imm << 22) >> 22;
+ instruction->type = NDS32_INSN_DATA_PROC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tADDI10.SP\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ break;
+ }
+ break;
+ case 2:
+ instruction->info.rt = nds32_extract_field_8u(opcode, 8, 3);
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 7) << 2;
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ nds32_get_mapped_reg(nds32, R31, &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end = instruction->access_start + 4;
+ if (nds32_extract_field_8u(opcode, 7, 1) == 0) { /* LWI37.SP */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tLWI37.SP\t$r%d,[+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ } else { /* SWI37.SP */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tSWI37.SP\t$r%d,[+#%d]",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ }
+ break;
+ case 3:
+ switch ((opcode >> 9) & 0x3) {
+ case 0: /* IFCALL9 */
+ instruction->info.imm = opcode & 0x1FF;
+ instruction->type = NDS32_INSN_JUMP_BRANCH;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tIFCALL9\t#%d",
+ address,
+ opcode, instruction->info.imm);
+ break;
+ case 1: /* MOVPI45 */
+ instruction->info.imm = nds32_extract_field_8u(opcode, 0, 5) + 16;
+ instruction->info.rt = nds32_extract_field_8u(opcode, 5, 4);
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32 "\t\tMOVPI45\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.imm);
+ break;
+ case 2: /* PUSH25, POP25, MOVD44 */
+ switch ((opcode >> 7) & 0x3) {
+ case 0: /* PUSH25 */
+ {
+ uint8_t re;
+ uint8_t gpr_count;
+
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ instruction->info.imm =
+ nds32_extract_field_8u(opcode, 0, 5) << 3;
+ re = nds32_extract_field_8u(opcode, 5, 2);
+
+ if (re == 0)
+ re = 6;
+ else if (re == 1)
+ re = 8;
+ else if (re == 2)
+ re = 10;
+ else if (re == 3)
+ re = 14;
+
+ instruction->info.rd = re;
+ /* GPRs list: R6 ~ Re and fp, gp, lp */
+ gpr_count = 3 + (re - 5);
+
+ nds32_get_mapped_reg(nds32, R31,
+ &(instruction->access_end));
+ instruction->access_start =
+ instruction->access_end - (gpr_count * 4);
+
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tPUSH25\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rd,
+ instruction->info.imm);
+ }
+ break;
+ case 1: /* POP25 */
+ {
+ uint8_t re;
+ uint8_t gpr_count;
+
+ instruction->type = NDS32_INSN_LOAD_STORE;
+ instruction->info.imm =
+ nds32_extract_field_8u(opcode, 0, 5) << 3;
+ re = nds32_extract_field_8u(opcode, 5, 2);
+
+ if (re == 0)
+ re = 6;
+ else if (re == 1)
+ re = 8;
+ else if (re == 2)
+ re = 10;
+ else if (re == 3)
+ re = 14;
+
+ instruction->info.rd = re;
+ /* GPRs list: R6 ~ Re and fp, gp, lp */
+ gpr_count = 3 + (re - 5);
+
+ nds32_get_mapped_reg(nds32, R31,
+ &(instruction->access_start));
+ instruction->access_start += instruction->info.imm;
+ instruction->access_end =
+ instruction->access_start + (gpr_count * 4);
+
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tPOP25\t$r%d,#%d",
+ address,
+ opcode, instruction->info.rd,
+ instruction->info.imm);
+ }
+ break;
+ case 2: /* MOVD44 */
+ case 3:
+ instruction->info.ra =
+ nds32_extract_field_8u(opcode, 0, 4) * 2;
+ instruction->info.rt =
+ nds32_extract_field_8u(opcode, 4, 4) * 2;
+ instruction->type = NDS32_INSN_MISC;
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tMOVD44\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ }
+ break;
+ case 3: /* NEG33, NOT33, MUL33, XOR33, AND33, OR33 */
+ instruction->info.ra = nds32_extract_field_8u(opcode, 3, 3);
+ instruction->info.rt = nds32_extract_field_8u(opcode, 6, 3);
+ instruction->type = NDS32_INSN_DATA_PROC;
+ switch (opcode & 0x7) {
+ case 2: /* NEG33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tNEG33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 3: /* NOT33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tNOT33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 4: /* MUL33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tMUL33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 5: /* XOR33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tXOR33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 6: /* AND33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tAND33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ case 7: /* OR33 */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%4.4" PRIx32
+ "\t\tOR33\t$r%d,$r%d",
+ address,
+ opcode, instruction->info.rt, instruction->info.ra);
+ break;
+ }
+ break;
+ }
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+
+ return ERROR_OK;
+}
+
+int nds32_evaluate_opcode(struct nds32 *nds32, uint32_t opcode, uint32_t address,
+ struct nds32_instruction *instruction)
+{
+ int retval = ERROR_OK;
+
+ /* clear fields, to avoid confusion */
+ memset(instruction, 0, sizeof(struct nds32_instruction));
+
+ if (opcode >> 31) {
+ /* 16 bits instruction */
+ instruction->instruction_size = 2;
+ opcode = (opcode >> 16) & 0xFFFF;
+ instruction->opcode = opcode;
+
+ switch ((opcode >> 13) & 0x3) {
+ case 0:
+ retval = nds32_parse_group_0_insn_16(nds32, opcode, address, instruction);
+ break;
+ case 1:
+ retval = nds32_parse_group_1_insn_16(nds32, opcode, address, instruction);
+ break;
+ case 2:
+ retval = nds32_parse_group_2_insn_16(nds32, opcode, address, instruction);
+ break;
+ case 3:
+ retval = nds32_parse_group_3_insn_16(nds32, opcode, address, instruction);
+ break;
+ default:
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+ } else {
+ /* 32 bits instruction */
+ instruction->instruction_size = 4;
+ instruction->opcode = opcode;
+
+ uint8_t opc_6;
+ opc_6 = opcode >> 25;
+ instruction->info.opc_6 = opc_6;
+
+ switch ((opc_6 >> 3) & 0x7) {
+ case 0: /* LBI, LHI, LWI, LBI.bi, LHI.bi, LWI.bi */
+ retval = nds32_parse_group_0_insn(nds32, opcode, address, instruction);
+ break;
+ case 1: /* SBI, SHI, SWI, SBI.bi, SHI.bi, SWI.bi */
+ retval = nds32_parse_group_1_insn(nds32, opcode, address, instruction);
+ break;
+ case 2: /* LBSI, LHSI, DPREFI, LBSI.bi, LHSI.bi, LBGP */
+ retval = nds32_parse_group_2_insn(nds32, opcode, address, instruction);
+ break;
+ case 3: /* MEM, LSMW, HWGP, SBGP */
+ retval = nds32_parse_group_3_insn(nds32, opcode, address, instruction);
+ break;
+ case 4: /* ALU_1, ALU_2, MOVI, SETHI, JI, JREG, BR1, BR2 */
+ retval = nds32_parse_group_4_insn(nds32, opcode, address, instruction);
+ break;
+ case 5: /* ADDI, SUBRI, ANDI, XORI, ORI, SLTI, SLTSI */
+ retval = nds32_parse_group_5_insn(nds32, opcode, address, instruction);
+ break;
+ case 6: /* MISC */
+ retval = nds32_parse_group_6_insn(nds32, opcode, address, instruction);
+ break;
+ default: /* ERROR */
+ snprintf(instruction->text,
+ 128,
+ "0x%8.8" PRIx32 "\t0x%8.8" PRIx32 "\tUNDEFINED INSTRUCTION",
+ address,
+ opcode);
+ return ERROR_FAIL;
+ }
+ }
+
+ return retval;
+}