| /* xtensa-dis.c.  Disassembly functions for Xtensa. | 
 |    Copyright (C) 2003-2025 Free Software Foundation, Inc. | 
 |    Contributed by Bob Wilson at Tensilica, Inc. (bwilson@tensilica.com) | 
 |  | 
 |    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 file; see the file COPYING.  If not, write to the | 
 |    Free Software Foundation, 51 Franklin Street - Fifth Floor, Boston, | 
 |    MA 02110-1301, USA.  */ | 
 |  | 
 | #include "sysdep.h" | 
 | #include <stdlib.h> | 
 | #include <stdio.h> | 
 | #include <sys/types.h> | 
 | #include <string.h> | 
 | #include "xtensa-isa.h" | 
 | #include "ansidecl.h" | 
 | #include "libiberty.h" | 
 | #include "bfd.h" | 
 | #include "elf/xtensa.h" | 
 | #include "disassemble.h" | 
 |  | 
 | #include <setjmp.h> | 
 |  | 
 | extern xtensa_isa xtensa_default_isa; | 
 |  | 
 | #ifndef MAX | 
 | #define MAX(a,b) (a > b ? a : b) | 
 | #endif | 
 |  | 
 | int show_raw_fields; | 
 |  | 
 | struct dis_private | 
 | { | 
 |   bfd_byte *byte_buf; | 
 |   OPCODES_SIGJMP_BUF bailout; | 
 |   /* Persistent fields, valid for last_section only.  */ | 
 |   asection *last_section; | 
 |   property_table_entry *insn_table_entries; | 
 |   int insn_table_entry_count; | 
 |   /* Cached property table search position.  */ | 
 |   bfd_vma insn_table_cur_addr; | 
 |   int insn_table_cur_idx; | 
 | }; | 
 |  | 
 | static void | 
 | xtensa_coalesce_insn_tables (struct dis_private *priv) | 
 | { | 
 |   const int mask = ~(XTENSA_PROP_DATA | XTENSA_PROP_NO_TRANSFORM); | 
 |   int count = priv->insn_table_entry_count; | 
 |   int i, j; | 
 |  | 
 |   /* Loop over all entries, combining adjacent ones that differ only in | 
 |      the flag bits XTENSA_PROP_DATA and XTENSA_PROP_NO_TRANSFORM.  */ | 
 |  | 
 |   for (i = j = 0; j < count; ++i) | 
 |     { | 
 |       property_table_entry *entry = priv->insn_table_entries + i; | 
 |  | 
 |       *entry = priv->insn_table_entries[j]; | 
 |  | 
 |       for (++j; j < count; ++j) | 
 | 	{ | 
 | 	  property_table_entry *next = priv->insn_table_entries + j; | 
 | 	  int fill = xtensa_compute_fill_extra_space (entry); | 
 | 	  int size = entry->size + fill; | 
 |  | 
 | 	  if (entry->address + size == next->address) | 
 | 	    { | 
 | 	      int entry_flags = entry->flags & mask; | 
 | 	      int next_flags = next->flags & mask; | 
 |  | 
 | 	      if (next_flags == entry_flags) | 
 | 		entry->size = next->address - entry->address + next->size; | 
 | 	      else | 
 | 		break; | 
 | 	    } | 
 | 	  else | 
 | 	    { | 
 | 	      break; | 
 | 	    } | 
 | 	} | 
 |     } | 
 |   priv->insn_table_entry_count = i; | 
 | } | 
 |  | 
 | static property_table_entry * | 
 | xtensa_find_table_entry (bfd_vma memaddr, struct disassemble_info *info) | 
 | { | 
 |   struct dis_private *priv = (struct dis_private *) info->private_data; | 
 |   int i; | 
 |  | 
 |   if (priv->insn_table_entries == NULL | 
 |       || priv->insn_table_entry_count < 0) | 
 |     return NULL; | 
 |  | 
 |   if (memaddr < priv->insn_table_cur_addr) | 
 |     priv->insn_table_cur_idx = 0; | 
 |  | 
 |   for (i = priv->insn_table_cur_idx; i < priv->insn_table_entry_count; ++i) | 
 |     { | 
 |       property_table_entry *block = priv->insn_table_entries + i; | 
 |  | 
 |       if (block->size != 0) | 
 | 	{ | 
 | 	  if ((memaddr >= block->address | 
 | 	       && memaddr < block->address + block->size) | 
 | 	      || memaddr < block->address) | 
 | 	    { | 
 | 	      priv->insn_table_cur_addr = memaddr; | 
 | 	      priv->insn_table_cur_idx = i; | 
 | 	      return block; | 
 | 	    } | 
 | 	} | 
 |     } | 
 |   return NULL; | 
 | } | 
 |  | 
 | /* Check whether an instruction crosses an instruction block boundary | 
 |    (according to property tables). | 
 |    If it does, return 0 (doesn't fit), else return 1.  */ | 
 |  | 
 | static int | 
 | xtensa_instruction_fits (bfd_vma memaddr, int size, | 
 | 			 property_table_entry *insn_block) | 
 | { | 
 |   unsigned max_size; | 
 |  | 
 |   /* If no property table info, assume it fits.  */ | 
 |   if (insn_block == NULL || size <= 0) | 
 |     return 1; | 
 |  | 
 |   /* If too high, limit nextstop by the next insn address.  */ | 
 |   if (insn_block->address > memaddr) | 
 |     { | 
 |       /* memaddr is not in an instruction block, but is followed by one.  */ | 
 |       max_size = insn_block->address - memaddr; | 
 |     } | 
 |   else | 
 |     { | 
 |       /* memaddr is in an instruction block, go no further than the end.  */ | 
 |       max_size = insn_block->address + insn_block->size - memaddr; | 
 |     } | 
 |  | 
 |   /* Crossing a boundary, doesn't "fit".  */ | 
 |   if ((unsigned)size > max_size) | 
 |     return 0; | 
 |   return 1; | 
 | } | 
 |  | 
 | static int | 
 | fetch_data (struct disassemble_info *info, bfd_vma memaddr) | 
 | { | 
 |   int length, status = 0; | 
 |   struct dis_private *priv = (struct dis_private *) info->private_data; | 
 |   int insn_size = xtensa_isa_maxlength (xtensa_default_isa); | 
 |  | 
 |   insn_size = MAX (insn_size, 4); | 
 |  | 
 |   /* Read the maximum instruction size, padding with zeros if we go past | 
 |      the end of the text section.  This code will automatically adjust | 
 |      length when we hit the end of the buffer.  */ | 
 |  | 
 |   memset (priv->byte_buf, 0, insn_size); | 
 |   for (length = insn_size; length > 0; length--) | 
 |     { | 
 |       status = (*info->read_memory_func) (memaddr, priv->byte_buf, length, | 
 | 					  info); | 
 |       if (status == 0) | 
 | 	return length; | 
 |     } | 
 |   (*info->memory_error_func) (status, memaddr, info); | 
 |   OPCODES_SIGLONGJMP (priv->bailout, 1); | 
 |   /*NOTREACHED*/ | 
 | } | 
 |  | 
 |  | 
 | static void | 
 | print_xtensa_operand (bfd_vma memaddr, | 
 | 		      struct disassemble_info *info, | 
 | 		      xtensa_opcode opc, | 
 | 		      int opnd, | 
 | 		      unsigned operand_val) | 
 | { | 
 |   xtensa_isa isa = xtensa_default_isa; | 
 |   int signed_operand_val, status; | 
 |   bfd_byte litbuf[4]; | 
 |  | 
 |   if (show_raw_fields) | 
 |     { | 
 |       if (operand_val < 0xa) | 
 | 	(*info->fprintf_func) (info->stream, "%u", operand_val); | 
 |       else | 
 | 	(*info->fprintf_func) (info->stream, "0x%x", operand_val); | 
 |       return; | 
 |     } | 
 |  | 
 |   (void) xtensa_operand_decode (isa, opc, opnd, &operand_val); | 
 |   signed_operand_val = (int) operand_val; | 
 |  | 
 |   if (xtensa_operand_is_register (isa, opc, opnd) == 0) | 
 |     { | 
 |       if (xtensa_operand_is_PCrelative (isa, opc, opnd) == 1) | 
 | 	{ | 
 | 	  (void) xtensa_operand_undo_reloc (isa, opc, opnd, | 
 | 					    &operand_val, memaddr); | 
 | 	  info->target = operand_val; | 
 | 	  (*info->print_address_func) (info->target, info); | 
 | 	  /*  Also display value loaded by L32R (but not if reloc exists, | 
 | 	      those tend to be wrong):  */ | 
 | 	  if ((info->flags & INSN_HAS_RELOC) == 0 | 
 | 	      && !strcmp ("l32r", xtensa_opcode_name (isa, opc))) | 
 | 	    status = (*info->read_memory_func) (operand_val, litbuf, 4, info); | 
 | 	  else | 
 | 	    status = -1; | 
 |  | 
 | 	  if (status == 0) | 
 | 	    { | 
 | 	      unsigned literal = bfd_get_bits (litbuf, 32, | 
 | 					       info->endian == BFD_ENDIAN_BIG); | 
 |  | 
 | 	      (*info->fprintf_func) (info->stream, " ("); | 
 | 	      (*info->print_address_func) (literal, info); | 
 | 	      (*info->fprintf_func) (info->stream, ")"); | 
 | 	    } | 
 | 	} | 
 |       else | 
 | 	{ | 
 | 	  if ((signed_operand_val > -256) && (signed_operand_val < 256)) | 
 | 	    (*info->fprintf_styled_func) (info->stream, dis_style_immediate, | 
 | 					  "%d", signed_operand_val); | 
 | 	  else | 
 | 	    (*info->fprintf_styled_func) (info->stream, dis_style_immediate, | 
 | 					  "0x%x", signed_operand_val); | 
 | 	} | 
 |     } | 
 |   else | 
 |     { | 
 |       int i = 1; | 
 |       xtensa_regfile opnd_rf = xtensa_operand_regfile (isa, opc, opnd); | 
 |       (*info->fprintf_styled_func) (info->stream, dis_style_register, | 
 | 				    "%s%u", | 
 | 				    xtensa_regfile_shortname (isa, opnd_rf), | 
 | 				    operand_val); | 
 |       while (i < xtensa_operand_num_regs (isa, opc, opnd)) | 
 | 	{ | 
 | 	  operand_val++; | 
 | 	  (*info->fprintf_styled_func) (info->stream, dis_style_register, | 
 | 					":%s%u", | 
 | 					xtensa_regfile_shortname (isa, opnd_rf), | 
 | 					operand_val); | 
 | 	  i++; | 
 | 	} | 
 |     } | 
 | } | 
 |  | 
 |  | 
 | /* Print the Xtensa instruction at address MEMADDR on info->stream. | 
 |    Returns length of the instruction in bytes.  */ | 
 |  | 
 | int | 
 | print_insn_xtensa (bfd_vma memaddr, struct disassemble_info *info) | 
 | { | 
 |   unsigned operand_val; | 
 |   int bytes_fetched, size, maxsize, i, n, noperands, nslots; | 
 |   xtensa_isa isa; | 
 |   xtensa_opcode opc; | 
 |   xtensa_format fmt; | 
 |   static struct dis_private priv; | 
 |   static bfd_byte *byte_buf = NULL; | 
 |   static xtensa_insnbuf insn_buffer = NULL; | 
 |   static xtensa_insnbuf slot_buffer = NULL; | 
 |   int first, first_slot, valid_insn; | 
 |   property_table_entry *insn_block; | 
 |   enum dis_insn_type insn_type; | 
 |   bfd_vma target; | 
 |  | 
 |   if (!xtensa_default_isa) | 
 |     xtensa_default_isa = xtensa_isa_init (0, 0); | 
 |  | 
 |   info->target = 0; | 
 |   maxsize = xtensa_isa_maxlength (xtensa_default_isa); | 
 |  | 
 |   /* Set bytes_per_line to control the amount of whitespace between the hex | 
 |      values and the opcode.  For Xtensa, we always print one "chunk" and we | 
 |      vary bytes_per_chunk to determine how many bytes to print.  (objdump | 
 |      would apparently prefer that we set bytes_per_chunk to 1 and vary | 
 |      bytes_per_line but that makes it hard to fit 64-bit instructions on | 
 |      an 80-column screen.)  The value of bytes_per_line here is not exactly | 
 |      right, because objdump adds an extra space for each chunk so that the | 
 |      amount of whitespace depends on the chunk size.  Oh well, it's good | 
 |      enough....  Note that we set the minimum size to 4 to accomodate | 
 |      literal pools.  */ | 
 |   info->bytes_per_line = MAX (maxsize, 4); | 
 |  | 
 |   /* Allocate buffers the first time through.  */ | 
 |   if (!insn_buffer) | 
 |     { | 
 |       insn_buffer = xtensa_insnbuf_alloc (xtensa_default_isa); | 
 |       slot_buffer = xtensa_insnbuf_alloc (xtensa_default_isa); | 
 |       byte_buf = (bfd_byte *) xmalloc (MAX (maxsize, 4)); | 
 |     } | 
 |  | 
 |   priv.byte_buf = byte_buf; | 
 |  | 
 |   info->private_data = (void *) &priv; | 
 |  | 
 |   /* Prepare instruction tables.  */ | 
 |  | 
 |   if (info->section != NULL) | 
 |     { | 
 |       asection *section = info->section; | 
 |  | 
 |       if (priv.last_section != section) | 
 | 	{ | 
 | 	  bfd *abfd = section->owner; | 
 |  | 
 | 	  if (priv.last_section != NULL) | 
 | 	    { | 
 | 	      /* Reset insn_table_entries.  */ | 
 | 	      priv.insn_table_entry_count = 0; | 
 | 	      free (priv.insn_table_entries); | 
 | 	      priv.insn_table_entries = NULL; | 
 | 	    } | 
 | 	  priv.last_section = section; | 
 |  | 
 | 	  /* Read insn_table_entries.  */ | 
 | 	  priv.insn_table_entry_count = | 
 | 	    xtensa_read_table_entries (abfd, section, | 
 | 				       &priv.insn_table_entries, | 
 | 				       XTENSA_PROP_SEC_NAME, false); | 
 | 	  if (priv.insn_table_entry_count == 0) | 
 | 	    { | 
 | 	      free (priv.insn_table_entries); | 
 | 	      priv.insn_table_entries = NULL; | 
 | 	      /* Backwards compatibility support.  */ | 
 | 	      priv.insn_table_entry_count = | 
 | 		xtensa_read_table_entries (abfd, section, | 
 | 					   &priv.insn_table_entries, | 
 | 					   XTENSA_INSN_SEC_NAME, false); | 
 | 	    } | 
 | 	  priv.insn_table_cur_idx = 0; | 
 | 	  xtensa_coalesce_insn_tables (&priv); | 
 | 	} | 
 |       /* Else nothing to do, same section as last time.  */ | 
 |     } | 
 |  | 
 |   if (OPCODES_SIGSETJMP (priv.bailout) != 0) | 
 |       /* Error return.  */ | 
 |       return -1; | 
 |  | 
 |   /* Fetch the maximum size instruction.  */ | 
 |   bytes_fetched = fetch_data (info, memaddr); | 
 |  | 
 |   insn_block = xtensa_find_table_entry (memaddr, info); | 
 |  | 
 |   /* Don't set "isa" before the setjmp to keep the compiler from griping.  */ | 
 |   isa = xtensa_default_isa; | 
 |   size = 0; | 
 |   nslots = 0; | 
 |   valid_insn = 0; | 
 |   fmt = 0; | 
 |   if (!insn_block || (insn_block->flags & XTENSA_PROP_INSN)) | 
 |     { | 
 |       /* Copy the bytes into the decode buffer.  */ | 
 |       memset (insn_buffer, 0, (xtensa_insnbuf_size (isa) * | 
 | 			       sizeof (xtensa_insnbuf_word))); | 
 |       xtensa_insnbuf_from_chars (isa, insn_buffer, priv.byte_buf, | 
 | 				 bytes_fetched); | 
 |  | 
 |       fmt = xtensa_format_decode (isa, insn_buffer); | 
 |       if (fmt != XTENSA_UNDEFINED | 
 | 	  && ((size = xtensa_format_length (isa, fmt)) <= bytes_fetched) | 
 | 	  && xtensa_instruction_fits (memaddr, size, insn_block)) | 
 | 	{ | 
 | 	  /* Make sure all the opcodes are valid.  */ | 
 | 	  valid_insn = 1; | 
 | 	  nslots = xtensa_format_num_slots (isa, fmt); | 
 | 	  for (n = 0; n < nslots; n++) | 
 | 	    { | 
 | 	      xtensa_format_get_slot (isa, fmt, n, insn_buffer, slot_buffer); | 
 | 	      if (xtensa_opcode_decode (isa, fmt, n, slot_buffer) | 
 | 		  == XTENSA_UNDEFINED) | 
 | 		{ | 
 | 		  valid_insn = 0; | 
 | 		  break; | 
 | 		} | 
 | 	    } | 
 | 	} | 
 |     } | 
 |  | 
 |   if (!valid_insn) | 
 |     { | 
 |       if (insn_block && (insn_block->flags & XTENSA_PROP_LITERAL) | 
 | 	  && (memaddr & 3) == 0 && bytes_fetched >= 4) | 
 | 	{ | 
 | 	  info->bytes_per_chunk = 4; | 
 | 	  return 4; | 
 | 	} | 
 |       else | 
 | 	{ | 
 | 	  (*info->fprintf_styled_func) (info->stream, | 
 | 					dis_style_assembler_directive, | 
 | 					".byte"); | 
 | 	  (*info->fprintf_func) (info->stream, "\t"); | 
 | 	  (*info->fprintf_styled_func) (info->stream, | 
 | 					dis_style_immediate, | 
 | 					"%#02x", priv.byte_buf[0]); | 
 | 	  return 1; | 
 | 	} | 
 |     } | 
 |  | 
 |   if (nslots > 1) | 
 |     (*info->fprintf_func) (info->stream, "{ "); | 
 |  | 
 |   insn_type = dis_nonbranch; | 
 |   target = 0; | 
 |   first_slot = 1; | 
 |   for (n = 0; n < nslots; n++) | 
 |     { | 
 |       int imm_pcrel = 0; | 
 |  | 
 |       if (first_slot) | 
 | 	first_slot = 0; | 
 |       else | 
 | 	(*info->fprintf_func) (info->stream, "; "); | 
 |  | 
 |       xtensa_format_get_slot (isa, fmt, n, insn_buffer, slot_buffer); | 
 |       opc = xtensa_opcode_decode (isa, fmt, n, slot_buffer); | 
 |       (*info->fprintf_styled_func) (info->stream, | 
 | 				    dis_style_mnemonic, "%s", | 
 | 				    xtensa_opcode_name (isa, opc)); | 
 |  | 
 |       if (xtensa_opcode_is_branch (isa, opc)) | 
 | 	info->insn_type = dis_condbranch; | 
 |       else if (xtensa_opcode_is_jump (isa, opc)) | 
 | 	info->insn_type = dis_branch; | 
 |       else if (xtensa_opcode_is_call (isa, opc)) | 
 | 	info->insn_type = dis_jsr; | 
 |       else | 
 | 	info->insn_type = dis_nonbranch; | 
 |  | 
 |       /* Print the operands (if any).  */ | 
 |       noperands = xtensa_opcode_num_operands (isa, opc); | 
 |       first = 1; | 
 |       for (i = 0; i < noperands; i++) | 
 | 	{ | 
 | 	  if (xtensa_operand_is_visible (isa, opc, i) == 0) | 
 | 	    continue; | 
 | 	  if (first) | 
 | 	    { | 
 | 	      (*info->fprintf_func) (info->stream, "\t"); | 
 | 	      first = 0; | 
 | 	    } | 
 | 	  else | 
 | 	    (*info->fprintf_func) (info->stream, ", "); | 
 | 	  (void) xtensa_operand_get_field (isa, opc, i, fmt, n, | 
 | 					   slot_buffer, &operand_val); | 
 |  | 
 | 	  print_xtensa_operand (memaddr, info, opc, i, operand_val); | 
 | 	  if (xtensa_operand_is_PCrelative (isa, opc, i)) | 
 | 	    ++imm_pcrel; | 
 | 	} | 
 |       if (!imm_pcrel) | 
 | 	info->insn_type = dis_nonbranch; | 
 |       if (info->insn_type != dis_nonbranch) | 
 | 	{ | 
 | 	  insn_type = info->insn_type; | 
 | 	  target = info->target; | 
 | 	} | 
 |     } | 
 |   info->insn_type = insn_type; | 
 |   info->target = target; | 
 |   info->insn_info_valid = 1; | 
 |  | 
 |   if (nslots > 1) | 
 |     (*info->fprintf_func) (info->stream, " }"); | 
 |  | 
 |   info->bytes_per_chunk = size; | 
 |   info->display_endian = info->endian; | 
 |  | 
 |   return size; | 
 | } | 
 |  |