/* xgate-dis.c -- Freescale XGATE disassembly Copyright 2009, 2010, 2011 Free Software Foundation, Inc. Written by Sean Keys (skeys@ipdatasys.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 #include "sysdep.h" #include "dis-asm.h" #include "opintl.h" #include "libiberty.h" #include "ansidecl.h" #include "opcode/xgate.h" #define XGATE_TWO_BYTES 0x02 #define XGATE_NINE_BITS 0x1FF #define XGATE_TEN_BITS 0x3FF #define XGATE_NINE_SIGNBIT 0x100 #define XGATE_TEN_SIGNBIT 0x200 /* Structures */ struct decodeInfo { unsigned int operMask; unsigned int operMasksRegisterBits; struct xgate_opcode *opcodePTR; }; /* Prototypes for local functions. */ static int print_insn( bfd_vma, struct disassemble_info *); static int read_memory( bfd_vma, bfd_byte*, int, struct disassemble_info *); static int ripBits(unsigned int *, int, struct xgate_opcode *, unsigned int); int macro_search(char *, char *); struct decodeInfo * find_match(unsigned int raw_code); /* statics */ static struct decodeInfo *decodeTable; static int initialized; static char previousOpName[10]; static unsigned int perviousBin; /* Disassemble one instruction at address 'memaddr'. Returns the number of bytes used by that instruction. */ static int print_insn (bfd_vma memaddr, struct disassemble_info* info) { int status; unsigned int raw_code; char *s = 0; long bytesRead = 0; int i = 0; struct xgate_opcode *opcodePTR = (struct xgate_opcode*) xgate_opcodes; struct decodeInfo *decodeTablePTR = 0; struct decodeInfo *decodePTR = 0; unsigned int operandRegisterBits = 0; signed int relAddr = 0; signed int operandOne = 0; signed int operandTwo = 0; bfd_byte buffer[4]; bfd_vma absAddress; unsigned int operMaskReg = 0; /* initialize our array of opcode masks and check them against our constant table */ if (!initialized) { decodeTable = xmalloc(sizeof(struct decodeInfo) * xgate_num_opcodes); for (i = 0, decodeTablePTR = decodeTable; i < xgate_num_opcodes; i++, decodeTablePTR++, opcodePTR++) { unsigned int bin = 0; unsigned int mask = 0; for (s = opcodePTR->format; *s; s++) { bin <<= 1; mask <<= 1; operandRegisterBits <<= 1; bin |= (*s == '1'); mask |= (*s == '0' || *s == '1'); operandRegisterBits |= (*s == 'r'); } /* asserting will uncover inconsistencies in our table */ assert( (s - opcodePTR->format) == 16 || (s - opcodePTR->format) == 32); assert(opcodePTR->bin_opcode == bin); decodeTablePTR->operMask = mask; decodeTablePTR->operMasksRegisterBits = operandRegisterBits; decodeTablePTR->opcodePTR = opcodePTR; } initialized = 1; } /* read 16 bits */ bytesRead += XGATE_TWO_BYTES; status = read_memory(memaddr, buffer, XGATE_TWO_BYTES, info); if (status == 0) { raw_code = buffer[0]; raw_code <<= 8; raw_code += buffer[1]; decodePTR = find_match(raw_code); if (decodePTR) { operMaskReg = decodePTR->operMasksRegisterBits; (*info->fprintf_func)(info->stream, "%s", decodePTR->opcodePTR->name); /* First we compare the shorthand format of the constraints. If we still are unable to pinpoint the operands we analyze the opcodes constraint string. */ switch (decodePTR->opcodePTR->sh_format) { case XG_R_C: (*info->fprintf_func)(info->stream, " R%x, CCR", (raw_code >> 8) & 0x7); break; case XG_C_R: (*info->fprintf_func)(info->stream, " CCR, R%x", (raw_code >> 8) & 0x7); break; case XG_R_P: (*info->fprintf_func)(info->stream, " R%x, PC", (raw_code >> 8) & 0x7); break; case XG_INH: break; case XG_R_R_R: if (!strcmp(decodePTR->opcodePTR->constraints, XGATE_OP_TRI)) { (*info->fprintf_func)(info->stream, " R%x, R%x, R%x", (raw_code >> 8) & 0x7, (raw_code >> 5) & 0x7, (raw_code >> 2) & 0x7); } else if (!strcmp(decodePTR->opcodePTR->constraints, XGATE_OP_IDR)) { if (raw_code & 0x01) { (*info->fprintf_func)(info->stream, " R%x, (R%x, R%x+)", (raw_code >> 8) & 0x7, (raw_code >> 5) & 0x7, (raw_code >> 2) & 0x7); } else if (raw_code & 0x02) { (*info->fprintf_func)(info->stream, " R%x, (R%x, -R%x)", (raw_code >> 8) & 0x7, (raw_code >> 5) & 0x7, (raw_code >> 2) & 0x7); } else { (*info->fprintf_func)(info->stream, " R%x, (R%x, R%x)", (raw_code >> 8) & 0x7, (raw_code >> 5) & 0x7, (raw_code >> 2) & 0x7); } } else { (*info->fprintf_func)(info->stream, " unhandled mode %s", decodePTR->opcodePTR->constraints); } break; case XG_R_R: if (!strcmp(decodePTR->opcodePTR->constraints, XGATE_OP_DYA_MON)) { operandOne = ripBits(&operMaskReg, 3, decodePTR->opcodePTR, raw_code); operandTwo = ripBits(&operMaskReg, 3, decodePTR->opcodePTR, raw_code); (*info->fprintf_func)(info->stream, " R%x, R%x", operandOne, operandTwo); } else if (!strcmp(decodePTR->opcodePTR->constraints, XGATE_OP_DYA)) { operandOne = ripBits(&operMaskReg, 3, opcodePTR, raw_code); operandTwo = ripBits(&operMaskReg, 3, opcodePTR, raw_code); (*info->fprintf_func)(info->stream, " R%x, R%x", operandOne, operandTwo); } else { (*info->fprintf_func)(info->stream, " unhandled mode %s", opcodePTR->constraints); } break; case XG_R_R_I: (*info->fprintf_func)(info->stream, " R%x, (R%x, #0x%x)", (raw_code >> 8) & 0x7, (raw_code >> 5) & 0x7, raw_code & 0x1f); break; case XG_R: operandOne = ripBits(&operMaskReg, 3, decodePTR->opcodePTR, raw_code); (*info->fprintf_func)(info->stream, " R%x", operandOne); break; case XG_I | XG_PCREL: if (!strcmp(decodePTR->opcodePTR->constraints, XGATE_OP_REL9)) { /* if address is negative handle it accordingly */ if (raw_code & XGATE_NINE_SIGNBIT) { relAddr = XGATE_NINE_BITS >> 1; /* clip sign bit */ relAddr = ~relAddr; /* make signed */ relAddr |= (raw_code & 0xFF) + 1; /* apply our value */ relAddr <<= 1; /* multiply by two as per processor docs */ } else { relAddr = raw_code & 0xff; relAddr = (relAddr << 1) + 2; } (*info->fprintf_func)(info->stream, " *%d", relAddr); (*info->fprintf_func)(info->stream, " Abs* 0x"); (*info->print_address_func)(memaddr + relAddr, info); } else if (!strcmp(decodePTR->opcodePTR->constraints, XGATE_OP_REL10)) { /* if address is negative handle it accordingly */ if (raw_code & XGATE_TEN_SIGNBIT) { relAddr = XGATE_TEN_BITS >> 1; /* clip sign bit */ relAddr = ~relAddr; /* make signed */ relAddr |= (raw_code & 0x1FF) + 1; /* apply our value */ relAddr <<= 1; /* multiply by two as per processor docs */ } else { relAddr = raw_code & 0x1FF; relAddr = (relAddr << 1) + 2; } (*info->fprintf_func)(info->stream, " *%d", relAddr); (*info->fprintf_func)(info->stream, " Abs* 0x"); (*info->print_address_func)(memaddr + relAddr, info); } else { (*info->fprintf_func)(info->stream, " Can't disassemble for mode) %s", decodePTR->opcodePTR->constraints); } break; case XG_R_I: if (!strcmp(decodePTR->opcodePTR->constraints, XGATE_OP_IMM4)) { (*info->fprintf_func)(info->stream, " R%x, #0x%02x", (raw_code >> 8) & 0x7, (raw_code >> 4) & 0xF); } else if (!strcmp(decodePTR->opcodePTR->constraints, XGATE_OP_IMM8)) { if (macro_search(decodePTR->opcodePTR->name, previousOpName) && previousOpName[0]) { absAddress = (0xFF & raw_code) << 8; absAddress |= perviousBin & 0xFF; (*info->fprintf_func)(info->stream, " R%x, #0x%02x Abs* 0x", (raw_code >> 8) & 0x7, raw_code & 0xff); (*info->print_address_func)(absAddress, info); previousOpName[0] = 0; } else { strcpy(previousOpName, decodePTR->opcodePTR->name); (*info->fprintf_func)(info->stream, " R%x, #0x%02x", (raw_code >> 8) & 0x7, raw_code & 0xff); } } else { (*info->fprintf_func)(info->stream, " Can't disassemble for mode %s", decodePTR->opcodePTR->constraints); } break; case XG_I: (*info->fprintf_func)(info->stream, " #0x%x", (raw_code >> 8) & 0x7); break; default: (*info->fprintf_func)(info->stream, "address mode not found\t %x", opcodePTR->bin_opcode); break; } perviousBin = raw_code; } else { (*info->fprintf_func)(info->stream, " unable to find opcode match #0%x", raw_code); } } return bytesRead; } int print_insn_xgate (bfd_vma memaddr, struct disassemble_info* info) { return print_insn (memaddr, info); } static int read_memory (bfd_vma memaddr, bfd_byte* buffer, int size, struct disassemble_info* info) { int status; status = (*info->read_memory_func) (memaddr, buffer, size, info); if (status != 0) { (*info->memory_error_func) (status, memaddr, info); return -1; } return 0; } static int ripBits(unsigned int *operandBitsRemaining, int numBitsRequested, struct xgate_opcode *opcodePTR, unsigned int memory) { unsigned int currentBit; int operand; int numBitsFound; for (operand = 0, numBitsFound = 0, currentBit = 1 << ((opcodePTR->size * 8) - 1); (numBitsFound < numBitsRequested) && currentBit; currentBit >>= 1) { if(currentBit & *operandBitsRemaining) { *operandBitsRemaining &= ~(currentBit); /* consume the current bit */ operand <<= 1; /* make room for our next bit */ numBitsFound++; operand |= (currentBit & memory) > 0; } } return operand; } int macro_search(char *currentName, char *lastName) { int i; int length = 0; char *where; for (i = 0; i < xgate_num_opcodes; i++) { where = strstr(xgate_opcodes[i].constraints, lastName); if (where) { length = strlen(where); } if (length) { where = strstr(xgate_opcodes[i].constraints, currentName); if (where) { length = strlen(where); return 1; } } } return 0; } struct decodeInfo* find_match(unsigned int raw_code) { struct decodeInfo *decodeTablePTR = 0; int i; for (i = 0, decodeTablePTR = decodeTable; i < xgate_num_opcodes; i++, decodeTablePTR++) { if ((raw_code & decodeTablePTR->operMask) == decodeTablePTR->opcodePTR->bin_opcode) { /* make sure we didn't run into a macro or alias */ if (decodeTablePTR->opcodePTR->cycles_min != 0) { return decodeTablePTR; break; } else { continue; } } } return 0; }