/* tilegx-dis.c. Disassembly routines for the TILE-Gx architecture. Copyright (C) 2011-2022 Free Software Foundation, Inc. This file is part of the GNU opcodes library. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */ #include "sysdep.h" #include <stddef.h> #include <assert.h> #include "bfd.h" #include "elf/tilegx.h" #include "elf-bfd.h" #include "disassemble.h" #include "opcode/tilegx.h" int print_insn_tilegx (bfd_vma memaddr, disassemble_info *info) { struct tilegx_decoded_instruction decoded[TILEGX_MAX_INSTRUCTIONS_PER_BUNDLE]; bfd_byte opbuf[TILEGX_BUNDLE_SIZE_IN_BYTES]; int status, i, num_instructions, num_printed; tilegx_mnemonic padding_mnemonic; status = (*info->read_memory_func) (memaddr, opbuf, TILEGX_BUNDLE_SIZE_IN_BYTES, info); if (status != 0) { (*info->memory_error_func) (status, memaddr, info); return -1; } info->bytes_per_line = TILEGX_BUNDLE_SIZE_IN_BYTES; info->bytes_per_chunk = TILEGX_BUNDLE_SIZE_IN_BYTES; info->octets_per_byte = 1; info->display_endian = BFD_ENDIAN_LITTLE; /* Parse the instructions in the bundle. */ num_instructions = parse_insn_tilegx (bfd_getl64 (opbuf), memaddr, decoded); /* Print the instructions in the bundle. */ info->fprintf_func (info->stream, "{ "); num_printed = 0; /* Determine which nop opcode is used for padding and should be skipped. */ padding_mnemonic = TILEGX_OPC_FNOP; for (i = 0; i < num_instructions; i++) { if (!decoded[i].opcode->can_bundle) { /* Instructions that cannot be bundled are padded out with nops, rather than fnops. Displaying them is always clutter. */ padding_mnemonic = TILEGX_OPC_NOP; break; } } for (i = 0; i < num_instructions; i++) { const struct tilegx_opcode *opcode = decoded[i].opcode; const char *name; int j; /* Do not print out fnops, unless everything is an fnop, in which case we will print out just the last one. */ if (opcode->mnemonic == padding_mnemonic && (num_printed > 0 || i + 1 < num_instructions)) continue; if (num_printed > 0) info->fprintf_func (info->stream, " ; "); ++num_printed; name = opcode->name; if (name == NULL) name = "<invalid>"; info->fprintf_func (info->stream, "%s", name); for (j = 0; j < opcode->num_operands; j++) { bfd_vma num; const struct tilegx_operand *op; const char *spr_name; if (j > 0) info->fprintf_func (info->stream, ","); info->fprintf_func (info->stream, " "); num = decoded[i].operand_values[j]; op = decoded[i].operands[j]; switch (op->type) { case TILEGX_OP_TYPE_REGISTER: info->fprintf_func (info->stream, "%s", tilegx_register_names[(int) num]); break; case TILEGX_OP_TYPE_SPR: spr_name = get_tilegx_spr_name (num); if (spr_name != NULL) info->fprintf_func (info->stream, "%s", spr_name); else info->fprintf_func (info->stream, "%d", (int)num); break; case TILEGX_OP_TYPE_IMMEDIATE: info->fprintf_func (info->stream, "%d", (int)num); break; case TILEGX_OP_TYPE_ADDRESS: info->print_address_func (num, info); break; default: abort (); } } } info->fprintf_func (info->stream, " }"); return TILEGX_BUNDLE_SIZE_IN_BYTES; }