disas/microblaze: Split get_field_special

Extract the raw special index and a function to lookup a name.

Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
Reviewed-by: Pierrick Bouvier <pierrick.bouvier@linaro.org>
Message-Id: <20240412073346.458116-21-richard.henderson@linaro.org>
This commit is contained in:
Richard Henderson 2024-04-12 00:33:39 -07:00
parent b35ab133eb
commit a45e0b5290
1 changed files with 61 additions and 81 deletions

View File

@ -564,8 +564,6 @@ static const struct op_code_struct {
/* prefix for register names */ /* prefix for register names */
#define register_prefix "r" #define register_prefix "r"
static const char pvr_register_prefix[] = "rpvr";
/* #defines for valid immediate range */ /* #defines for valid immediate range */
#define MIN_IMM ((int) 0x80000000) #define MIN_IMM ((int) 0x80000000)
@ -580,6 +578,7 @@ static const char pvr_register_prefix[] = "rpvr";
#define PRIreg register_prefix "%ld" #define PRIreg register_prefix "%ld"
#define PRIrfsl register_prefix "fsl%ld" #define PRIrfsl register_prefix "fsl%ld"
#define PRIpvr register_prefix "pvr%d"
#define PRIimm "%d" #define PRIimm "%d"
#define get_field_rd(instr) ((instr & RD_MASK) >> RD_LOW) #define get_field_rd(instr) ((instr & RD_MASK) >> RD_LOW)
@ -593,83 +592,48 @@ static const char pvr_register_prefix[] = "rpvr";
#define get_int_field_imm(instr) ((instr & IMM_MASK) >> IMM_LOW) #define get_int_field_imm(instr) ((instr & IMM_MASK) >> IMM_LOW)
#define get_int_field_r1(instr) ((instr & RA_MASK) >> RA_LOW) #define get_int_field_r1(instr) ((instr & RA_MASK) >> RA_LOW)
/* static int get_field_special(long instr, const struct op_code_struct *op)
char *
get_field_special (instr)
long instr;
{
char tmpstr[25];
snprintf(tmpstr, sizeof(tmpstr), "%s%s", register_prefix,
(((instr & IMM_MASK) >> IMM_LOW) & REG_MSR_MASK) == 0 ? "pc" : "msr");
return(strdup(tmpstr));
}
*/
static char *
get_field_special(long instr, const struct op_code_struct *op)
{ {
char tmpstr[25]; return ((instr & IMM_MASK) >> IMM_LOW) ^ op->immval_mask;
char spr[6]; }
switch ( (((instr & IMM_MASK) >> IMM_LOW) ^ op->immval_mask) ) { /* Returns NULL for PVR registers, which should be rendered differently. */
static const char *get_special_name(int special)
case REG_MSR_MASK : {
strcpy(spr, "msr"); switch (special) {
break; case REG_MSR_MASK:
case REG_PC_MASK : return register_prefix "msr";
strcpy(spr, "pc"); case REG_PC_MASK:
break; return register_prefix "pc";
case REG_EAR_MASK : case REG_EAR_MASK:
strcpy(spr, "ear"); return register_prefix "ear";
break; case REG_ESR_MASK:
case REG_ESR_MASK : return register_prefix "esr";
strcpy(spr, "esr"); case REG_FSR_MASK:
break; return register_prefix "fsr";
case REG_FSR_MASK : case REG_BTR_MASK:
strcpy(spr, "fsr"); return register_prefix "btr";
break; case REG_EDR_MASK:
case REG_BTR_MASK : return register_prefix "edr";
strcpy(spr, "btr"); case REG_PID_MASK:
break; return register_prefix "pid";
case REG_EDR_MASK : case REG_ZPR_MASK:
strcpy(spr, "edr"); return register_prefix "zpr";
break; case REG_TLBX_MASK:
case REG_PID_MASK : return register_prefix "tlbx";
strcpy(spr, "pid"); case REG_TLBLO_MASK:
break; return register_prefix "tlblo";
case REG_ZPR_MASK : case REG_TLBHI_MASK:
strcpy(spr, "zpr"); return register_prefix "tlbhi";
break; case REG_TLBSX_MASK:
case REG_TLBX_MASK : return register_prefix "tlbsx";
strcpy(spr, "tlbx"); default:
break; if ((special & 0xE000) == REG_PVR_MASK) {
case REG_TLBLO_MASK : /* pvr register */
strcpy(spr, "tlblo"); return NULL;
break; }
case REG_TLBHI_MASK : return register_prefix "pc";
strcpy(spr, "tlbhi"); }
break;
case REG_TLBSX_MASK :
strcpy(spr, "tlbsx");
break;
default :
{
if ( ((((instr & IMM_MASK) >> IMM_LOW) ^ op->immval_mask) & 0xE000) == REG_PVR_MASK) {
snprintf(tmpstr, sizeof(tmpstr), "%s%u", pvr_register_prefix,
(unsigned short)(((instr & IMM_MASK) >> IMM_LOW) ^
op->immval_mask) ^ REG_PVR_MASK);
return(strdup(tmpstr));
} else {
strcpy(spr, "pc");
}
}
break;
}
snprintf(tmpstr, sizeof(tmpstr), "%s%s", register_prefix, spr);
return(strdup(tmpstr));
} }
static unsigned long static unsigned long
@ -739,6 +703,8 @@ print_insn_microblaze(bfd_vma memaddr, struct disassemble_info *info)
static bfd_vma prev_insn_addr = -1; /*init the prev insn addr */ static bfd_vma prev_insn_addr = -1; /*init the prev insn addr */
static int prev_insn_vma = -1; /*init the prev insn vma */ static int prev_insn_vma = -1; /*init the prev insn vma */
int curr_insn_vma = info->buffer_vma; int curr_insn_vma = info->buffer_vma;
int special;
const char *special_name;
info->bytes_per_chunk = 4; info->bytes_per_chunk = 4;
@ -799,12 +765,26 @@ print_insn_microblaze(bfd_vma memaddr, struct disassemble_info *info)
op->name, get_field_r1(inst), get_field_rfsl(inst)); op->name, get_field_r1(inst), get_field_rfsl(inst));
break; break;
case INST_TYPE_RD_SPECIAL: case INST_TYPE_RD_SPECIAL:
fprintf_func(stream, "%s\t" PRIreg ", %s", special = get_field_special(inst, op);
op->name, get_field_rd(inst), get_field_special(inst, op)); special_name = get_special_name(special);
if (special_name) {
fprintf_func(stream, "%s\t" PRIreg ", %s",
op->name, get_field_rd(inst), special_name);
} else {
fprintf_func(stream, "%s\t" PRIreg ", " PRIpvr,
op->name, get_field_rd(inst), special ^ REG_PVR_MASK);
}
break; break;
case INST_TYPE_SPECIAL_R1: case INST_TYPE_SPECIAL_R1:
fprintf_func(stream, "%s\t%s, " PRIreg, special = get_field_special(inst, op);
op->name, get_field_special(inst, op), get_field_r1(inst)); special_name = get_special_name(special);
if (special_name) {
fprintf_func(stream, "%s\t%s, " PRIreg,
op->name, special_name, get_field_r1(inst));
} else {
fprintf_func(stream, "%s\t" PRIpvr ", " PRIreg,
op->name, special ^ REG_PVR_MASK, get_field_r1(inst));
}
break; break;
case INST_TYPE_RD_R1: case INST_TYPE_RD_R1:
fprintf_func(stream, "%s\t" PRIreg ", " PRIreg, fprintf_func(stream, "%s\t" PRIreg ", " PRIreg,