diff options
author | Alan Modra <amodra@gmail.com> | 2019-12-20 10:56:29 +1030 |
---|---|---|
committer | Alan Modra <amodra@gmail.com> | 2019-12-20 13:36:06 +1030 |
commit | 000fe1a756e707624d387eeacbaa959a01f0d040 (patch) | |
tree | 0f606d22dd6404817490848b2719ba9c9ab2107a | |
parent | 9d78076ef8ef07890ad89c1122bdf49932a979a5 (diff) | |
download | gdb-000fe1a756e707624d387eeacbaa959a01f0d040.zip gdb-000fe1a756e707624d387eeacbaa959a01f0d040.tar.gz gdb-000fe1a756e707624d387eeacbaa959a01f0d040.tar.bz2 |
ubsan: m68hc1x: left shift of negative value
* m68hc11-dis.c (read_memory): Delete forward decls.
(print_indexed_operand, print_insn): Likewise.
(print_indexed_operand): Formatting. Don't rely on short being
exactly 16 bits, make sign extension explicit.
(print_insn): Likewise. Avoid signed overflow.
-rw-r--r-- | opcodes/ChangeLog | 8 | ||||
-rw-r--r-- | opcodes/m68hc11-dis.c | 178 |
2 files changed, 81 insertions, 105 deletions
diff --git a/opcodes/ChangeLog b/opcodes/ChangeLog index 49b94e3..36667ae 100644 --- a/opcodes/ChangeLog +++ b/opcodes/ChangeLog @@ -1,3 +1,11 @@ +2019-12-20 Alan Modra <amodra@gmail.com> + + * m68hc11-dis.c (read_memory): Delete forward decls. + (print_indexed_operand, print_insn): Likewise. + (print_indexed_operand): Formatting. Don't rely on short being + exactly 16 bits, make sign extension explicit. + (print_insn): Likewise. Avoid signed overflow. + 2019-12-19 Alan Modra <amodra@gmail.com> * vax-dis.c (print_insn_mode): Stop index mode recursion. diff --git a/opcodes/m68hc11-dis.c b/opcodes/m68hc11-dis.c index 97647b4..2a10481 100644 --- a/opcodes/m68hc11-dis.c +++ b/opcodes/m68hc11-dis.c @@ -45,11 +45,6 @@ static const char *const reg_dst_table[] = #define OP_PAGE_MASK (M6811_OP_PAGE2|M6811_OP_PAGE3|M6811_OP_PAGE4) -/* Prototypes for local functions. */ -static int read_memory (bfd_vma, bfd_byte *, int, struct disassemble_info *); -static int print_indexed_operand (bfd_vma, struct disassemble_info *, - int*, int, int, bfd_vma, int); -static int print_insn (bfd_vma, struct disassemble_info *, int); static int read_memory (bfd_vma memaddr, bfd_byte* buffer, int size, @@ -73,13 +68,13 @@ read_memory (bfd_vma memaddr, bfd_byte* buffer, int size, Returns the number of bytes read or -1 if failure. */ static int print_indexed_operand (bfd_vma memaddr, struct disassemble_info* info, - int* indirect, int mov_insn, int pc_offset, - bfd_vma endaddr, int arch) + int* indirect, int mov_insn, int pc_offset, + bfd_vma endaddr, int arch) { bfd_byte buffer[4]; int reg; int status; - short sval; + bfd_vma val; int pos = 1; if (indirect) @@ -95,23 +90,22 @@ print_indexed_operand (bfd_vma memaddr, struct disassemble_info* info, if ((buffer[0] & 0x20) == 0) { reg = (buffer[0] >> 6) & 3; - sval = (buffer[0] & 0x1f); - if (sval & 0x10) - sval |= 0xfff0; + val = ((buffer[0] & 0x1f) ^ 0x10) - 0x10; /* 68HC12 requires an adjustment for movb/movw pc relative modes. */ if (reg == PC_REGNUM && info->mach == bfd_mach_m6812 && mov_insn) - sval += pc_offset; + val += pc_offset; (*info->fprintf_func) (info->stream, "0x%x,%s", - (unsigned short) sval, reg_name[reg]); + (unsigned) val & 0xffff, reg_name[reg]); if (reg == PC_REGNUM) - { - (* info->fprintf_func) (info->stream, " {"); - if (info->symtab_size > 0) /* Avoid duplicate 0x from core binutils. */ - (*info->fprintf_func) (info->stream, "0x"); - (* info->print_address_func) (endaddr + sval, info); - (* info->fprintf_func) (info->stream, "}"); - } + { + (* info->fprintf_func) (info->stream, " {"); + /* Avoid duplicate 0x from core binutils. */ + if (info->symtab_size > 0) + (*info->fprintf_func) (info->stream, "0x"); + (* info->print_address_func) (endaddr + val, info); + (* info->fprintf_func) (info->stream, "}"); + } } /* Auto pre/post increment/decrement. */ @@ -120,88 +114,81 @@ print_indexed_operand (bfd_vma memaddr, struct disassemble_info* info, const char *mode; reg = (buffer[0] >> 6) & 3; - sval = (buffer[0] & 0x0f); - if (sval & 0x8) + val = buffer[0] & 7; + if (buffer[0] & 8) { - sval |= 0xfff0; - sval = -sval; + val = 8 - val; mode = "-"; } else { - sval = sval + 1; + val = val + 1; mode = "+"; } (*info->fprintf_func) (info->stream, "%d,%s%s%s", - (unsigned short) sval, - (buffer[0] & 0x10 ? "" : mode), - reg_name[reg], (buffer[0] & 0x10 ? mode : "")); + (unsigned) val, + buffer[0] & 0x10 ? "" : mode, + reg_name[reg], buffer[0] & 0x10 ? mode : ""); } /* [n,r] 16-bits offset indexed indirect. */ else if ((buffer[0] & 0x07) == 3) { if ((mov_insn) && (!(arch & cpu9s12x))) - { - (*info->fprintf_func) (info->stream, "<invalid op: 0x%x>", - buffer[0] & 0x0ff); - return 0; - } + { + (*info->fprintf_func) (info->stream, "<invalid op: 0x%x>", + buffer[0] & 0x0ff); + return 0; + } reg = (buffer[0] >> 3) & 0x03; status = read_memory (memaddr + pos, &buffer[0], 2, info); if (status != 0) - { - return status; - } + return status; pos += 2; - sval = ((buffer[0] << 8) | (buffer[1] & 0x0FF)); + val = (buffer[0] << 8) | buffer[1]; (*info->fprintf_func) (info->stream, "[0x%x,%s]", - sval & 0x0ffff, reg_name[reg]); + (unsigned) val & 0xffff, reg_name[reg]); if (indirect) - *indirect = 1; + *indirect = 1; } /* n,r with 9 and 16 bit signed constant. */ else if ((buffer[0] & 0x4) == 0) { if ((mov_insn) && (!(arch & cpu9s12x))) - { - (*info->fprintf_func) (info->stream, "<invalid op: 0x%x>", - buffer[0] & 0x0ff); - return 0; - } + { + (*info->fprintf_func) (info->stream, "<invalid op: 0x%x>", + buffer[0] & 0x0ff); + return 0; + } reg = (buffer[0] >> 3) & 0x03; status = read_memory (memaddr + pos, &buffer[1], (buffer[0] & 0x2 ? 2 : 1), info); if (status != 0) - { - return status; - } + return status; + if (buffer[0] & 2) { - sval = ((buffer[1] << 8) | (buffer[2] & 0x0FF)); - sval &= 0x0FFFF; + val = (((buffer[1] << 8) | buffer[2]) ^ 0x8000) - 0x8000; pos += 2; - endaddr += 2; + endaddr += 2; } else { - sval = buffer[1] & 0x00ff; - if (buffer[0] & 0x01) - sval |= 0xff00; + val = buffer[1] - ((buffer[0] & 1) << 8); pos++; - endaddr++; + endaddr++; } (*info->fprintf_func) (info->stream, "0x%x,%s", - (unsigned short) sval, reg_name[reg]); + (unsigned) val & 0xffff, reg_name[reg]); if (reg == PC_REGNUM) - { - (* info->fprintf_func) (info->stream, " {0x"); - (* info->print_address_func) (endaddr + sval, info); - (* info->fprintf_func) (info->stream, "}"); - } + { + (* info->fprintf_func) (info->stream, " {0x"); + (* info->print_address_func) (endaddr + val, info); + (* info->fprintf_func) (info->stream, "}"); + } } else { @@ -220,8 +207,8 @@ print_indexed_operand (bfd_vma memaddr, struct disassemble_info* info, case 3: default: (*info->fprintf_func) (info->stream, "[D,%s]", reg_name[reg]); - if (indirect) - *indirect = 1; + if (indirect) + *indirect = 1; break; } } @@ -238,12 +225,11 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) bfd_byte buffer[4]; unsigned int code; long format, pos, i; - short sval; + bfd_vma val; const struct m68hc11_opcode *opcode; if (arch & cpuxgate) { - int val; /* Get two bytes as all XGATE instructions are 16bit. */ status = read_memory (memaddr, buffer, 2, info); if (status != 0) @@ -295,17 +281,14 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) else if (format & M68XG_OP_REL9) { (*info->fprintf_func) (info->stream, " 0x"); - val = (buffer[0] & 0x1) ? buffer[1] | 0xFFFFFF00 : buffer[1]; + val = buffer[1] - ((buffer[0] & 1) << 8); (*info->print_address_func) (memaddr + (val << 1) + 2, info); } else if (format & M68XG_OP_REL10) { (*info->fprintf_func) (info->stream, " 0x"); - val = (buffer[0] << 8) | (unsigned int) buffer[1]; - if (val & 0x200) - val |= 0xfffffc00; - else - val &= 0x000001ff; + val = (buffer[0] << 8) | buffer[1]; + val = ((val & 0x3ff) ^ 0x200) - 0x200; (*info->print_address_func) (memaddr + (val << 1) + 2, info); } else if ((code & 0x00ff) == 0x00f8) @@ -639,13 +622,11 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) (*info->fprintf_func) (info->stream, "%s,", reg_src_table[buffer[0] & 0x07]); - sval = buffer[1] & 0x0ff; - if (buffer[0] & 0x10) - sval |= 0xff00; + val = buffer[1] - ((buffer[0] & 0x10) << 4); pos += 2; (*info->fprintf_func) (info->stream, "0x"); - (*info->print_address_func) (memaddr + pos + sval, info); + (*info->print_address_func) (memaddr + pos + val, info); format &= ~(M6812_OP_REG | M6811_OP_JUMP_REL); } else if (format & (M6812_OP_REG | M6812_OP_REG_2)) @@ -662,7 +643,6 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) if (format & (M6811_OP_IMM16 | M6811_OP_IND16)) { - int val; bfd_vma addr; unsigned page = 0; @@ -676,9 +656,7 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) offset = 0; pos += 2; - val = ((buffer[0] << 8) | (buffer[1] & 0x0FF)); - val &= 0x0FFFF; - addr = val; + addr = val = (buffer[0] << 8) | buffer[1]; pc_dst_offset = 2; if (format & M6812_OP_PAGE) { @@ -686,16 +664,15 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) if (status != 0) return status; - page = (unsigned) buffer[0]; + page = buffer[0]; if (addr >= M68HC12_BANK_BASE && addr < 0x0c000) - addr = ((val - M68HC12_BANK_BASE) - | (page << M68HC12_BANK_SHIFT)) - + M68HC12_BANK_VIRT; + addr = (val - M68HC12_BANK_BASE + (page << M68HC12_BANK_SHIFT) + + M68HC12_BANK_VIRT); } else if ((arch & cpu6812) && addr >= M68HC12_BANK_BASE && addr < 0x0c000) { - int cur_page; + unsigned cur_page; bfd_vma vaddr; if (memaddr >= M68HC12_BANK_VIRT) @@ -704,9 +681,8 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) else cur_page = 0; - vaddr = ((addr - M68HC12_BANK_BASE) - + (cur_page << M68HC12_BANK_SHIFT)) - + M68HC12_BANK_VIRT; + vaddr = (addr - M68HC12_BANK_BASE + + (cur_page << M68HC12_BANK_SHIFT)) + M68HC12_BANK_VIRT; if (!info->symbol_at_address_func (addr, info) && info->symbol_at_address_func (vaddr, info)) addr = vaddr; @@ -721,14 +697,16 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) format &= ~M6811_OP_IND16; } - if (info->symtab_size > 0) /* Avoid duplicate 0x from core binutils. */ + /* Avoid duplicate 0x from core binutils. */ + if (info->symtab_size > 0) (*info->fprintf_func) (info->stream, "0x"); (*info->print_address_func) (addr, info); if (format & M6812_OP_PAGE) { (* info->fprintf_func) (info->stream, " {"); - if (info->symtab_size > 0) /* Avoid duplicate 0x from core binutils. */ + /* Avoid duplicate 0x from core binutils. */ + if (info->symtab_size > 0) (*info->fprintf_func) (info->stream, "0x"); (* info->print_address_func) (val, info); (* info->fprintf_func) (info->stream, ", 0x%x}", page); @@ -750,8 +728,6 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) if (format & M6812_OP_IND16_P2) { - int val; - (*info->fprintf_func) (info->stream, ", "); status = read_memory (memaddr + pos + offset, &buffer[0], 2, info); @@ -760,9 +736,9 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) pos += 2; - val = ((buffer[0] << 8) | (buffer[1] & 0x0FF)); - val &= 0x0FFFF; - if (info->symtab_size > 0) /* Avoid duplicate 0x from core binutils. */ + val = (buffer[0] << 8) | buffer[1]; + /* Avoid duplicate 0x from core binutils. */ + if (info->symtab_size > 0) (*info->fprintf_func) (info->stream, "0x"); (*info->print_address_func) (val, info); } @@ -784,30 +760,24 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) } if (format & M6811_OP_JUMP_REL) { - int val; - status = read_memory (memaddr + pos, &buffer[0], 1, info); if (status != 0) return status; (*info->fprintf_func) (info->stream, "0x"); pos++; - val = (buffer[0] & 0x80) ? buffer[0] | 0xFFFFFF00 : buffer[0]; + val = (buffer[0] ^ 0x80) - 0x80; (*info->print_address_func) (memaddr + pos + val, info); format &= ~M6811_OP_JUMP_REL; } else if (format & M6812_OP_JUMP_REL16) { - int val; - status = read_memory (memaddr + pos, &buffer[0], 2, info); if (status != 0) return status; pos += 2; - val = ((buffer[0] << 8) | (buffer[1] & 0x0FF)); - if (val & 0x8000) - val |= 0xffff0000; + val = (((buffer[0] << 8) | buffer[1]) ^ 0x8000) - 0x8000; (*info->fprintf_func) (info->stream, "0x"); (*info->print_address_func) (memaddr + pos + val, info); @@ -816,16 +786,14 @@ print_insn (bfd_vma memaddr, struct disassemble_info* info, int arch) if (format & M6812_OP_PAGE) { - int val; - status = read_memory (memaddr + pos + offset, &buffer[0], 1, info); if (status != 0) return status; pos += 1; - val = buffer[0] & 0x0ff; - (*info->fprintf_func) (info->stream, ", 0x%x", val); + val = buffer[0]; + (*info->fprintf_func) (info->stream, ", 0x%x", (unsigned) val); } #ifdef DEBUG |