/* s12z-decode.c -- Freescale S12Z disassembly Copyright (C) 2018-2025 Free Software Foundation, Inc. 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 #include #include #include #include "opcode/s12z.h" #include "bfd.h" #include "s12z-opc.h" typedef int (*insn_bytes_f) (struct mem_read_abstraction_base *); typedef int (*operands_f) (struct mem_read_abstraction_base *, int *n_operands, struct operand **operand); typedef enum optr (*discriminator_f) (struct mem_read_abstraction_base *, enum optr hint); enum OPR_MODE { OPR_IMMe4, OPR_REG, OPR_OFXYS, OPR_XY_PRE_INC, OPR_XY_POST_INC, OPR_XY_PRE_DEC, OPR_XY_POST_DEC, OPR_S_PRE_DEC, OPR_S_POST_INC, OPR_REG_DIRECT, OPR_REG_INDIRECT, OPR_IDX_DIRECT, OPR_IDX_INDIRECT, OPR_EXT1, OPR_IDX2_REG, OPR_IDX3_DIRECT, OPR_IDX3_INDIRECT, OPR_EXT18, OPR_IDX3_DIRECT_REG, OPR_EXT3_DIRECT, OPR_EXT3_INDIRECT }; struct opr_pb { uint8_t mask; uint8_t value; int n_operands; enum OPR_MODE mode; }; static const struct opr_pb opr_pb[] = { {0xF0, 0x70, 1, OPR_IMMe4}, {0xF8, 0xB8, 1, OPR_REG}, {0xC0, 0x40, 1, OPR_OFXYS}, {0xEF, 0xE3, 1, OPR_XY_PRE_INC}, {0xEF, 0xE7, 1, OPR_XY_POST_INC}, {0xEF, 0xC3, 1, OPR_XY_PRE_DEC}, {0xEF, 0xC7, 1, OPR_XY_POST_DEC}, {0xFF, 0xFB, 1, OPR_S_PRE_DEC}, {0xFF, 0xFF, 1, OPR_S_POST_INC}, {0xC8, 0x88, 1, OPR_REG_DIRECT}, {0xE8, 0xC8, 1, OPR_REG_INDIRECT}, {0xCE, 0xC0, 2, OPR_IDX_DIRECT}, {0xCE, 0xC4, 2, OPR_IDX_INDIRECT}, {0xC0, 0x00, 2, OPR_EXT1}, {0xC8, 0x80, 3, OPR_IDX2_REG}, {0xFA, 0xF8, 3, OPR_EXT18}, {0xCF, 0xC2, 4, OPR_IDX3_DIRECT}, {0xCF, 0xC6, 4, OPR_IDX3_INDIRECT}, {0xF8, 0xE8, 4, OPR_IDX3_DIRECT_REG}, {0xFF, 0xFA, 4, OPR_EXT3_DIRECT}, {0xFF, 0xFE, 4, OPR_EXT3_INDIRECT}, }; /* Return the number of bytes in a OPR operand, including the XB postbyte. It does not include any preceeding opcodes. */ static int x_opr_n_bytes (struct mem_read_abstraction_base *mra, int offset) { bfd_byte xb; int status = mra->read (mra, offset, 1, &xb); if (status < 0) return status; size_t i; for (i = 0; i < sizeof (opr_pb) / sizeof (opr_pb[0]); ++i) { const struct opr_pb *pb = opr_pb + i; if ((xb & pb->mask) == pb->value) { return pb->n_operands; } } return 1; } static int opr_n_bytes_p1 (struct mem_read_abstraction_base *mra) { int n = x_opr_n_bytes (mra, 0); if (n < 0) return n; return 1 + n; } static int opr_n_bytes2 (struct mem_read_abstraction_base *mra) { int s = x_opr_n_bytes (mra, 0); if (s < 0) return s; int n = x_opr_n_bytes (mra, s); if (n < 0) return n; return s + n + 1; } enum BB_MODE { BB_REG_REG_REG, BB_REG_REG_IMM, BB_REG_OPR_REG, BB_OPR_REG_REG, BB_REG_OPR_IMM, BB_OPR_REG_IMM }; struct opr_bb { uint8_t mask; uint8_t value; int n_operands; bool opr; enum BB_MODE mode; }; static const struct opr_bb bb_modes[] = { {0x60, 0x00, 2, false, BB_REG_REG_REG}, {0x60, 0x20, 3, false, BB_REG_REG_IMM}, {0x70, 0x40, 2, true, BB_REG_OPR_REG}, {0x70, 0x50, 2, true, BB_OPR_REG_REG}, {0x70, 0x60, 3, true, BB_REG_OPR_IMM}, {0x70, 0x70, 3, true, BB_OPR_REG_IMM} }; static int bfextins_n_bytes (struct mem_read_abstraction_base *mra) { bfd_byte bb; int status = mra->read (mra, 0, 1, &bb); if (status < 0) return status; size_t i; const struct opr_bb *bbs = 0; for (i = 0; i < sizeof (bb_modes) / sizeof (bb_modes[0]); ++i) { bbs = bb_modes + i; if ((bb & bbs->mask) == bbs->value) { break; } } int n = bbs->n_operands; if (bbs->opr) { int x = x_opr_n_bytes (mra, n - 1); if (x < 0) return x; n += x; } return n; } static int single (struct mem_read_abstraction_base *mra ATTRIBUTE_UNUSED) { return 1; } static int two (struct mem_read_abstraction_base *mra ATTRIBUTE_UNUSED) { return 2; } static int three (struct mem_read_abstraction_base *mra ATTRIBUTE_UNUSED) { return 3; } static int four (struct mem_read_abstraction_base *mra ATTRIBUTE_UNUSED) { return 4; } static int five (struct mem_read_abstraction_base *mra ATTRIBUTE_UNUSED) { return 5; } static int pcrel_15bit (struct mem_read_abstraction_base *mra) { bfd_byte byte; int status = mra->read (mra, 0, 1, &byte); if (status < 0) return status; return (byte & 0x80) ? 3 : 2; } static int xysp_reg_from_postbyte (uint8_t postbyte) { int reg = -1; switch ((postbyte & 0x30) >> 4) { case 0: reg = REG_X; break; case 1: reg = REG_Y; break; case 2: reg = REG_S; break; default: reg = REG_P; } return reg; } static struct operand * create_immediate_operand (int value) { struct immediate_operand *op = malloc (sizeof (*op)); if (op != NULL) { op->parent.cl = OPND_CL_IMMEDIATE; op->parent.osize = -1; op->value = value; } return (struct operand *) op; } static struct operand * create_bitfield_operand (int width, int offset) { struct bitfield_operand *op = malloc (sizeof (*op)); if (op != NULL) { op->parent.cl = OPND_CL_BIT_FIELD; op->parent.osize = -1; op->width = width; op->offset = offset; } return (struct operand *) op; } static struct operand * create_register_operand_with_size (int reg, short osize) { struct register_operand *op = malloc (sizeof (*op)); if (op != NULL) { op->parent.cl = OPND_CL_REGISTER; op->parent.osize = osize; op->reg = reg; } return (struct operand *) op; } static struct operand * create_register_operand (int reg) { return create_register_operand_with_size (reg, -1); } static struct operand * create_register_all_operand (void) { struct register_operand *op = malloc (sizeof (*op)); if (op != NULL) { op->parent.cl = OPND_CL_REGISTER_ALL; op->parent.osize = -1; } return (struct operand *) op; } static struct operand * create_register_all16_operand (void) { struct register_operand *op = malloc (sizeof (*op)); if (op != NULL) { op->parent.cl = OPND_CL_REGISTER_ALL16; op->parent.osize = -1; } return (struct operand *) op; } static struct operand * create_simple_memory_operand (bfd_vma addr, bfd_vma base, bool relative) { struct simple_memory_operand *op; assert (relative || base == 0); op = malloc (sizeof (*op)); if (op != NULL) { op->parent.cl = OPND_CL_SIMPLE_MEMORY; op->parent.osize = -1; op->addr = addr; op->base = base; op->relative = relative; } return (struct operand *) op; } static struct operand * create_memory_operand (bool indirect, int base, int n_regs, int reg0, int reg1) { struct memory_operand *op = malloc (sizeof (*op)); if (op != NULL) { op->parent.cl = OPND_CL_MEMORY; op->parent.osize = -1; op->indirect = indirect; op->base_offset = base; op->mutation = OPND_RM_NONE; op->n_regs = n_regs; op->regs[0] = reg0; op->regs[1] = reg1; } return (struct operand *) op; } static struct operand * create_memory_auto_operand (enum op_reg_mutation mutation, int reg) { struct memory_operand *op = malloc (sizeof (*op)); if (op != NULL) { op->parent.cl = OPND_CL_MEMORY; op->parent.osize = -1; op->indirect = false; op->base_offset = 0; op->mutation = mutation; op->n_regs = 1; op->regs[0] = reg; op->regs[1] = -1; } return (struct operand *) op; } static int z_ext24_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; uint8_t buffer[3]; int status = mra->read (mra, 0, 3, buffer); if (status < 0) return status; int i; uint32_t addr = 0; for (i = 0; i < 3; ++i) { addr <<= 8; addr |= buffer[i]; } op = create_simple_memory_operand (addr, 0, false); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int z_decode_signed_value (struct mem_read_abstraction_base *mra, int offset, short size, uint32_t *result) { assert (size >0); assert (size <= 4); bfd_byte buffer[4]; int status = mra->read (mra, offset, size, buffer); if (status < 0) return status; int i; uint32_t value = 0; for (i = 0; i < size; ++i) value = (value << 8) | buffer[i]; if (buffer[0] & 0x80) { /* Deal with negative values */ value -= 1u << (size * 4) << (size * 4); } *result = value; return 0; } static int decode_signed_value (struct mem_read_abstraction_base *mra, short size, uint32_t *result) { return z_decode_signed_value (mra, 0, size, result); } static int x_imm1 (struct mem_read_abstraction_base *mra, int offset, int *n_operands, struct operand **operand) { struct operand *op; bfd_byte byte; int status = mra->read (mra, offset, 1, &byte); if (status < 0) return status; op = create_immediate_operand (byte); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } /* An eight bit immediate operand. */ static int imm1_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { return x_imm1 (mra, 0, n_operands, operand); } static int trap_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { return x_imm1 (mra, -1, n_operands, operand); } static struct operand * x_opr_decode_with_size (struct mem_read_abstraction_base *mra, int offset, short osize) { bfd_byte postbyte; int status = mra->read (mra, offset, 1, &postbyte); if (status < 0) return NULL; offset++; enum OPR_MODE mode = -1; size_t i; for (i = 0; i < sizeof (opr_pb) / sizeof (opr_pb[0]); ++i) { const struct opr_pb *pb = opr_pb + i; if ((postbyte & pb->mask) == pb->value) { mode = pb->mode; break; } } struct operand *operand = NULL; switch (mode) { case OPR_IMMe4: { int n; uint8_t x = (postbyte & 0x0F); if (x == 0) n = -1; else n = x; operand = create_immediate_operand (n); break; } case OPR_REG: { uint8_t x = (postbyte & 0x07); operand = create_register_operand (x); break; } case OPR_OFXYS: { operand = create_memory_operand (false, postbyte & 0x0F, 1, xysp_reg_from_postbyte (postbyte), -1); break; } case OPR_REG_DIRECT: { operand = create_memory_operand (false, 0, 2, postbyte & 0x07, xysp_reg_from_postbyte (postbyte)); break; } case OPR_REG_INDIRECT: { operand = create_memory_operand (true, 0, 2, postbyte & 0x07, (postbyte & 0x10) ? REG_Y : REG_X); break; } case OPR_IDX_INDIRECT: { uint8_t x1; status = mra->read (mra, offset, 1, &x1); if (status < 0) return NULL; int idx = x1; if (postbyte & 0x01) { /* Deal with negative values */ idx -= 0x1UL << 8; } operand = create_memory_operand (true, idx, 1, xysp_reg_from_postbyte (postbyte), -1); break; } case OPR_IDX3_DIRECT: { uint8_t x[3]; status = mra->read (mra, offset, 3, x); if (status < 0) return NULL; int idx = x[0] << 16 | x[1] << 8 | x[2]; if (x[0] & 0x80) { /* Deal with negative values */ idx -= 0x1UL << 24; } operand = create_memory_operand (false, idx, 1, xysp_reg_from_postbyte (postbyte), -1); break; } case OPR_IDX3_DIRECT_REG: { uint8_t x[3]; status = mra->read (mra, offset, 3, x); if (status < 0) return NULL; int idx = x[0] << 16 | x[1] << 8 | x[2]; if (x[0] & 0x80) { /* Deal with negative values */ idx -= 0x1UL << 24; } operand = create_memory_operand (false, idx, 1, postbyte & 0x07, -1); break; } case OPR_IDX3_INDIRECT: { uint8_t x[3]; status = mra->read (mra, offset, 3, x); if (status < 0) return NULL; int idx = x[0] << 16 | x[1] << 8 | x[2]; if (x[0] & 0x80) { /* Deal with negative values */ idx -= 0x1UL << 24; } operand = create_memory_operand (true, idx, 1, xysp_reg_from_postbyte (postbyte), -1); break; } case OPR_IDX_DIRECT: { uint8_t x1; status = mra->read (mra, offset, 1, &x1); if (status < 0) return NULL; int idx = x1; if (postbyte & 0x01) { /* Deal with negative values */ idx -= 0x1UL << 8; } operand = create_memory_operand (false, idx, 1, xysp_reg_from_postbyte (postbyte), -1); break; } case OPR_IDX2_REG: { uint8_t x[2]; status = mra->read (mra, offset, 2, x); if (status < 0) return NULL; uint32_t idx = x[1] | x[0] << 8 ; idx |= (postbyte & 0x30) << 12; operand = create_memory_operand (false, idx, 1, postbyte & 0x07, -1); break; } case OPR_XY_PRE_INC: { operand = create_memory_auto_operand (OPND_RM_PRE_INC, (postbyte & 0x10) ? REG_Y: REG_X); break; } case OPR_XY_POST_INC: { operand = create_memory_auto_operand (OPND_RM_POST_INC, (postbyte & 0x10) ? REG_Y: REG_X); break; } case OPR_XY_PRE_DEC: { operand = create_memory_auto_operand (OPND_RM_PRE_DEC, (postbyte & 0x10) ? REG_Y: REG_X); break; } case OPR_XY_POST_DEC: { operand = create_memory_auto_operand (OPND_RM_POST_DEC, (postbyte & 0x10) ? REG_Y: REG_X); break; } case OPR_S_PRE_DEC: { operand = create_memory_auto_operand (OPND_RM_PRE_DEC, REG_S); break; } case OPR_S_POST_INC: { operand = create_memory_auto_operand (OPND_RM_POST_INC, REG_S); break; } case OPR_EXT18: { const size_t size = 2; bfd_byte buffer[4]; status = mra->read (mra, offset, size, buffer); if (status < 0) return NULL; uint32_t ext18 = 0; for (i = 0; i < size; ++i) { ext18 <<= 8; ext18 |= buffer[i]; } ext18 |= (postbyte & 0x01) << 16; ext18 |= (postbyte & 0x04) << 15; operand = create_simple_memory_operand (ext18, 0, false); break; } case OPR_EXT1: { uint8_t x1 = 0; status = mra->read (mra, offset, 1, &x1); if (status < 0) return NULL; int16_t addr; addr = x1; addr |= (postbyte & 0x3f) << 8; operand = create_simple_memory_operand (addr, 0, false); break; } case OPR_EXT3_DIRECT: { const size_t size = 3; bfd_byte buffer[4]; status = mra->read (mra, offset, size, buffer); if (status < 0) return NULL; uint32_t ext24 = 0; for (i = 0; i < size; ++i) { ext24 |= buffer[i] << (8 * (size - i - 1)); } operand = create_simple_memory_operand (ext24, 0, false); break; } case OPR_EXT3_INDIRECT: { const size_t size = 3; bfd_byte buffer[4]; status = mra->read (mra, offset, size, buffer); if (status < 0) return NULL; uint32_t ext24 = 0; for (i = 0; i < size; ++i) { ext24 |= buffer[i] << (8 * (size - i - 1)); } operand = create_memory_operand (true, ext24, 0, -1, -1); break; } default: printf ("Unknown OPR mode #0x%x (%d)", postbyte, mode); abort (); } if (operand != NULL) operand->osize = osize; return operand; } static struct operand * x_opr_decode (struct mem_read_abstraction_base *mra, int offset) { return x_opr_decode_with_size (mra, offset, -1); } static int z_opr_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op = x_opr_decode (mra, 0); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int z_opr_decode2 (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { int n = x_opr_n_bytes (mra, 0); if (n < 0) return n; struct operand *op = x_opr_decode (mra, 0); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = x_opr_decode (mra, n); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int imm1234 (struct mem_read_abstraction_base *mra, int base, int *n_operands, struct operand **operand) { struct operand *op; bfd_byte opcode; int status = mra->read (mra, -1, 1, &opcode); if (status < 0) return status; opcode -= base; int size = registers[opcode & 0xF].bytes; uint32_t imm; if (decode_signed_value (mra, size, &imm) < 0) return -1; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } /* Special case of LD and CMP with register S and IMM operand */ static int reg_s_imm (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; op = create_register_operand (REG_S); if (op == NULL) return -1; operand[(*n_operands)++] = op; uint32_t imm; if (decode_signed_value (mra, 3, &imm) < 0) return -1; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } /* Special case of LD, CMP and ST with register S and OPR operand */ static int reg_s_opr (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; op = create_register_operand (REG_S); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = x_opr_decode (mra, 0); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int z_imm1234_8base (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { return imm1234 (mra, 8, n_operands, operand); } static int z_imm1234_0base (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { return imm1234 (mra, 0, n_operands, operand); } static int z_tfr (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; bfd_byte byte; int status = mra->read (mra, 0, 1, &byte); if (status < 0) return status; op = create_register_operand (byte >> 4); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = create_register_operand (byte & 0x0F); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int z_reg (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; bfd_byte byte; int status = mra->read (mra, -1, 1, &byte); if (status < 0) return status; op = create_register_operand (byte & 0x07); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int reg_xy (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; bfd_byte byte; int status = mra->read (mra, -1, 1, &byte); if (status < 0) return status; op = create_register_operand ((byte & 0x01) ? REG_Y : REG_X); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int lea_reg_xys_opr (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; bfd_byte byte; int status = mra->read (mra, -1, 1, &byte); if (status < 0) return status; int reg_xys = -1; switch (byte & 0x03) { case 0x00: reg_xys = REG_X; break; case 0x01: reg_xys = REG_Y; break; case 0x02: reg_xys = REG_S; break; } op = create_register_operand (reg_xys); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = x_opr_decode (mra, 0); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int lea_reg_xys (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; bfd_byte byte; int status = mra->read (mra, -1, 1, &byte); if (status < 0) return status; int reg_n = -1; switch (byte & 0x03) { case 0x00: reg_n = REG_X; break; case 0x01: reg_n = REG_Y; break; case 0x02: reg_n = REG_S; break; } status = mra->read (mra, 0, 1, &byte); if (status < 0) return status; op = create_register_operand (reg_n); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = create_memory_operand (false, (int8_t) byte, 1, reg_n, -1); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } /* PC Relative offsets of size 15 or 7 bits */ static int rel_15_7 (struct mem_read_abstraction_base *mra, int offset, int *n_operands, struct operand **operands) { struct operand *op; bfd_byte upper; int status = mra->read (mra, offset - 1, 1, &upper); if (status < 0) return status; bool rel_size = (upper & 0x80); int16_t addr = upper; if (rel_size) { /* 15 bits. Get the next byte */ bfd_byte lower; status = mra->read (mra, offset, 1, &lower); if (status < 0) return status; addr <<= 8; addr |= lower; addr &= 0x7FFF; bool negative = (addr & 0x4000); addr &= 0x3FFF; if (negative) addr = addr - 0x4000; } else { /* 7 bits. */ bool negative = (addr & 0x40); addr &= 0x3F; if (negative) addr = addr - 0x40; } op = create_simple_memory_operand (addr, mra->posn (mra) - 1, true); if (op == NULL) return -1; operands[(*n_operands)++] = op; return 0; } /* PC Relative offsets of size 15 or 7 bits */ static int decode_rel_15_7 (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { return rel_15_7 (mra, 1, n_operands, operand); } static int shift_n_bytes (struct mem_read_abstraction_base *); static int mov_imm_opr_n_bytes (struct mem_read_abstraction_base *); static int loop_prim_n_bytes (struct mem_read_abstraction_base *); static int bm_rel_n_bytes (struct mem_read_abstraction_base *); static int mul_n_bytes (struct mem_read_abstraction_base *); static int bm_n_bytes (struct mem_read_abstraction_base *); static int psh_pul_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand); static int shift_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand); static int mul_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand); static int bm_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand); static int bm_rel_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand); static int mov_imm_opr (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand); static int loop_primitive_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operands); static int bit_field_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operands); static int exg_sex_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operands); static enum optr shift_discrim (struct mem_read_abstraction_base *mra, enum optr hint); static enum optr psh_pul_discrim (struct mem_read_abstraction_base *mra, enum optr hint); static enum optr mul_discrim (struct mem_read_abstraction_base *mra, enum optr hint); static enum optr loop_primitive_discrim (struct mem_read_abstraction_base *mra, enum optr hint); static enum optr bit_field_discrim (struct mem_read_abstraction_base *mra, enum optr hint); static enum optr exg_sex_discrim (struct mem_read_abstraction_base *mra, enum optr hint); static int cmp_xy (struct mem_read_abstraction_base *mra ATTRIBUTE_UNUSED, int *n_operands, struct operand **operand) { struct operand *op; op = create_register_operand (REG_X); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = create_register_operand (REG_Y); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int sub_d6_x_y (struct mem_read_abstraction_base *mra ATTRIBUTE_UNUSED, int *n_operands, struct operand **operand) { struct operand *op; op = create_register_operand (REG_D6); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = create_register_operand (REG_X); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = create_register_operand (REG_Y); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int sub_d6_y_x (struct mem_read_abstraction_base *mra ATTRIBUTE_UNUSED, int *n_operands, struct operand **operand) { struct operand *op; op = create_register_operand (REG_D6); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = create_register_operand (REG_Y); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = create_register_operand (REG_X); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int ld_18bit_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand); static enum optr mul_discrim (struct mem_read_abstraction_base *mra, enum optr hint) { uint8_t mb; int status = mra->read (mra, 0, 1, &mb); if (status < 0) return OP_INVALID; bool signed_op = (mb & 0x80); switch (hint) { case OPBASE_mul: return signed_op ? OP_muls : OP_mulu; break; case OPBASE_div: return signed_op ? OP_divs : OP_divu; break; case OPBASE_mod: return signed_op ? OP_mods : OP_modu; break; case OPBASE_mac: return signed_op ? OP_macs : OP_macu; break; case OPBASE_qmul: return signed_op ? OP_qmuls : OP_qmulu; break; default: abort (); } return OP_INVALID; } struct opcode { /* The operation that this opcode performs. */ enum optr operator; /* The size of this operation. May be -1 if it is implied in the operands or if size is not applicable. */ short osize; /* Some operations need this function to work out which operation is intended. */ discriminator_f discriminator; /* A function returning the number of bytes in this instruction. */ insn_bytes_f insn_bytes; operands_f operands; operands_f operands2; }; static const struct opcode page2[] = { [0x00] = {OP_ld, -1, 0, opr_n_bytes_p1, reg_s_opr, 0}, [0x01] = {OP_st, -1, 0, opr_n_bytes_p1, reg_s_opr, 0}, [0x02] = {OP_cmp, -1, 0, opr_n_bytes_p1, reg_s_opr, 0}, [0x03] = {OP_ld, -1, 0, four, reg_s_imm, 0}, [0x04] = {OP_cmp, -1, 0, four, reg_s_imm, 0}, [0x05] = {OP_stop, -1, 0, single, 0, 0}, [0x06] = {OP_wai, -1, 0, single, 0, 0}, [0x07] = {OP_sys, -1, 0, single, 0, 0}, [0x08] = {0xFFFF, -1, bit_field_discrim, bfextins_n_bytes, bit_field_decode, 0}, /* BFEXT / BFINS */ [0x09] = {0xFFFF, -1, bit_field_discrim, bfextins_n_bytes, bit_field_decode, 0}, [0x0a] = {0xFFFF, -1, bit_field_discrim, bfextins_n_bytes, bit_field_decode, 0}, [0x0b] = {0xFFFF, -1, bit_field_discrim, bfextins_n_bytes, bit_field_decode, 0}, [0x0c] = {0xFFFF, -1, bit_field_discrim, bfextins_n_bytes, bit_field_decode, 0}, [0x0d] = {0xFFFF, -1, bit_field_discrim, bfextins_n_bytes, bit_field_decode, 0}, [0x0e] = {0xFFFF, -1, bit_field_discrim, bfextins_n_bytes, bit_field_decode, 0}, [0x0f] = {0xFFFF, -1, bit_field_discrim, bfextins_n_bytes, bit_field_decode, 0}, [0x10] = {OP_minu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x11] = {OP_minu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x12] = {OP_minu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x13] = {OP_minu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x14] = {OP_minu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x15] = {OP_minu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x16] = {OP_minu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x17] = {OP_minu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x18] = {OP_maxu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x19] = {OP_maxu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x1a] = {OP_maxu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x1b] = {OP_maxu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x1c] = {OP_maxu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x1d] = {OP_maxu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x1e] = {OP_maxu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x1f] = {OP_maxu, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x20] = {OP_mins, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x21] = {OP_mins, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x22] = {OP_mins, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x23] = {OP_mins, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x24] = {OP_mins, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x25] = {OP_mins, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x26] = {OP_mins, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x27] = {OP_mins, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x28] = {OP_maxs, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x29] = {OP_maxs, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x2a] = {OP_maxs, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x2b] = {OP_maxs, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x2c] = {OP_maxs, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x2d] = {OP_maxs, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x2e] = {OP_maxs, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x2f] = {OP_maxs, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x30] = {OPBASE_div, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x31] = {OPBASE_div, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x32] = {OPBASE_div, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x33] = {OPBASE_div, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x34] = {OPBASE_div, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x35] = {OPBASE_div, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x36] = {OPBASE_div, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x37] = {OPBASE_div, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x38] = {OPBASE_mod, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x39] = {OPBASE_mod, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x3a] = {OPBASE_mod, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x3b] = {OPBASE_mod, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x3c] = {OPBASE_mod, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x3d] = {OPBASE_mod, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x3e] = {OPBASE_mod, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x3f] = {OPBASE_mod, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x40] = {OP_abs, -1, 0, single, z_reg, 0}, [0x41] = {OP_abs, -1, 0, single, z_reg, 0}, [0x42] = {OP_abs, -1, 0, single, z_reg, 0}, [0x43] = {OP_abs, -1, 0, single, z_reg, 0}, [0x44] = {OP_abs, -1, 0, single, z_reg, 0}, [0x45] = {OP_abs, -1, 0, single, z_reg, 0}, [0x46] = {OP_abs, -1, 0, single, z_reg, 0}, [0x47] = {OP_abs, -1, 0, single, z_reg, 0}, [0x48] = {OPBASE_mac, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x49] = {OPBASE_mac, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4a] = {OPBASE_mac, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4b] = {OPBASE_mac, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4c] = {OPBASE_mac, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4d] = {OPBASE_mac, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4e] = {OPBASE_mac, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4f] = {OPBASE_mac, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x50] = {OP_adc, -1, 0, three, z_reg, z_imm1234_0base}, [0x51] = {OP_adc, -1, 0, three, z_reg, z_imm1234_0base}, [0x52] = {OP_adc, -1, 0, three, z_reg, z_imm1234_0base}, [0x53] = {OP_adc, -1, 0, three, z_reg, z_imm1234_0base}, [0x54] = {OP_adc, -1, 0, two, z_reg, z_imm1234_0base}, [0x55] = {OP_adc, -1, 0, two, z_reg, z_imm1234_0base}, [0x56] = {OP_adc, -1, 0, five, z_reg, z_imm1234_0base}, [0x57] = {OP_adc, -1, 0, five, z_reg, z_imm1234_0base}, [0x58] = {OP_bit, -1, 0, three, z_reg, z_imm1234_8base}, [0x59] = {OP_bit, -1, 0, three, z_reg, z_imm1234_8base}, [0x5a] = {OP_bit, -1, 0, three, z_reg, z_imm1234_8base}, [0x5b] = {OP_bit, -1, 0, three, z_reg, z_imm1234_8base}, [0x5c] = {OP_bit, -1, 0, two, z_reg, z_imm1234_8base}, [0x5d] = {OP_bit, -1, 0, two, z_reg, z_imm1234_8base}, [0x5e] = {OP_bit, -1, 0, five, z_reg, z_imm1234_8base}, [0x5f] = {OP_bit, -1, 0, five, z_reg, z_imm1234_8base}, [0x60] = {OP_adc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x61] = {OP_adc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x62] = {OP_adc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x63] = {OP_adc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x64] = {OP_adc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x65] = {OP_adc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x66] = {OP_adc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x67] = {OP_adc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x68] = {OP_bit, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x69] = {OP_bit, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6a] = {OP_bit, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6b] = {OP_bit, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6c] = {OP_bit, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6d] = {OP_bit, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6e] = {OP_bit, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6f] = {OP_bit, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x70] = {OP_sbc, -1, 0, three, z_reg, z_imm1234_0base}, [0x71] = {OP_sbc, -1, 0, three, z_reg, z_imm1234_0base}, [0x72] = {OP_sbc, -1, 0, three, z_reg, z_imm1234_0base}, [0x73] = {OP_sbc, -1, 0, three, z_reg, z_imm1234_0base}, [0x74] = {OP_sbc, -1, 0, two, z_reg, z_imm1234_0base}, [0x75] = {OP_sbc, -1, 0, two, z_reg, z_imm1234_0base}, [0x76] = {OP_sbc, -1, 0, five, z_reg, z_imm1234_0base}, [0x77] = {OP_sbc, -1, 0, five, z_reg, z_imm1234_0base}, [0x78] = {OP_eor, -1, 0, three, z_reg, z_imm1234_8base}, [0x79] = {OP_eor, -1, 0, three, z_reg, z_imm1234_8base}, [0x7a] = {OP_eor, -1, 0, three, z_reg, z_imm1234_8base}, [0x7b] = {OP_eor, -1, 0, three, z_reg, z_imm1234_8base}, [0x7c] = {OP_eor, -1, 0, two, z_reg, z_imm1234_8base}, [0x7d] = {OP_eor, -1, 0, two, z_reg, z_imm1234_8base}, [0x7e] = {OP_eor, -1, 0, five, z_reg, z_imm1234_8base}, [0x7f] = {OP_eor, -1, 0, five, z_reg, z_imm1234_8base}, [0x80] = {OP_sbc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x81] = {OP_sbc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x82] = {OP_sbc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x83] = {OP_sbc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x84] = {OP_sbc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x85] = {OP_sbc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x86] = {OP_sbc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x87] = {OP_sbc, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x88] = {OP_eor, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x89] = {OP_eor, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8a] = {OP_eor, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8b] = {OP_eor, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8c] = {OP_eor, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8d] = {OP_eor, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8e] = {OP_eor, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8f] = {OP_eor, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x90] = {OP_rti, -1, 0, single, 0, 0}, [0x91] = {OP_clb, -1, 0, two, z_tfr, 0}, [0x92] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x93] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x94] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x95] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x96] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x97] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x98] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x99] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x9a] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x9b] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x9c] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x9d] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x9e] = {OP_trap, -1, 0, single, trap_decode, 0}, [0x9f] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xa0] = {OP_sat, -1, 0, single, z_reg, 0}, [0xa1] = {OP_sat, -1, 0, single, z_reg, 0}, [0xa2] = {OP_sat, -1, 0, single, z_reg, 0}, [0xa3] = {OP_sat, -1, 0, single, z_reg, 0}, [0xa4] = {OP_sat, -1, 0, single, z_reg, 0}, [0xa5] = {OP_sat, -1, 0, single, z_reg, 0}, [0xa6] = {OP_sat, -1, 0, single, z_reg, 0}, [0xa7] = {OP_sat, -1, 0, single, z_reg, 0}, [0xa8] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xa9] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xaa] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xab] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xac] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xad] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xae] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xaf] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xb0] = {OPBASE_qmul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0xb1] = {OPBASE_qmul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0xb2] = {OPBASE_qmul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0xb3] = {OPBASE_qmul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0xb4] = {OPBASE_qmul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0xb5] = {OPBASE_qmul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0xb6] = {OPBASE_qmul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0xb7] = {OPBASE_qmul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0xb8] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xb9] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xba] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xbb] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xbc] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xbd] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xbe] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xbf] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc0] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc1] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc2] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc3] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc4] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc5] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc6] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc7] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc8] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xc9] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xca] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xcb] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xcc] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xcd] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xce] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xcf] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd0] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd1] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd2] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd3] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd4] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd5] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd6] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd7] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd8] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xd9] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xda] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xdb] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xdc] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xdd] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xde] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xdf] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe0] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe1] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe2] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe3] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe4] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe5] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe6] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe7] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe8] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xe9] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xea] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xeb] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xec] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xed] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xee] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xef] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf0] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf1] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf2] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf3] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf4] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf5] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf6] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf7] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf8] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xf9] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xfa] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xfb] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xfc] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xfd] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xfe] = {OP_trap, -1, 0, single, trap_decode, 0}, [0xff] = {OP_trap, -1, 0, single, trap_decode, 0}, }; static const struct opcode page1[] = { [0x00] = {OP_bgnd, -1, 0, single, 0, 0}, [0x01] = {OP_nop, -1, 0, single, 0, 0}, [0x02] = {OP_brclr, -1, 0, bm_rel_n_bytes, bm_rel_decode, 0}, [0x03] = {OP_brset, -1, 0, bm_rel_n_bytes, bm_rel_decode, 0}, [0x04] = {0xFFFF, -1, psh_pul_discrim, two, psh_pul_decode, 0}, /* psh/pul */ [0x05] = {OP_rts, -1, 0, single, 0, 0}, [0x06] = {OP_lea, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x07] = {OP_lea, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x08] = {OP_lea, -1, 0, opr_n_bytes_p1, lea_reg_xys_opr, 0}, [0x09] = {OP_lea, -1, 0, opr_n_bytes_p1, lea_reg_xys_opr, 0}, [0x0a] = {OP_lea, -1, 0, opr_n_bytes_p1, lea_reg_xys_opr, 0}, [0x0b] = {0xFFFF, -1, loop_primitive_discrim, loop_prim_n_bytes, loop_primitive_decode, 0}, /* Loop primitives TBcc / DBcc */ [0x0c] = {OP_mov, 0, 0, mov_imm_opr_n_bytes, mov_imm_opr, 0}, [0x0d] = {OP_mov, 1, 0, mov_imm_opr_n_bytes, mov_imm_opr, 0}, [0x0e] = {OP_mov, 2, 0, mov_imm_opr_n_bytes, mov_imm_opr, 0}, [0x0f] = {OP_mov, 3, 0, mov_imm_opr_n_bytes, mov_imm_opr, 0}, [0x10] = {0xFFFF, -1, shift_discrim, shift_n_bytes, shift_decode, 0}, /* lsr/lsl/asl/asr/rol/ror */ [0x11] = {0xFFFF, -1, shift_discrim, shift_n_bytes, shift_decode, 0}, [0x12] = {0xFFFF, -1, shift_discrim, shift_n_bytes, shift_decode, 0}, [0x13] = {0xFFFF, -1, shift_discrim, shift_n_bytes, shift_decode, 0}, [0x14] = {0xFFFF, -1, shift_discrim, shift_n_bytes, shift_decode, 0}, [0x15] = {0xFFFF, -1, shift_discrim, shift_n_bytes, shift_decode, 0}, [0x16] = {0xFFFF, -1, shift_discrim, shift_n_bytes, shift_decode, 0}, [0x17] = {0xFFFF, -1, shift_discrim, shift_n_bytes, shift_decode, 0}, [0x18] = {OP_lea, -1, 0, two, lea_reg_xys, NULL}, [0x19] = {OP_lea, -1, 0, two, lea_reg_xys, NULL}, [0x1a] = {OP_lea, -1, 0, two, lea_reg_xys, NULL}, /* 0x1b PG2 */ [0x1c] = {OP_mov, 0, 0, opr_n_bytes2, z_opr_decode2, 0}, [0x1d] = {OP_mov, 1, 0, opr_n_bytes2, z_opr_decode2, 0}, [0x1e] = {OP_mov, 2, 0, opr_n_bytes2, z_opr_decode2, 0}, [0x1f] = {OP_mov, 3, 0, opr_n_bytes2, z_opr_decode2, 0}, [0x20] = {OP_bra, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x21] = {OP_bsr, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x22] = {OP_bhi, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x23] = {OP_bls, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x24] = {OP_bcc, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x25] = {OP_bcs, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x26] = {OP_bne, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x27] = {OP_beq, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x28] = {OP_bvc, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x29] = {OP_bvs, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x2a] = {OP_bpl, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x2b] = {OP_bmi, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x2c] = {OP_bge, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x2d] = {OP_blt, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x2e] = {OP_bgt, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x2f] = {OP_ble, -1, 0, pcrel_15bit, decode_rel_15_7, 0}, [0x30] = {OP_inc, -1, 0, single, z_reg, 0}, [0x31] = {OP_inc, -1, 0, single, z_reg, 0}, [0x32] = {OP_inc, -1, 0, single, z_reg, 0}, [0x33] = {OP_inc, -1, 0, single, z_reg, 0}, [0x34] = {OP_inc, -1, 0, single, z_reg, 0}, [0x35] = {OP_inc, -1, 0, single, z_reg, 0}, [0x36] = {OP_inc, -1, 0, single, z_reg, 0}, [0x37] = {OP_inc, -1, 0, single, z_reg, 0}, [0x38] = {OP_clr, -1, 0, single, z_reg, 0}, [0x39] = {OP_clr, -1, 0, single, z_reg, 0}, [0x3a] = {OP_clr, -1, 0, single, z_reg, 0}, [0x3b] = {OP_clr, -1, 0, single, z_reg, 0}, [0x3c] = {OP_clr, -1, 0, single, z_reg, 0}, [0x3d] = {OP_clr, -1, 0, single, z_reg, 0}, [0x3e] = {OP_clr, -1, 0, single, z_reg, 0}, [0x3f] = {OP_clr, -1, 0, single, z_reg, 0}, [0x40] = {OP_dec, -1, 0, single, z_reg, 0}, [0x41] = {OP_dec, -1, 0, single, z_reg, 0}, [0x42] = {OP_dec, -1, 0, single, z_reg, 0}, [0x43] = {OP_dec, -1, 0, single, z_reg, 0}, [0x44] = {OP_dec, -1, 0, single, z_reg, 0}, [0x45] = {OP_dec, -1, 0, single, z_reg, 0}, [0x46] = {OP_dec, -1, 0, single, z_reg, 0}, [0x47] = {OP_dec, -1, 0, single, z_reg, 0}, [0x48] = {OPBASE_mul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x49] = {OPBASE_mul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4a] = {OPBASE_mul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4b] = {OPBASE_mul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4c] = {OPBASE_mul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4d] = {OPBASE_mul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4e] = {OPBASE_mul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x4f] = {OPBASE_mul, -1, mul_discrim, mul_n_bytes, mul_decode, 0}, [0x50] = {OP_add, -1, 0, three, z_reg, z_imm1234_0base}, [0x51] = {OP_add, -1, 0, three, z_reg, z_imm1234_0base}, [0x52] = {OP_add, -1, 0, three, z_reg, z_imm1234_0base}, [0x53] = {OP_add, -1, 0, three, z_reg, z_imm1234_0base}, [0x54] = {OP_add, -1, 0, two, z_reg, z_imm1234_0base}, [0x55] = {OP_add, -1, 0, two, z_reg, z_imm1234_0base}, [0x56] = {OP_add, -1, 0, five, z_reg, z_imm1234_0base}, [0x57] = {OP_add, -1, 0, five, z_reg, z_imm1234_0base}, [0x58] = {OP_and, -1, 0, three, z_reg, z_imm1234_8base}, [0x59] = {OP_and, -1, 0, three, z_reg, z_imm1234_8base}, [0x5a] = {OP_and, -1, 0, three, z_reg, z_imm1234_8base}, [0x5b] = {OP_and, -1, 0, three, z_reg, z_imm1234_8base}, [0x5c] = {OP_and, -1, 0, two, z_reg, z_imm1234_8base}, [0x5d] = {OP_and, -1, 0, two, z_reg, z_imm1234_8base}, [0x5e] = {OP_and, -1, 0, five, z_reg, z_imm1234_8base}, [0x5f] = {OP_and, -1, 0, five, z_reg, z_imm1234_8base}, [0x60] = {OP_add, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x61] = {OP_add, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x62] = {OP_add, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x63] = {OP_add, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x64] = {OP_add, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x65] = {OP_add, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x66] = {OP_add, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x67] = {OP_add, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x68] = {OP_and, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x69] = {OP_and, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6a] = {OP_and, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6b] = {OP_and, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6c] = {OP_and, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6d] = {OP_and, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6e] = {OP_and, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x6f] = {OP_and, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x70] = {OP_sub, -1, 0, three, z_reg, z_imm1234_0base}, [0x71] = {OP_sub, -1, 0, three, z_reg, z_imm1234_0base}, [0x72] = {OP_sub, -1, 0, three, z_reg, z_imm1234_0base}, [0x73] = {OP_sub, -1, 0, three, z_reg, z_imm1234_0base}, [0x74] = {OP_sub, -1, 0, two, z_reg, z_imm1234_0base}, [0x75] = {OP_sub, -1, 0, two, z_reg, z_imm1234_0base}, [0x76] = {OP_sub, -1, 0, five, z_reg, z_imm1234_0base}, [0x77] = {OP_sub, -1, 0, five, z_reg, z_imm1234_0base}, [0x78] = {OP_or, -1, 0, three, z_reg, z_imm1234_8base}, [0x79] = {OP_or, -1, 0, three, z_reg, z_imm1234_8base}, [0x7a] = {OP_or, -1, 0, three, z_reg, z_imm1234_8base}, [0x7b] = {OP_or, -1, 0, three, z_reg, z_imm1234_8base}, [0x7c] = {OP_or, -1, 0, two, z_reg, z_imm1234_8base}, [0x7d] = {OP_or, -1, 0, two, z_reg, z_imm1234_8base}, [0x7e] = {OP_or, -1, 0, five, z_reg, z_imm1234_8base}, [0x7f] = {OP_or, -1, 0, five, z_reg, z_imm1234_8base}, [0x80] = {OP_sub, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x81] = {OP_sub, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x82] = {OP_sub, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x83] = {OP_sub, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x84] = {OP_sub, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x85] = {OP_sub, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x86] = {OP_sub, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x87] = {OP_sub, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x88] = {OP_or, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x89] = {OP_or, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8a] = {OP_or, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8b] = {OP_or, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8c] = {OP_or, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8d] = {OP_or, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8e] = {OP_or, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x8f] = {OP_or, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0x90] = {OP_ld, -1, 0, three, z_reg, z_imm1234_0base}, [0x91] = {OP_ld, -1, 0, three, z_reg, z_imm1234_0base}, [0x92] = {OP_ld, -1, 0, three, z_reg, z_imm1234_0base}, [0x93] = {OP_ld, -1, 0, three, z_reg, z_imm1234_0base}, [0x94] = {OP_ld, -1, 0, two, z_reg, z_imm1234_0base}, [0x95] = {OP_ld, -1, 0, two, z_reg, z_imm1234_0base}, [0x96] = {OP_ld, -1, 0, five, z_reg, z_imm1234_0base}, [0x97] = {OP_ld, -1, 0, five, z_reg, z_imm1234_0base}, [0x98] = {OP_ld, -1, 0, four, reg_xy, z_imm1234_0base}, [0x99] = {OP_ld, -1, 0, four, reg_xy, z_imm1234_0base}, [0x9a] = {OP_clr, -1, 0, single, reg_xy, 0}, [0x9b] = {OP_clr, -1, 0, single, reg_xy, 0}, [0x9c] = {OP_inc, 0, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0x9d] = {OP_inc, 1, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0x9e] = {OP_tfr, -1, 0, two, z_tfr, NULL}, [0x9f] = {OP_inc, 3, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xa0] = {OP_ld, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xa1] = {OP_ld, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xa2] = {OP_ld, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xa3] = {OP_ld, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xa4] = {OP_ld, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xa5] = {OP_ld, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xa6] = {OP_ld, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xa7] = {OP_ld, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xa8] = {OP_ld, -1, 0, opr_n_bytes_p1, reg_xy, z_opr_decode}, [0xa9] = {OP_ld, -1, 0, opr_n_bytes_p1, reg_xy, z_opr_decode}, [0xaa] = {OP_jmp, -1, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xab] = {OP_jsr, -1, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xac] = {OP_dec, 0, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xad] = {OP_dec, 1, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xae] = {0xFFFF, -1, exg_sex_discrim, two, exg_sex_decode, 0}, /* EXG / SEX */ [0xaf] = {OP_dec, 3, 0, opr_n_bytes_p1, 0, z_opr_decode}, [0xb0] = {OP_ld, -1, 0, four, z_reg, z_ext24_decode}, [0xb1] = {OP_ld, -1, 0, four, z_reg, z_ext24_decode}, [0xb2] = {OP_ld, -1, 0, four, z_reg, z_ext24_decode}, [0xb3] = {OP_ld, -1, 0, four, z_reg, z_ext24_decode}, [0xb4] = {OP_ld, -1, 0, four, z_reg, z_ext24_decode}, [0xb5] = {OP_ld, -1, 0, four, z_reg, z_ext24_decode}, [0xb6] = {OP_ld, -1, 0, four, z_reg, z_ext24_decode}, [0xb7] = {OP_ld, -1, 0, four, z_reg, z_ext24_decode}, [0xb8] = {OP_ld, -1, 0, four, reg_xy, z_ext24_decode}, [0xb9] = {OP_ld, -1, 0, four, reg_xy, z_ext24_decode}, [0xba] = {OP_jmp, -1, 0, four, z_ext24_decode, 0}, [0xbb] = {OP_jsr, -1, 0, four, z_ext24_decode, 0}, [0xbc] = {OP_clr, 0, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xbd] = {OP_clr, 1, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xbe] = {OP_clr, 2, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xbf] = {OP_clr, 3, 0, opr_n_bytes_p1, z_opr_decode, 0}, [0xc0] = {OP_st, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xc1] = {OP_st, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xc2] = {OP_st, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xc3] = {OP_st, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xc4] = {OP_st, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xc5] = {OP_st, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xc6] = {OP_st, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xc7] = {OP_st, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xc8] = {OP_st, -1, 0, opr_n_bytes_p1, reg_xy, z_opr_decode}, [0xc9] = {OP_st, -1, 0, opr_n_bytes_p1, reg_xy, z_opr_decode}, [0xca] = {OP_ld, -1, 0, three, reg_xy, ld_18bit_decode}, [0xcb] = {OP_ld, -1, 0, three, reg_xy, ld_18bit_decode}, [0xcc] = {OP_com, 0, 0, opr_n_bytes_p1, NULL, z_opr_decode}, [0xcd] = {OP_com, 1, 0, opr_n_bytes_p1, NULL, z_opr_decode}, [0xce] = {OP_andcc, -1, 0, two, imm1_decode, 0}, [0xcf] = {OP_com, 3, 0, opr_n_bytes_p1, NULL, z_opr_decode}, [0xd0] = {OP_st, -1, 0, four, z_reg, z_ext24_decode}, [0xd1] = {OP_st, -1, 0, four, z_reg, z_ext24_decode}, [0xd2] = {OP_st, -1, 0, four, z_reg, z_ext24_decode}, [0xd3] = {OP_st, -1, 0, four, z_reg, z_ext24_decode}, [0xd4] = {OP_st, -1, 0, four, z_reg, z_ext24_decode}, [0xd5] = {OP_st, -1, 0, four, z_reg, z_ext24_decode}, [0xd6] = {OP_st, -1, 0, four, z_reg, z_ext24_decode}, [0xd7] = {OP_st, -1, 0, four, z_reg, z_ext24_decode}, [0xd8] = {OP_st, -1, 0, four, reg_xy, z_ext24_decode}, [0xd9] = {OP_st, -1, 0, four, reg_xy, z_ext24_decode}, [0xda] = {OP_ld, -1, 0, three, reg_xy, ld_18bit_decode}, [0xdb] = {OP_ld, -1, 0, three, reg_xy, ld_18bit_decode}, [0xdc] = {OP_neg, 0, 0, opr_n_bytes_p1, NULL, z_opr_decode}, [0xdd] = {OP_neg, 1, 0, opr_n_bytes_p1, NULL, z_opr_decode}, [0xde] = {OP_orcc, -1, 0, two, imm1_decode, 0}, [0xdf] = {OP_neg, 3, 0, opr_n_bytes_p1, NULL, z_opr_decode}, [0xe0] = {OP_cmp, -1, 0, three, z_reg, z_imm1234_0base}, [0xe1] = {OP_cmp, -1, 0, three, z_reg, z_imm1234_0base}, [0xe2] = {OP_cmp, -1, 0, three, z_reg, z_imm1234_0base}, [0xe3] = {OP_cmp, -1, 0, three, z_reg, z_imm1234_0base}, [0xe4] = {OP_cmp, -1, 0, two, z_reg, z_imm1234_0base}, [0xe5] = {OP_cmp, -1, 0, two, z_reg, z_imm1234_0base}, [0xe6] = {OP_cmp, -1, 0, five, z_reg, z_imm1234_0base}, [0xe7] = {OP_cmp, -1, 0, five, z_reg, z_imm1234_0base}, [0xe8] = {OP_cmp, -1, 0, four, reg_xy, z_imm1234_0base}, [0xe9] = {OP_cmp, -1, 0, four, reg_xy, z_imm1234_0base}, [0xea] = {OP_ld, -1, 0, three, reg_xy, ld_18bit_decode}, [0xeb] = {OP_ld, -1, 0, three, reg_xy, ld_18bit_decode}, [0xec] = {OP_bclr, -1, 0, bm_n_bytes, bm_decode, 0}, [0xed] = {OP_bset, -1, 0, bm_n_bytes, bm_decode, 0}, [0xee] = {OP_btgl, -1, 0, bm_n_bytes, bm_decode, 0}, [0xef] = {OP_INVALID, -1, 0, NULL, NULL, NULL}, /* SPARE */ [0xf0] = {OP_cmp, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xf1] = {OP_cmp, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xf2] = {OP_cmp, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xf3] = {OP_cmp, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xf4] = {OP_cmp, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xf5] = {OP_cmp, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xf6] = {OP_cmp, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xf7] = {OP_cmp, -1, 0, opr_n_bytes_p1, z_reg, z_opr_decode}, [0xf8] = {OP_cmp, -1, 0, opr_n_bytes_p1, reg_xy, z_opr_decode}, [0xf9] = {OP_cmp, -1, 0, opr_n_bytes_p1, reg_xy, z_opr_decode}, [0xfa] = {OP_ld, -1, 0, three, reg_xy, ld_18bit_decode}, [0xfb] = {OP_ld, -1, 0, three, reg_xy, ld_18bit_decode}, [0xfc] = {OP_cmp, -1, 0, single, cmp_xy, 0}, [0xfd] = {OP_sub, -1, 0, single, sub_d6_x_y, 0}, [0xfe] = {OP_sub, -1, 0, single, sub_d6_y_x, 0}, [0xff] = {OP_swi, -1, 0, single, 0, 0} }; static const int oprregs1[] = { REG_D3, REG_D2, REG_D1, REG_D0, REG_CCL, REG_CCH }; static const int oprregs2[] = { REG_Y, REG_X, REG_D7, REG_D6, REG_D5, REG_D4 }; enum MUL_MODE { MUL_REG_REG, MUL_REG_OPR, MUL_REG_IMM, MUL_OPR_OPR }; struct mb { uint8_t mask; uint8_t value; enum MUL_MODE mode; }; static const struct mb mul_table[] = { {0x40, 0x00, MUL_REG_REG}, {0x47, 0x40, MUL_REG_OPR}, {0x47, 0x41, MUL_REG_OPR}, {0x47, 0x43, MUL_REG_OPR}, {0x47, 0x44, MUL_REG_IMM}, {0x47, 0x45, MUL_REG_IMM}, {0x47, 0x47, MUL_REG_IMM}, {0x43, 0x42, MUL_OPR_OPR}, }; static int mul_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { uint8_t mb; struct operand *op; int status = mra->read (mra, 0, 1, &mb); if (status < 0) return status; uint8_t byte; status = mra->read (mra, -1, 1, &byte); if (status < 0) return status; enum MUL_MODE mode = -1; size_t i; for (i = 0; i < sizeof (mul_table) / sizeof (mul_table[0]); ++i) { const struct mb *mm = mul_table + i; if ((mb & mm->mask) == mm->value) { mode = mm->mode; break; } } op = create_register_operand (byte & 0x07); if (op == NULL) return -1; operand[(*n_operands)++] = op; switch (mode) { case MUL_REG_IMM: { int size = (mb & 0x3); op = create_register_operand_with_size ((mb & 0x38) >> 3, size); if (op == NULL) return -1; operand[(*n_operands)++] = op; uint32_t imm; if (z_decode_signed_value (mra, 1, size + 1, &imm) < 0) return -1; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; } break; case MUL_REG_REG: op = create_register_operand ((mb & 0x38) >> 3); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = create_register_operand (mb & 0x07); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case MUL_REG_OPR: op = create_register_operand ((mb & 0x38) >> 3); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = x_opr_decode_with_size (mra, 1, mb & 0x3); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case MUL_OPR_OPR: { int first = x_opr_n_bytes (mra, 1); if (first < 0) return first; op = x_opr_decode_with_size (mra, 1, (mb & 0x30) >> 4); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = x_opr_decode_with_size (mra, first + 1, (mb & 0x0c) >> 2); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; } } return 0; } static int mul_n_bytes (struct mem_read_abstraction_base *mra) { int nx = 2; int first, second; uint8_t mb; int status = mra->read (mra, 0, 1, &mb); if (status < 0) return status; enum MUL_MODE mode = -1; size_t i; for (i = 0; i < sizeof (mul_table) / sizeof (mul_table[0]); ++i) { const struct mb *mm = mul_table + i; if ((mb & mm->mask) == mm->value) { mode = mm->mode; break; } } int size = (mb & 0x3) + 1; switch (mode) { case MUL_REG_IMM: nx += size; break; case MUL_REG_REG: break; case MUL_REG_OPR: first = x_opr_n_bytes (mra, 1); if (first < 0) return first; nx += first; break; case MUL_OPR_OPR: first = x_opr_n_bytes (mra, nx - 1); if (first < 0) return first; nx += first; second = x_opr_n_bytes (mra, nx - 1); if (second < 0) return second; nx += second; break; } return nx; } /* The NXP documentation is vague about BM_RESERVED0 and BM_RESERVED1, and contains obvious typos. However the Freescale tools and experiments with the chip itself seem to indicate that they behave like BM_REG_IMM and BM_OPR_REG respectively. */ enum BM_MODE { BM_REG_IMM, BM_RESERVED0, BM_OPR_B, BM_OPR_W, BM_OPR_L, BM_OPR_REG, BM_RESERVED1 }; struct bm { uint8_t mask; uint8_t value; enum BM_MODE mode; }; static const struct bm bm_table[] = { { 0xC6, 0x04, BM_REG_IMM}, { 0x84, 0x00, BM_REG_IMM}, { 0x06, 0x06, BM_REG_IMM}, { 0xC6, 0x44, BM_RESERVED0}, { 0x8F, 0x80, BM_OPR_B}, { 0x8E, 0x82, BM_OPR_W}, { 0x8C, 0x88, BM_OPR_L}, { 0x83, 0x81, BM_OPR_REG}, { 0x87, 0x84, BM_RESERVED1}, }; static int bm_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; uint8_t bm; int status = mra->read (mra, 0, 1, &bm); if (status < 0) return status; size_t i; enum BM_MODE mode = -1; for (i = 0; i < sizeof (bm_table) / sizeof (bm_table[0]); ++i) { const struct bm *bme = bm_table + i; if ((bm & bme->mask) == bme->value) { mode = bme->mode; break; } } switch (mode) { case BM_REG_IMM: case BM_RESERVED0: op = create_register_operand (bm & 0x07); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_OPR_B: op = x_opr_decode_with_size (mra, 1, 0); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_OPR_W: op = x_opr_decode_with_size (mra, 1, 1); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_OPR_L: op = x_opr_decode_with_size (mra, 1, 3); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_OPR_REG: case BM_RESERVED1: { uint8_t xb; status = mra->read (mra, 1, 1, &xb); if (status < 0) return status; /* Don't emit a size suffix for register operands */ if ((xb & 0xF8) != 0xB8) op = x_opr_decode_with_size (mra, 1, (bm & 0x0c) >> 2); else op = x_opr_decode (mra, 1); if (op == NULL) return -1; operand[(*n_operands)++] = op; } break; } uint8_t imm = 0; switch (mode) { case BM_REG_IMM: case BM_RESERVED0: imm = (bm & 0x38) >> 3; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_OPR_L: imm |= (bm & 0x03) << 3; /* fallthrough */ case BM_OPR_W: imm |= (bm & 0x01) << 3; /* fallthrough */ case BM_OPR_B: imm |= (bm & 0x70) >> 4; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_OPR_REG: case BM_RESERVED1: op = create_register_operand ((bm & 0x70) >> 4); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; } return 0; } static int bm_rel_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; uint8_t bm; int status = mra->read (mra, 0, 1, &bm); if (status < 0) return status; size_t i; enum BM_MODE mode = -1; for (i = 0; i < sizeof (bm_table) / sizeof (bm_table[0]); ++i) { const struct bm *bme = bm_table + i; if ((bm & bme->mask) == bme->value) { mode = bme->mode; break; } } int n = 1; switch (mode) { case BM_REG_IMM: case BM_RESERVED0: op = create_register_operand (bm & 0x07); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_OPR_B: op = x_opr_decode_with_size (mra, 1, 0); if (op == NULL) return -1; operand[(*n_operands)++] = op; n = x_opr_n_bytes (mra, 1); if (n < 0) return n; n += 1; break; case BM_OPR_W: op = x_opr_decode_with_size (mra, 1, 1); if (op == NULL) return -1; operand[(*n_operands)++] = op; n = x_opr_n_bytes (mra, 1); if (n < 0) return n; n += 1; break; case BM_OPR_L: op = x_opr_decode_with_size (mra, 1, 3); if (op == NULL) return -1; operand[(*n_operands)++] = op; n = x_opr_n_bytes (mra, 1); if (n < 0) return n; n += 1; break; case BM_OPR_REG: case BM_RESERVED1: { uint8_t xb; status = mra->read (mra, +1, 1, &xb); if (status < 0) return status; /* Don't emit a size suffix for register operands */ if ((xb & 0xF8) != 0xB8) { short os = (bm & 0x0c) >> 2; op = x_opr_decode_with_size (mra, 1, os); } else op = x_opr_decode (mra, 1); if (op == NULL) return -1; operand[(*n_operands)++] = op; } break; } int x, imm = 0; switch (mode) { case BM_OPR_L: imm |= (bm & 0x02) << 3; /* fall through */ case BM_OPR_W: imm |= (bm & 0x01) << 3; /* fall through */ case BM_OPR_B: imm |= (bm & 0x70) >> 4; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_RESERVED0: imm = (bm & 0x38) >> 3; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_REG_IMM: imm = (bm & 0xF8) >> 3; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; break; case BM_OPR_REG: case BM_RESERVED1: op = create_register_operand ((bm & 0x70) >> 4); if (op == NULL) return -1; operand[(*n_operands)++] = op; x = x_opr_n_bytes (mra, 1); if (x < 0) return x; n += x; break; } return rel_15_7 (mra, n + 1, n_operands, operand); } static int bm_n_bytes (struct mem_read_abstraction_base *mra) { uint8_t bm; int status = mra->read (mra, 0, 1, &bm); if (status < 0) return status; size_t i; enum BM_MODE mode = -1; for (i = 0; i < sizeof (bm_table) / sizeof (bm_table[0]); ++i) { const struct bm *bme = bm_table + i; if ((bm & bme->mask) == bme->value) { mode = bme->mode; break; } } int n = 0; switch (mode) { case BM_REG_IMM: case BM_RESERVED0: break; case BM_OPR_B: case BM_OPR_W: case BM_OPR_L: case BM_OPR_REG: case BM_RESERVED1: n = x_opr_n_bytes (mra, 1); if (n < 0) return n; break; } return n + 2; } static int bm_rel_n_bytes (struct mem_read_abstraction_base *mra) { int n = 1 + bm_n_bytes (mra); bfd_byte rb; int status = mra->read (mra, n - 2, 1, &rb); if (status != 0) return status; if (rb & 0x80) n++; return n; } /* shift direction */ enum SB_DIR { SB_LEFT, SB_RIGHT }; enum SB_TYPE { SB_ARITHMETIC, SB_LOGICAL }; enum SB_MODE { SB_REG_REG_N_EFF, SB_REG_REG_N, SB_REG_OPR_EFF, SB_ROT, SB_REG_OPR_OPR, SB_OPR_N }; struct sb { uint8_t mask; uint8_t value; enum SB_MODE mode; }; static const struct sb sb_table[] = { {0x30, 0x00, SB_REG_REG_N_EFF}, {0x30, 0x10, SB_REG_REG_N}, {0x34, 0x20, SB_REG_OPR_EFF}, {0x34, 0x24, SB_ROT}, {0x34, 0x30, SB_REG_OPR_OPR}, {0x34, 0x34, SB_OPR_N}, }; static int shift_n_bytes (struct mem_read_abstraction_base *mra) { bfd_byte sb; int opr1, opr2; int status = mra->read (mra, 0, 1, &sb); if (status != 0) return status; size_t i; enum SB_MODE mode = -1; for (i = 0; i < sizeof (sb_table) / sizeof (sb_table[0]); ++i) { const struct sb *sbe = sb_table + i; if ((sb & sbe->mask) == sbe->value) mode = sbe->mode; } switch (mode) { case SB_REG_REG_N_EFF: return 2; case SB_REG_OPR_EFF: case SB_ROT: opr1 = x_opr_n_bytes (mra, 1); if (opr1 < 0) return opr1; return 2 + opr1; case SB_REG_OPR_OPR: opr1 = x_opr_n_bytes (mra, 1); if (opr1 < 0) return opr1; opr2 = 0; if ((sb & 0x30) != 0x20) { opr2 = x_opr_n_bytes (mra, opr1 + 1); if (opr2 < 0) return opr2; } return 2 + opr1 + opr2; default: return 3; } /* not reached */ return -1; } static int mov_imm_opr_n_bytes (struct mem_read_abstraction_base *mra) { bfd_byte byte; int status = mra->read (mra, -1, 1, &byte); if (status < 0) return status; int size = byte - 0x0c + 1; int n = x_opr_n_bytes (mra, size); if (n < 0) return n; return size + n + 1; } static int mov_imm_opr (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; bfd_byte byte; int status = mra->read (mra, -1, 1, &byte); if (status < 0) return status; int size = byte - 0x0c + 1; uint32_t imm; if (decode_signed_value (mra, size, &imm)) return -1; op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; op = x_opr_decode (mra, size); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } static int ld_18bit_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; size_t size = 3; bfd_byte buffer[3]; int status = mra->read (mra, 0, 2, buffer + 1); if (status < 0) return status; status = mra->read (mra, -1, 1, buffer); if (status < 0) return status; buffer[0] = (buffer[0] & 0x30) >> 4; size_t i; uint32_t imm = 0; for (i = 0; i < size; ++i) { imm |= buffer[i] << (8 * (size - i - 1)); } op = create_immediate_operand (imm); if (op == NULL) return -1; operand[(*n_operands)++] = op; return 0; } /* Loop Primitives */ enum LP_MODE { LP_REG, LP_XY, LP_OPR }; struct lp { uint8_t mask; uint8_t value; enum LP_MODE mode; }; static const struct lp lp_mode[] = { {0x08, 0x00, LP_REG}, {0x0C, 0x08, LP_XY}, {0x0C, 0x0C, LP_OPR}, }; static int loop_prim_n_bytes (struct mem_read_abstraction_base *mra) { int mx = 0; uint8_t lb; int status = mra->read (mra, mx++, 1, &lb); if (status < 0) return status; enum LP_MODE mode = -1; size_t i; for (i = 0; i < sizeof (lp_mode) / sizeof (lp_mode[0]); ++i) { const struct lp *pb = lp_mode + i; if ((lb & pb->mask) == pb->value) { mode = pb->mode; break; } } if (mode == LP_OPR) { int n = x_opr_n_bytes (mra, mx); if (n < 0) return n; mx += n; } uint8_t rb; status = mra->read (mra, mx++, 1, &rb); if (status < 0) return status; if (rb & 0x80) mx++; return mx + 1; } static enum optr exg_sex_discrim (struct mem_read_abstraction_base *mra, enum optr hint ATTRIBUTE_UNUSED) { uint8_t eb; int status = mra->read (mra, 0, 1, &eb); enum optr operator = OP_INVALID; if (status < 0) return operator; struct operand *op0 = create_register_operand ((eb & 0xf0) >> 4); if (op0 == NULL) return -1; struct operand *op1 = create_register_operand (eb & 0xf); if (op1 == NULL) return -1; int reg0 = ((struct register_operand *) op0)->reg; int reg1 = ((struct register_operand *) op1)->reg; if (reg0 >= 0 && reg0 < S12Z_N_REGISTERS && reg1 >= 0 && reg1 < S12Z_N_REGISTERS) { const struct reg *r0 = registers + reg0; const struct reg *r1 = registers + reg1; operator = r0->bytes < r1->bytes ? OP_sex : OP_exg; } free (op0); free (op1); return operator; } static int exg_sex_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operands) { struct operand *op; uint8_t eb; int status = mra->read (mra, 0, 1, &eb); if (status < 0) return status; /* Ship out the operands. */ op = create_register_operand ((eb & 0xf0) >> 4); if (op == NULL) return -1; operands[(*n_operands)++] = op; op = create_register_operand (eb & 0xf); if (op == NULL) return -1; operands[(*n_operands)++] = op; return 0; } static enum optr loop_primitive_discrim (struct mem_read_abstraction_base *mra, enum optr hint ATTRIBUTE_UNUSED) { uint8_t lb; int status = mra->read (mra, 0, 1, &lb); if (status < 0) return OP_INVALID; enum optr opbase = (lb & 0x80) ? OP_dbNE : OP_tbNE; return opbase + ((lb & 0x70) >> 4); } static int loop_primitive_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operands) { struct operand *op; int n, offs = 1; uint8_t lb; int status = mra->read (mra, 0, 1, &lb); if (status < 0) return status; enum LP_MODE mode = -1; size_t i; for (i = 0; i < sizeof (lp_mode) / sizeof (lp_mode[0]); ++i) { const struct lp *pb = lp_mode + i; if ((lb & pb->mask) == pb->value) { mode = pb->mode; break; } } switch (mode) { case LP_REG: op = create_register_operand (lb & 0x07); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case LP_XY: op = create_register_operand ((lb & 0x01) + REG_X); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case LP_OPR: n = x_opr_n_bytes (mra, 1); if (n < 0) return n; offs += n; op = x_opr_decode_with_size (mra, 1, lb & 0x03); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; } return rel_15_7 (mra, offs + 1, n_operands, operands); } static enum optr shift_discrim (struct mem_read_abstraction_base *mra, enum optr hint ATTRIBUTE_UNUSED) { size_t i; uint8_t sb; int status = mra->read (mra, 0, 1, &sb); if (status < 0) return OP_INVALID; enum SB_DIR dir = (sb & 0x40) ? SB_LEFT : SB_RIGHT; enum SB_TYPE type = (sb & 0x80) ? SB_ARITHMETIC : SB_LOGICAL; enum SB_MODE mode = -1; for (i = 0; i < sizeof (sb_table) / sizeof (sb_table[0]); ++i) { const struct sb *sbe = sb_table + i; if ((sb & sbe->mask) == sbe->value) mode = sbe->mode; } if (mode == SB_ROT) return (dir == SB_LEFT) ? OP_rol : OP_ror; if (type == SB_LOGICAL) return (dir == SB_LEFT) ? OP_lsl : OP_lsr; return (dir == SB_LEFT) ? OP_asl : OP_asr; } static int shift_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operands) { struct operand *op; size_t i; uint8_t byte; int status = mra->read (mra, -1, 1, &byte); if (status < 0) return status; uint8_t sb; status = mra->read (mra, 0, 1, &sb); if (status < 0) return status; enum SB_MODE mode = -1; for (i = 0; i < sizeof (sb_table) / sizeof (sb_table[0]); ++i) { const struct sb *sbe = sb_table + i; if ((sb & sbe->mask) == sbe->value) mode = sbe->mode; } short osize = -1; switch (mode) { case SB_REG_OPR_EFF: case SB_ROT: case SB_REG_OPR_OPR: osize = sb & 0x03; break; case SB_OPR_N: { uint8_t xb; status = mra->read (mra, 1, 1, &xb); if (status < 0) return status; /* The size suffix is not printed if the OPR operand refers directly to a register, because the size is implied by the size of that register. */ if ((xb & 0xF8) != 0xB8) osize = sb & 0x03; } break; default: break; }; /* Destination register */ switch (mode) { case SB_REG_REG_N_EFF: case SB_REG_REG_N: op = create_register_operand (byte & 0x07); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case SB_REG_OPR_EFF: case SB_REG_OPR_OPR: op = create_register_operand (byte & 0x07); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case SB_ROT: op = x_opr_decode_with_size (mra, 1, osize); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; default: break; } /* Source register */ switch (mode) { case SB_REG_REG_N_EFF: case SB_REG_REG_N: op = create_register_operand_with_size (sb & 0x07, osize); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case SB_REG_OPR_OPR: op = x_opr_decode_with_size (mra, 1, osize); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; default: break; } /* 3rd arg */ switch (mode) { case SB_REG_OPR_EFF: case SB_OPR_N: op = x_opr_decode_with_size (mra, 1, osize); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case SB_REG_REG_N: { uint8_t xb; status = mra->read (mra, 1, 1, &xb); if (status < 0) return status; /* This case is slightly unusual. If XB matches the binary pattern 0111XXXX, then instead of interpreting this as a general OPR postbyte in the IMMe4 mode, the XB byte is interpreted in s special way. */ if ((xb & 0xF0) == 0x70) { if (byte & 0x10) { int shift = ((sb & 0x08) >> 3) | ((xb & 0x0f) << 1); op = create_immediate_operand (shift); if (op == NULL) return -1; operands[(*n_operands)++] = op; } else { /* This should not happen. */ abort (); } } else { op = x_opr_decode (mra, 1); if (op == NULL) return -1; operands[(*n_operands)++] = op; } } break; case SB_REG_OPR_OPR: { uint8_t xb; int n = x_opr_n_bytes (mra, 1); if (n < 0) return n; status = mra->read (mra, 1 + n, 1, &xb); if (status < 0) return status; if ((xb & 0xF0) == 0x70) { int imm = xb & 0x0F; imm <<= 1; imm |= (sb & 0x08) >> 3; op = create_immediate_operand (imm); if (op == NULL) return -1; operands[(*n_operands)++] = op; } else { op = x_opr_decode (mra, 1 + n); if (op == NULL) return -1; operands[(*n_operands)++] = op; } } break; default: break; } switch (mode) { case SB_REG_REG_N_EFF: case SB_REG_OPR_EFF: case SB_OPR_N: { int imm = (sb & 0x08) ? 2 : 1; op = create_immediate_operand (imm); if (op == NULL) return -1; operands[(*n_operands)++] = op; } break; default: break; } return 0; } static enum optr psh_pul_discrim (struct mem_read_abstraction_base *mra, enum optr hint ATTRIBUTE_UNUSED) { uint8_t byte; int status = mra->read (mra, 0, 1, &byte); if (status != 0) return OP_INVALID; return (byte & 0x80) ? OP_pull: OP_push; } static int psh_pul_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operand) { struct operand *op; uint8_t byte; int status = mra->read (mra, 0, 1, &byte); if (status != 0) return status; int bit; if (byte & 0x40) { if ((byte & 0x3F) == 0) { op = create_register_all16_operand (); if (op == NULL) return -1; operand[(*n_operands)++] = op; } else for (bit = 5; bit >= 0; --bit) { if (byte & (0x1 << bit)) { op = create_register_operand (oprregs2[bit]); if (op == NULL) return -1; operand[(*n_operands)++] = op; } } } else { if ((byte & 0x3F) == 0) { op = create_register_all_operand (); if (op == NULL) return -1; operand[(*n_operands)++] = op; } else for (bit = 5; bit >= 0; --bit) { if (byte & (0x1 << bit)) { op = create_register_operand (oprregs1[bit]); if (op == NULL) return -1; operand[(*n_operands)++] = op; } } } return 0; } static enum optr bit_field_discrim (struct mem_read_abstraction_base *mra, enum optr hint ATTRIBUTE_UNUSED) { int status; bfd_byte bb; status = mra->read (mra, 0, 1, &bb); if (status != 0) return OP_INVALID; return (bb & 0x80) ? OP_bfins : OP_bfext; } static int bit_field_decode (struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operands) { struct operand *op; int status; bfd_byte byte2; status = mra->read (mra, -1, 1, &byte2); if (status != 0) return status; bfd_byte bb; status = mra->read (mra, 0, 1, &bb); if (status != 0) return status; enum BB_MODE mode = -1; size_t i; const struct opr_bb *bbs = 0; for (i = 0; i < sizeof (bb_modes) / sizeof (bb_modes[0]); ++i) { bbs = bb_modes + i; if ((bb & bbs->mask) == bbs->value) { mode = bbs->mode; break; } } int reg1 = byte2 & 0x07; /* First operand */ switch (mode) { case BB_REG_REG_REG: case BB_REG_REG_IMM: case BB_REG_OPR_REG: case BB_REG_OPR_IMM: op = create_register_operand (reg1); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case BB_OPR_REG_REG: op = x_opr_decode_with_size (mra, 1, (bb >> 2) & 0x03); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case BB_OPR_REG_IMM: op = x_opr_decode_with_size (mra, 2, (bb >> 2) & 0x03); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; } /* Second operand */ switch (mode) { case BB_REG_REG_REG: case BB_REG_REG_IMM: { int reg_src = (bb >> 2) & 0x07; op = create_register_operand (reg_src); if (op == NULL) return -1; operands[(*n_operands)++] = op; } break; case BB_OPR_REG_REG: case BB_OPR_REG_IMM: { int reg_src = (byte2 & 0x07); op = create_register_operand (reg_src); if (op == NULL) return -1; operands[(*n_operands)++] = op; } break; case BB_REG_OPR_REG: op = x_opr_decode_with_size (mra, 1, (bb >> 2) & 0x03); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; case BB_REG_OPR_IMM: op = x_opr_decode_with_size (mra, 2, (bb >> 2) & 0x03); if (op == NULL) return -1; operands[(*n_operands)++] = op; break; } /* Third operand */ switch (mode) { case BB_REG_REG_REG: case BB_OPR_REG_REG: case BB_REG_OPR_REG: { int reg_parm = bb & 0x03; op = create_register_operand (reg_parm); if (op == NULL) return -1; operands[(*n_operands)++] = op; } break; case BB_REG_REG_IMM: case BB_OPR_REG_IMM: case BB_REG_OPR_IMM: { bfd_byte i1; status = mra->read (mra, 1, 1, &i1); if (status < 0) return status; int offset = i1 & 0x1f; int width = bb & 0x03; width <<= 3; width |= i1 >> 5; op = create_bitfield_operand (width, offset); if (op == NULL) return -1; operands[(*n_operands)++] = op; } break; } return 0; } /* Decode the next instruction at MRA, according to OPC. The operation to be performed is returned. The number of operands, will be placed in N_OPERANDS. The operands themselved into OPERANDS. */ static enum optr decode_operation (const struct opcode *opc, struct mem_read_abstraction_base *mra, int *n_operands, struct operand **operands) { enum optr op = opc->operator; if (opc->discriminator) { op = opc->discriminator (mra, opc->operator); if (op == OP_INVALID) return op; } if (opc->operands) if (opc->operands (mra, n_operands, operands) < 0) return OP_INVALID; if (opc->operands2) if (opc->operands2 (mra, n_operands, operands) < 0) return OP_INVALID; return op; } int decode_s12z (enum optr *myoperator, short *osize, int *n_operands, struct operand **operands, struct mem_read_abstraction_base *mra) { int n_bytes = 0; bfd_byte byte; int status = mra->read (mra, 0, 1, &byte); if (status < 0) return status; mra->advance (mra); const struct opcode *opc = page1 + byte; if (byte == PAGE2_PREBYTE) { /* Opcodes in page2 have an additional byte */ n_bytes++; bfd_byte byte2; status = mra->read (mra, 0, 1, &byte2); if (status < 0) return status; mra->advance (mra); opc = page2 + byte2; } *myoperator = decode_operation (opc, mra, n_operands, operands); *osize = opc->osize; /* Return the number of bytes in the instruction. */ if (*myoperator != OP_INVALID && opc->insn_bytes) { int n = opc->insn_bytes (mra); if (n < 0) return n; n_bytes += n; } else n_bytes += 1; return n_bytes; }