mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-11-25 11:04:18 +08:00
2480b6fa94
The arc fix in create_map avoiding signed overflow by casting an unsigned char to unsigned int before shifting, shows one of the dangers of blinding doing that. The problem in this case was that the variable storing the value, newAuxRegister->address, was a long. Using the unsigned cast meant that the 32-bit value was zero extended when long is 64 bits. Previously we had a sign extension. Net result was that comparisons in arcExtMap_auxRegName didn't match. Of course, I could have cast the 32-bit unsigned value back to signed before storing in a long, but it's neater to just use an unsigned int for the address. opcodes/ * alpha-opc.c (OP): Avoid signed overflow. * arm-dis.c (print_insn): Likewise. * mcore-dis.c (print_insn_mcore): Likewise. * pj-dis.c (get_int): Likewise. * ppc-opc.c (EBD15, EBD15BI): Likewise. * score7-dis.c (s7_print_insn): Likewise. * tic30-dis.c (print_insn_tic30): Likewise. * v850-opc.c (insert_SELID): Likewise. * vax-dis.c (print_insn_vax): Likewise. * arc-ext.c (create_map): Likewise. (struct ExtAuxRegister): Make "address" field unsigned int. (arcExtMap_auxRegName): Pass unsigned address. (dump_ARC_extmap): Adjust. * arc-ext.h (arcExtMap_auxRegName): Update prototype.
178 lines
4.3 KiB
C
178 lines
4.3 KiB
C
/* pj-dis.c -- Disassemble picoJava instructions.
|
|
Copyright (C) 1999-2019 Free Software Foundation, Inc.
|
|
Contributed by Steve Chamberlain, of Transmeta (sac@pobox.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 program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
|
|
MA 02110-1301, USA. */
|
|
|
|
#include "sysdep.h"
|
|
#include <stdio.h>
|
|
#include "opcode/pj.h"
|
|
#include "disassemble.h"
|
|
|
|
extern const pj_opc_info_t pj_opc_info[512];
|
|
|
|
static int
|
|
get_int (bfd_vma memaddr, int *iptr, struct disassemble_info *info)
|
|
{
|
|
unsigned char ival[4];
|
|
int status = info->read_memory_func (memaddr, ival, 4, info);
|
|
|
|
*iptr = (((unsigned) ival[0] << 24)
|
|
| (ival[1] << 16)
|
|
| (ival[2] << 8)
|
|
| (ival[3] << 0));
|
|
|
|
return status;
|
|
}
|
|
|
|
int
|
|
print_insn_pj (bfd_vma addr, struct disassemble_info *info)
|
|
{
|
|
fprintf_ftype fprintf_fn = info->fprintf_func;
|
|
void *stream = info->stream;
|
|
unsigned char opcode;
|
|
int status;
|
|
|
|
if ((status = info->read_memory_func (addr, &opcode, 1, info)))
|
|
goto fail;
|
|
|
|
if (opcode == 0xff)
|
|
{
|
|
unsigned char byte_2;
|
|
|
|
if ((status = info->read_memory_func (addr + 1, &byte_2, 1, info)))
|
|
goto fail;
|
|
fprintf_fn (stream, "%s\t", pj_opc_info[opcode + byte_2].u.name);
|
|
return 2;
|
|
}
|
|
else
|
|
{
|
|
char *sep = "\t";
|
|
int insn_start = addr;
|
|
const pj_opc_info_t *op = &pj_opc_info[opcode];
|
|
int a;
|
|
|
|
addr++;
|
|
fprintf_fn (stream, "%s", op->u.name);
|
|
|
|
/* The tableswitch instruction is followed by the default
|
|
address, low value, high value and the destinations. */
|
|
|
|
if (strcmp (op->u.name, "tableswitch") == 0)
|
|
{
|
|
int lowval;
|
|
int highval;
|
|
int val;
|
|
|
|
addr = (addr + 3) & ~3;
|
|
if ((status = get_int (addr, &val, info)))
|
|
goto fail;
|
|
|
|
fprintf_fn (stream, " default: ");
|
|
(*info->print_address_func) (val + insn_start, info);
|
|
addr += 4;
|
|
|
|
if ((status = get_int (addr, &lowval, info)))
|
|
goto fail;
|
|
addr += 4;
|
|
|
|
if ((status = get_int (addr, &highval, info)))
|
|
goto fail;
|
|
addr += 4;
|
|
|
|
while (lowval <= highval)
|
|
{
|
|
if ((status = get_int (addr, &val, info)))
|
|
goto fail;
|
|
fprintf_fn (stream, " %d:[", lowval);
|
|
(*info->print_address_func) (val + insn_start, info);
|
|
fprintf_fn (stream, " ]");
|
|
addr += 4;
|
|
lowval++;
|
|
}
|
|
return addr - insn_start;
|
|
}
|
|
|
|
/* The lookupswitch instruction is followed by the default
|
|
address, element count and pairs of values and
|
|
addresses. */
|
|
if (strcmp (op->u.name, "lookupswitch") == 0)
|
|
{
|
|
int count;
|
|
int val;
|
|
|
|
addr = (addr + 3) & ~3;
|
|
if ((status = get_int (addr, &val, info)))
|
|
goto fail;
|
|
addr += 4;
|
|
|
|
fprintf_fn (stream, " default: ");
|
|
(*info->print_address_func) (val + insn_start, info);
|
|
|
|
if ((status = get_int (addr, &count, info)))
|
|
goto fail;
|
|
addr += 4;
|
|
|
|
while (count--)
|
|
{
|
|
if ((status = get_int (addr, &val, info)))
|
|
goto fail;
|
|
addr += 4;
|
|
fprintf_fn (stream, " %d:[", val);
|
|
|
|
if ((status = get_int (addr, &val, info)))
|
|
goto fail;
|
|
addr += 4;
|
|
|
|
(*info->print_address_func) (val + insn_start, info);
|
|
fprintf_fn (stream, " ]");
|
|
}
|
|
return addr - insn_start;
|
|
}
|
|
|
|
for (a = 0; op->arg[a]; a++)
|
|
{
|
|
unsigned char data[4];
|
|
int val = 0;
|
|
int i;
|
|
int size = ASIZE (op->arg[a]);
|
|
|
|
if ((status = info->read_memory_func (addr, data, size, info)))
|
|
goto fail;
|
|
|
|
val = (UNS (op->arg[0]) || ((data[0] & 0x80) == 0)) ? 0 : -1;
|
|
|
|
for (i = 0; i < size; i++)
|
|
val = (val << 8) | (data[i] & 0xff);
|
|
|
|
if (PCREL (op->arg[a]))
|
|
(*info->print_address_func) (val + insn_start, info);
|
|
else
|
|
fprintf_fn (stream, "%s%d", sep, val);
|
|
|
|
sep = ",";
|
|
addr += size;
|
|
}
|
|
return op->len;
|
|
}
|
|
|
|
fail:
|
|
info->memory_error_func (status, addr, info);
|
|
return -1;
|
|
}
|