/* Single instruction disassembler for the Visium. Copyright (C) 2002-2023 Free Software Foundation, Inc. This file is part of the GNU opcodes library. This library is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. It is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */ #include "sysdep.h" #include "disassemble.h" #include "opcode/visium.h" #include #include #include #include #include /* Maximum length of an instruction. */ #define MAXLEN 4 struct private { /* Points to first byte not fetched. */ bfd_byte *max_fetched; bfd_byte the_buffer[MAXLEN]; bfd_vma insn_start; jmp_buf bailout; }; /* Make sure that bytes from INFO->PRIVATE_DATA->BUFFER (inclusive) to ADDR (exclusive) are valid. Returns 1 for success, longjmps on error. */ #define FETCH_DATA(info, addr) \ ((addr) <= ((struct private *)(info->private_data))->max_fetched \ ? 1 : fetch_data ((info), (addr))) static int fetch_data (struct disassemble_info *info, bfd_byte * addr); static int fetch_data (struct disassemble_info *info, bfd_byte *addr) { int status; struct private *priv = (struct private *) info->private_data; bfd_vma start = priv->insn_start + (priv->max_fetched - priv->the_buffer); status = (*info->read_memory_func) (start, priv->max_fetched, addr - priv->max_fetched, info); if (status != 0) { (*info->memory_error_func) (status, start, info); longjmp (priv->bailout, 1); } else priv->max_fetched = addr; return 1; } static char *size_names[] = { "?", "b", "w", "?", "l", "?", "?", "?" }; static char *cc_names[] = { "fa", "eq", "cs", "os", "ns", "ne", "cc", "oc", "nc", "ge", "gt", "hi", "le", "ls", "lt", "tr" }; /* Disassemble non-storage relative instructions. */ static int disassem_class0 (disassemble_info *info, unsigned int ins) { int opcode = (ins >> 21) & 0x000f; if (ins & CLASS0_UNUSED_MASK) goto illegal_opcode; switch (opcode) { case 0: /* BRR instruction. */ { unsigned cbf = (ins >> 27) & 0x000f; int displacement = ((ins & 0xffff) ^ 0x8000) - 0x8000; if (ins == 0) (*info->fprintf_func) (info->stream, "nop"); else (*info->fprintf_func) (info->stream, "brr %s,%+d", cc_names[cbf], displacement); } break; case 1: /* Illegal opcode. */ goto illegal_opcode; break; case 2: /* Illegal opcode. */ goto illegal_opcode; break; case 3: /* Illegal opcode. */ goto illegal_opcode; break; case 4: /* Illegal opcode. */ goto illegal_opcode; break; case 5: /* Illegal opcode. */ goto illegal_opcode; break; case 6: /* Illegal opcode. */ goto illegal_opcode; break; case 7: /* Illegal opcode. */ goto illegal_opcode; break; case 8: /* Illegal opcode. */ goto illegal_opcode; break; case 9: /* Illegal opcode. */ goto illegal_opcode; break; case 10: /* Illegal opcode. */ goto illegal_opcode; break; case 11: /* Illegal opcode. */ goto illegal_opcode; break; case 12: /* Illegal opcode. */ goto illegal_opcode; break; case 13: /* Illegal opcode. */ goto illegal_opcode; break; case 14: /* Illegal opcode. */ goto illegal_opcode; break; case 15: /* Illegal opcode. */ goto illegal_opcode; break; } return 0; illegal_opcode: return -1; } /* Disassemble non-storage register class instructions. */ static int disassem_class1 (disassemble_info *info, unsigned int ins) { int opcode = (ins >> 21) & 0xf; int source_a = (ins >> 16) & 0x1f; int source_b = (ins >> 4) & 0x1f; int indx = (ins >> 10) & 0x1f; int size = ins & 0x7; if (ins & CLASS1_UNUSED_MASK) goto illegal_opcode; switch (opcode) { case 0: /* Stop. */ (*info->fprintf_func) (info->stream, "stop %d,r%d", indx, source_a); break; case 1: /* BMI - Block Move Indirect. */ if (ins != BMI) goto illegal_opcode; (*info->fprintf_func) (info->stream, "bmi r1,r2,r3"); break; case 2: /* Illegal opcode. */ goto illegal_opcode; break; case 3: /* BMD - Block Move Direct. */ if (ins != BMD) goto illegal_opcode; (*info->fprintf_func) (info->stream, "bmd r1,r2,r3"); break; case 4: /* DSI - Disable Interrupts. */ if (ins != DSI) goto illegal_opcode; (*info->fprintf_func) (info->stream, "dsi"); break; case 5: /* ENI - Enable Interrupts. */ if (ins != ENI) goto illegal_opcode; (*info->fprintf_func) (info->stream, "eni"); break; case 6: /* Illegal opcode (was EUT). */ goto illegal_opcode; break; case 7: /* RFI - Return from Interrupt. */ if (ins != RFI) goto illegal_opcode; (*info->fprintf_func) (info->stream, "rfi"); break; case 8: /* Illegal opcode. */ goto illegal_opcode; break; case 9: /* Illegal opcode. */ goto illegal_opcode; break; case 10: /* Illegal opcode. */ goto illegal_opcode; break; case 11: /* Illegal opcode. */ goto illegal_opcode; break; case 12: /* Illegal opcode. */ goto illegal_opcode; break; case 13: goto illegal_opcode; break; case 14: goto illegal_opcode; break; case 15: if (ins & EAM_SELECT_MASK) { /* Extension arithmetic module write */ int fp_ins = (ins >> 27) & 0xf; if (size != 4) goto illegal_opcode; if (ins & FP_SELECT_MASK) { /* Which floating point instructions don't need a fsrcB register. */ const int no_fsrcb[16] = { 1, 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 1, 0, 0, 0 }; if (no_fsrcb[fp_ins] && source_b) goto illegal_opcode; /* Check that none of the floating register register numbers is higher than 15. (If this is fload, then srcA is a general register. */ if (ins & ((1 << 14) | (1 << 8)) || (fp_ins && ins & (1 << 20))) goto illegal_opcode; switch (fp_ins) { case 0: (*info->fprintf_func) (info->stream, "fload f%d,r%d", indx, source_a); break; case 1: (*info->fprintf_func) (info->stream, "fadd f%d,f%d,f%d", indx, source_a, source_b); break; case 2: (*info->fprintf_func) (info->stream, "fsub f%d,f%d,f%d", indx, source_a, source_b); break; case 3: (*info->fprintf_func) (info->stream, "fmult f%d,f%d,f%d", indx, source_a, source_b); break; case 4: (*info->fprintf_func) (info->stream, "fdiv f%d,f%d,f%d", indx, source_a, source_b); break; case 5: (*info->fprintf_func) (info->stream, "fsqrt f%d,f%d", indx, source_a); break; case 6: (*info->fprintf_func) (info->stream, "fneg f%d,f%d", indx, source_a); break; case 7: (*info->fprintf_func) (info->stream, "fabs f%d,f%d", indx, source_a); break; case 8: (*info->fprintf_func) (info->stream, "ftoi f%d,f%d", indx, source_a); break; case 9: (*info->fprintf_func) (info->stream, "itof f%d,f%d", indx, source_a); break; case 12: (*info->fprintf_func) (info->stream, "fmove f%d,f%d", indx, source_a); break; default: (*info->fprintf_func) (info->stream, "fpinst %d,f%d,f%d,f%d", fp_ins, indx, source_a, source_b); break; } } else { /* Which EAM operations do not need a srcB register. */ const int no_srcb[32] = { 0, 0, 1, 1, 0, 1, 1, 1, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; if (no_srcb[indx] && source_b) goto illegal_opcode; if (fp_ins) goto illegal_opcode; switch (indx) { case 0: (*info->fprintf_func) (info->stream, "mults r%d,r%d", source_a, source_b); break; case 1: (*info->fprintf_func) (info->stream, "multu r%d,r%d", source_a, source_b); break; case 2: (*info->fprintf_func) (info->stream, "divs r%d", source_a); break; case 3: (*info->fprintf_func) (info->stream, "divu r%d", source_a); break; case 4: (*info->fprintf_func) (info->stream, "writemd r%d,r%d", source_a, source_b); break; case 5: (*info->fprintf_func) (info->stream, "writemdc r%d", source_a); break; case 6: (*info->fprintf_func) (info->stream, "divds r%d", source_a); break; case 7: (*info->fprintf_func) (info->stream, "divdu r%d", source_a); break; case 9: (*info->fprintf_func) (info->stream, "asrd r%d", source_a); break; case 10: (*info->fprintf_func) (info->stream, "lsrd r%d", source_a); break; case 11: (*info->fprintf_func) (info->stream, "asld r%d", source_a); break; default: (*info->fprintf_func) (info->stream, "eamwrite %d,r%d,r%d", indx, source_a, source_b); break; } } } else { /* WRITE - write to memory. */ (*info->fprintf_func) (info->stream, "write.%s %d(r%d),r%d", size_names[size], indx, source_a, source_b); } break; } return 0; illegal_opcode: return -1; } /* Disassemble storage immediate class instructions. */ static int disassem_class2 (disassemble_info *info, unsigned int ins) { int opcode = (ins >> 21) & 0xf; int source_a = (ins >> 16) & 0x1f; unsigned immediate = ins & 0x0000ffff; if (ins & CC_MASK) goto illegal_opcode; switch (opcode) { case 0: /* ADDI instruction. */ (*info->fprintf_func) (info->stream, "addi r%d,%d", source_a, immediate); break; case 1: /* Illegal opcode. */ goto illegal_opcode; break; case 2: /* SUBI instruction. */ (*info->fprintf_func) (info->stream, "subi r%d,%d", source_a, immediate); break; case 3: /* Illegal opcode. */ goto illegal_opcode; break; case 4: /* MOVIL instruction. */ (*info->fprintf_func) (info->stream, "movil r%d,0x%04X", source_a, immediate); break; case 5: /* MOVIU instruction. */ (*info->fprintf_func) (info->stream, "moviu r%d,0x%04X", source_a, immediate); break; case 6: /* MOVIQ instruction. */ (*info->fprintf_func) (info->stream, "moviq r%d,%u", source_a, immediate); break; case 7: /* Illegal opcode. */ goto illegal_opcode; break; case 8: /* WRTL instruction. */ if (source_a != 0) goto illegal_opcode; (*info->fprintf_func) (info->stream, "wrtl 0x%04X", immediate); break; case 9: /* WRTU instruction. */ if (source_a != 0) goto illegal_opcode; (*info->fprintf_func) (info->stream, "wrtu 0x%04X", immediate); break; case 10: /* Illegal opcode. */ goto illegal_opcode; break; case 11: /* Illegal opcode. */ goto illegal_opcode; break; case 12: /* Illegal opcode. */ goto illegal_opcode; break; case 13: /* Illegal opcode. */ goto illegal_opcode; break; case 14: /* Illegal opcode. */ goto illegal_opcode; break; case 15: /* Illegal opcode. */ goto illegal_opcode; break; } return 0; illegal_opcode: return -1; } /* Disassemble storage register class instructions. */ static int disassem_class3 (disassemble_info *info, unsigned int ins) { int opcode = (ins >> 21) & 0xf; int source_b = (ins >> 4) & 0x1f; int source_a = (ins >> 16) & 0x1f; int size = ins & 0x7; int dest = (ins >> 10) & 0x1f; /* Those instructions that don't have a srcB register. */ const int no_srcb[16] = { 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0 }; /* These are instructions which can take an immediate srcB value. */ const int srcb_immed[16] = { 0, 0, 0, 0, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 1 }; /* User opcodes should not provide a non-zero srcB register when none is required. Only a BRA or floating point instruction should have a non-zero condition code field. Only a WRITE or EAMWRITE (opcode 15) should select an EAM or floating point operation. Note that FP_SELECT_MASK is the same bit (bit 3) as the interrupt bit which distinguishes SYS1 from BRA and SYS2 from RFLAG. */ if ((no_srcb[opcode] && source_b) || (!srcb_immed[opcode] && ins & CLASS3_SOURCEB_IMMED) || (opcode != 12 && opcode != 15 && ins & CC_MASK) || (opcode != 15 && ins & (EAM_SELECT_MASK | FP_SELECT_MASK))) goto illegal_opcode; switch (opcode) { case 0: /* ADD instruction. */ (*info->fprintf_func) (info->stream, "add.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 1: /* ADC instruction. */ (*info->fprintf_func) (info->stream, "adc.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 2: /* SUB instruction. */ if (dest == 0) (*info->fprintf_func) (info->stream, "cmp.%s r%d,r%d", size_names[size], source_a, source_b); else (*info->fprintf_func) (info->stream, "sub.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 3: /* SUBC instruction. */ if (dest == 0) (*info->fprintf_func) (info->stream, "cmpc.%s r%d,r%d", size_names[size], source_a, source_b); else (*info->fprintf_func) (info->stream, "subc.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 4: /* EXTW instruction. */ if (size == 1) goto illegal_opcode; (*info->fprintf_func) (info->stream, "extw.%s r%d,r%d", size_names[size], dest, source_a); break; case 5: /* ASR instruction. */ if (ins & CLASS3_SOURCEB_IMMED) (*info->fprintf_func) (info->stream, "asr.%s r%d,r%d,%d", size_names[size], dest, source_a, source_b); else (*info->fprintf_func) (info->stream, "asr.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 6: /* LSR instruction. */ if (ins & CLASS3_SOURCEB_IMMED) (*info->fprintf_func) (info->stream, "lsr.%s r%d,r%d,%d", size_names[size], dest, source_a, source_b); else (*info->fprintf_func) (info->stream, "lsr.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 7: /* ASL instruction. */ if (ins & CLASS3_SOURCEB_IMMED) (*info->fprintf_func) (info->stream, "asl.%s r%d,r%d,%d", size_names[size], dest, source_a, source_b); else (*info->fprintf_func) (info->stream, "asl.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 8: /* XOR instruction. */ (*info->fprintf_func) (info->stream, "xor.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 9: /* OR instruction. */ if (source_b == 0) (*info->fprintf_func) (info->stream, "move.%s r%d,r%d", size_names[size], dest, source_a); else (*info->fprintf_func) (info->stream, "or.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 10: /* AND instruction. */ (*info->fprintf_func) (info->stream, "and.%s r%d,r%d,r%d", size_names[size], dest, source_a, source_b); break; case 11: /* NOT instruction. */ (*info->fprintf_func) (info->stream, "not.%s r%d,r%d", size_names[size], dest, source_a); break; case 12: /* BRA instruction. */ { unsigned cbf = (ins >> 27) & 0x000f; if (size != 4) goto illegal_opcode; (*info->fprintf_func) (info->stream, "bra %s,r%d,r%d", cc_names[cbf], source_a, dest); } break; case 13: /* RFLAG instruction. */ if (source_a || size != 4) goto illegal_opcode; (*info->fprintf_func) (info->stream, "rflag r%d", dest); break; case 14: /* EXTB instruction. */ (*info->fprintf_func) (info->stream, "extb.%s r%d,r%d", size_names[size], dest, source_a); break; case 15: if (!(ins & CLASS3_SOURCEB_IMMED)) goto illegal_opcode; if (ins & EAM_SELECT_MASK) { /* Extension arithmetic module read. */ int fp_ins = (ins >> 27) & 0xf; if (size != 4) goto illegal_opcode; if (ins & FP_SELECT_MASK) { /* Check fsrcA <= 15 and fsrcB <= 15. */ if (ins & ((1 << 20) | (1 << 8))) goto illegal_opcode; switch (fp_ins) { case 0: if (source_b) goto illegal_opcode; (*info->fprintf_func) (info->stream, "fstore r%d,f%d", dest, source_a); break; case 10: (*info->fprintf_func) (info->stream, "fcmp r%d,f%d,f%d", dest, source_a, source_b); break; case 11: (*info->fprintf_func) (info->stream, "fcmpe r%d,f%d,f%d", dest, source_a, source_b); break; default: (*info->fprintf_func) (info->stream, "fpuread %d,r%d,f%d,f%d", fp_ins, dest, source_a, source_b); break; } } else { if (fp_ins || source_a) goto illegal_opcode; switch (source_b) { case 0: (*info->fprintf_func) (info->stream, "readmda r%d", dest); break; case 1: (*info->fprintf_func) (info->stream, "readmdb r%d", dest); break; case 2: (*info->fprintf_func) (info->stream, "readmdc r%d", dest); break; default: (*info->fprintf_func) (info->stream, "eamread r%d,%d", dest, source_b); break; } } } else { if (ins & FP_SELECT_MASK) goto illegal_opcode; /* READ instruction. */ (*info->fprintf_func) (info->stream, "read.%s r%d,%d(r%d)", size_names[size], dest, source_b, source_a); } break; } return 0; illegal_opcode: return -1; } /* Print the visium instruction at address addr in debugged memory, on info->stream. Return length of the instruction, in bytes. */ int print_insn_visium (bfd_vma addr, disassemble_info *info) { unsigned ins; unsigned p1, p2; int ans; int i; /* Stuff copied from m68k-dis.c. */ struct private priv; bfd_byte *buffer = priv.the_buffer; info->private_data = &priv; priv.max_fetched = priv.the_buffer; priv.insn_start = addr; if (setjmp (priv.bailout) != 0) { /* Error return. */ return -1; } /* We do return this info. */ info->insn_info_valid = 1; /* Assume non branch insn. */ info->insn_type = dis_nonbranch; /* Assume no delay. */ info->branch_delay_insns = 0; /* Assume no target known. */ info->target = 0; /* Get 32-bit instruction word. */ FETCH_DATA (info, buffer + 4); ins = (unsigned) buffer[0] << 24; ins |= buffer[1] << 16; ins |= buffer[2] << 8; ins |= buffer[3]; ans = 0; p1 = buffer[0] ^ buffer[1] ^ buffer[2] ^ buffer[3]; p2 = 0; for (i = 0; i < 8; i++) { p2 += p1 & 1; p1 >>= 1; } /* Decode the instruction. */ if (p2 & 1) ans = -1; else { switch ((ins >> 25) & 0x3) { case 0: ans = disassem_class0 (info, ins); break; case 1: ans = disassem_class1 (info, ins); break; case 2: ans = disassem_class2 (info, ins); break; case 3: ans = disassem_class3 (info, ins); break; } } if (ans != 0) (*info->fprintf_func) (info->stream, "err"); /* Return number of bytes consumed (always 4 for the Visium). */ return 4; }