aboutsummaryrefslogtreecommitdiff
path: root/opcodes
diff options
context:
space:
mode:
authorCharlie Jenkins <charlie@rivosinc.com>2025-01-07 13:35:44 -0800
committerNelson Chu <nelson@rivosinc.com>2025-01-09 10:09:16 +0800
commit6a04e8230707d8bbfaae8bd4c13f5c3018337971 (patch)
treee2960261e3ac23159ed0553988b70e8d37cb0d5f /opcodes
parent162502b593989171fafe5e6e8d92d5a626aa7568 (diff)
downloadgdb-6a04e8230707d8bbfaae8bd4c13f5c3018337971.zip
gdb-6a04e8230707d8bbfaae8bd4c13f5c3018337971.tar.gz
gdb-6a04e8230707d8bbfaae8bd4c13f5c3018337971.tar.bz2
RISC-V: Fix display of partial instructions
As of commit e43d8768d909 ("RISC-V: Fix disassemble fetch fail return value.") partial instructions are no longer displayed by objdump. While that commit fixed the behavior of print_insn_riscv() returning the arbitrary status value upon failure, it caused the behavior of dumping instructions to change. Allow partial instructions to be displayed once again and only return -1 if no part of the instruction was able to be displayed. Fixes: e43d8768d909 ("RISC-V: Fix disassemble fetch fail return value.") Signed-off-by: Charlie Jenkins <charlie@rivosinc.com> Acked-By: Andrew Burgess <aburgess@redhat.com>
Diffstat (limited to 'opcodes')
-rw-r--r--opcodes/riscv-dis.c53
1 files changed, 49 insertions, 4 deletions
diff --git a/opcodes/riscv-dis.c b/opcodes/riscv-dis.c
index 3be6e0c..367004d 100644
--- a/opcodes/riscv-dis.c
+++ b/opcodes/riscv-dis.c
@@ -1308,6 +1308,14 @@ riscv_disassemble_data (bfd_vma memaddr ATTRIBUTE_UNUSED,
(*info->fprintf_styled_func)
(info->stream, dis_style_immediate, "0x%04x", (unsigned) data);
break;
+ case 3:
+ info->bytes_per_line = 7;
+ (*info->fprintf_styled_func)
+ (info->stream, dis_style_assembler_directive, ".word");
+ (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
+ (*info->fprintf_styled_func)
+ (info->stream, dis_style_immediate, "0x%06x", (unsigned) data);
+ break;
case 4:
info->bytes_per_line = 8;
(*info->fprintf_styled_func)
@@ -1360,12 +1368,31 @@ riscv_init_disasm_info (struct disassemble_info *info)
return true;
}
+/* Fetch an instruction. If only a partial instruction is able to be fetched,
+ return the number of accessible bytes. */
+
+static bfd_vma
+fetch_insn (bfd_vma memaddr,
+ bfd_byte *packet,
+ bfd_vma dump_size,
+ struct disassemble_info *info,
+ volatile int *status)
+{
+ do
+ {
+ *status = (*info->read_memory_func) (memaddr, packet, dump_size, info);
+ }
+ while (*status != 0 && dump_size-- > 1);
+
+ return dump_size;
+}
+
int
print_insn_riscv (bfd_vma memaddr, struct disassemble_info *info)
{
bfd_byte packet[RISCV_MAX_INSN_LEN];
insn_t insn = 0;
- bfd_vma dump_size;
+ bfd_vma dump_size, bytes_fetched;
int status;
enum riscv_seg_mstate mstate;
int (*riscv_disassembler) (bfd_vma, insn_t, const bfd_byte *,
@@ -1398,24 +1425,42 @@ print_insn_riscv (bfd_vma memaddr, struct disassemble_info *info)
else
{
/* Get the first 2-bytes to check the lenghth of instruction. */
- status = (*info->read_memory_func) (memaddr, packet, 2, info);
+ bytes_fetched = fetch_insn (memaddr, packet, 2, info, &status);
if (status != 0)
{
(*info->memory_error_func) (status, memaddr, info);
return -1;
}
+ else if (bytes_fetched != 2)
+ {
+ /* Only the first byte was able to be read. Dump the partial
+ instruction. */
+ dump_size = bytes_fetched;
+ info->bytes_per_chunk = dump_size;
+ riscv_disassembler = riscv_disassemble_data;
+ goto print;
+ }
insn = (insn_t) bfd_getl16 (packet);
dump_size = riscv_insn_length (insn);
riscv_disassembler = riscv_disassemble_insn;
}
- /* Fetch the instruction to dump. */
- status = (*info->read_memory_func) (memaddr, packet, dump_size, info);
+ bytes_fetched = fetch_insn (memaddr, packet, dump_size, info, &status);
+
if (status != 0)
{
(*info->memory_error_func) (status, memaddr, info);
return -1;
}
+ else if (bytes_fetched != dump_size)
+ {
+ dump_size = bytes_fetched;
+ info->bytes_per_chunk = dump_size;
+ riscv_disassembler = riscv_disassemble_data;
+ }
+
+ print:
+
insn = (insn_t) bfd_get_bits (packet, dump_size * 8, false);
return (*riscv_disassembler) (memaddr, insn, packet, info);