2012-05-03 13:12:08 +00:00
|
|
|
/* xgate-dis.c -- Freescale XGATE disassembly
|
2016-01-01 21:55:12 +10:30
|
|
|
Copyright (C) 2009-2016 Free Software Foundation, Inc.
|
2012-05-03 13:12:08 +00:00
|
|
|
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,
|
2012-05-17 15:13:28 +00:00
|
|
|
MA 02110-1301, USA. */
|
2012-05-03 13:12:08 +00:00
|
|
|
|
|
|
|
#include "sysdep.h"
|
2012-05-17 15:13:28 +00:00
|
|
|
#include <assert.h>
|
2012-05-03 13:12:08 +00:00
|
|
|
#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
|
|
|
|
|
2012-05-17 15:13:28 +00:00
|
|
|
/* Structures. */
|
|
|
|
struct decodeInfo
|
|
|
|
{
|
2012-05-03 13:12:08 +00:00
|
|
|
unsigned int operMask;
|
|
|
|
unsigned int operMasksRegisterBits;
|
|
|
|
struct xgate_opcode *opcodePTR;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Prototypes for local functions. */
|
2012-05-17 15:13:28 +00:00
|
|
|
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);
|
|
|
|
static int macro_search (char *, char *);
|
|
|
|
static struct decodeInfo * find_match (unsigned int);
|
2012-05-03 13:12:08 +00:00
|
|
|
|
2012-05-17 15:13:28 +00:00
|
|
|
/* Statics. */
|
2012-05-03 13:12:08 +00:00
|
|
|
static struct decodeInfo *decodeTable;
|
|
|
|
static int initialized;
|
|
|
|
static char previousOpName[10];
|
|
|
|
static unsigned int perviousBin;
|
|
|
|
|
|
|
|
/* Disassemble one instruction at address 'memaddr'. Returns the number
|
2012-05-17 15:13:28 +00:00
|
|
|
of bytes used by that instruction. */
|
|
|
|
|
2012-05-03 13:12:08 +00:00
|
|
|
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;
|
2012-05-17 15:13:28 +00:00
|
|
|
/* Initialize our array of opcode masks and check them against our constant
|
|
|
|
table. */
|
2012-05-03 13:12:08 +00:00
|
|
|
if (!initialized)
|
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
decodeTable = xmalloc (sizeof (struct decodeInfo) * xgate_num_opcodes);
|
2012-05-03 13:12:08 +00:00
|
|
|
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');
|
|
|
|
}
|
2012-05-17 15:13:28 +00:00
|
|
|
/* Asserting will uncover inconsistencies in our table. */
|
|
|
|
assert ((s - opcodePTR->format) == 16 || (s - opcodePTR->format) == 32);
|
|
|
|
assert (opcodePTR->bin_opcode == bin);
|
|
|
|
|
2012-05-03 13:12:08 +00:00
|
|
|
decodeTablePTR->operMask = mask;
|
|
|
|
decodeTablePTR->operMasksRegisterBits = operandRegisterBits;
|
|
|
|
decodeTablePTR->opcodePTR = opcodePTR;
|
|
|
|
}
|
|
|
|
initialized = 1;
|
|
|
|
}
|
2012-05-17 15:13:28 +00:00
|
|
|
|
|
|
|
/* Read 16 bits. */
|
2012-05-03 13:12:08 +00:00
|
|
|
bytesRead += XGATE_TWO_BYTES;
|
2012-05-17 15:13:28 +00:00
|
|
|
status = read_memory (memaddr, buffer, XGATE_TWO_BYTES, info);
|
2012-05-03 13:12:08 +00:00
|
|
|
if (status == 0)
|
|
|
|
{
|
|
|
|
raw_code = buffer[0];
|
|
|
|
raw_code <<= 8;
|
|
|
|
raw_code += buffer[1];
|
|
|
|
|
2012-05-17 15:13:28 +00:00
|
|
|
decodePTR = find_match (raw_code);
|
2012-05-03 13:12:08 +00:00
|
|
|
if (decodePTR)
|
|
|
|
{
|
|
|
|
operMaskReg = decodePTR->operMasksRegisterBits;
|
|
|
|
(*info->fprintf_func)(info->stream, "%s", decodePTR->opcodePTR->name);
|
2012-05-17 15:13:28 +00:00
|
|
|
|
2012-05-03 13:12:08 +00:00
|
|
|
/* First we compare the shorthand format of the constraints. If we
|
2013-10-11 04:55:42 +00:00
|
|
|
still are unable to pinpoint the operands
|
|
|
|
we analyze the opcodes constraint string. */
|
|
|
|
if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_MON_R_C))
|
|
|
|
{
|
|
|
|
(*info->fprintf_func)(info->stream, " R%x, CCR",
|
|
|
|
(raw_code >> 8) & 0x7);
|
|
|
|
}
|
|
|
|
else if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_MON_C_R))
|
|
|
|
{
|
|
|
|
(*info->fprintf_func)(info->stream, " CCR, R%x",
|
|
|
|
(raw_code >> 8) & 0x7);
|
|
|
|
}
|
|
|
|
else if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_MON_R_P))
|
|
|
|
{
|
|
|
|
(*info->fprintf_func)(info->stream, " R%x, PC",
|
|
|
|
(raw_code >> 8) & 0x7);
|
|
|
|
}
|
|
|
|
else 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 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 if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_IDO5))
|
|
|
|
{
|
|
|
|
(*info->fprintf_func)(info->stream, " R%x, (R%x, #0x%x)",
|
|
|
|
(raw_code >> 8) & 0x7, (raw_code >> 5) & 0x7, raw_code & 0x1f);
|
|
|
|
}
|
|
|
|
else if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_MON))
|
|
|
|
{
|
|
|
|
operandOne = ripBits (&operMaskReg, 3, decodePTR->opcodePTR,
|
|
|
|
raw_code);
|
|
|
|
(*info->fprintf_func)(info->stream, " R%x", operandOne);
|
|
|
|
}
|
|
|
|
else if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_REL9))
|
2012-05-03 13:12:08 +00:00
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
/* If address is negative handle it accordingly. */
|
2012-05-03 13:12:08 +00:00
|
|
|
if (raw_code & XGATE_NINE_SIGNBIT)
|
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
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. */
|
2012-05-03 13:12:08 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
relAddr = raw_code & 0xff;
|
|
|
|
relAddr = (relAddr << 1) + 2;
|
|
|
|
}
|
2013-10-11 04:55:42 +00:00
|
|
|
(*info->fprintf_func)(info->stream, " *%d", relAddr);
|
|
|
|
(*info->fprintf_func)(info->stream, " Abs* 0x");
|
|
|
|
(*info->print_address_func)(memaddr + relAddr, info);
|
|
|
|
}
|
2012-05-17 15:13:28 +00:00
|
|
|
else if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_REL10))
|
2012-05-03 13:12:08 +00:00
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
/* If address is negative handle it accordingly. */
|
2012-05-03 13:12:08 +00:00
|
|
|
if (raw_code & XGATE_TEN_SIGNBIT)
|
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
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. */
|
2012-05-03 13:12:08 +00:00
|
|
|
}
|
|
|
|
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);
|
|
|
|
}
|
2013-10-11 04:55:42 +00:00
|
|
|
else 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 if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_IMM3))
|
|
|
|
{
|
|
|
|
(*info->fprintf_func)(info->stream, " #0x%x",
|
|
|
|
(raw_code >> 8) & 0x7);
|
|
|
|
}
|
|
|
|
else if (!strcmp (decodePTR->opcodePTR->constraints, XGATE_OP_INH))
|
|
|
|
{
|
|
|
|
//
|
|
|
|
}
|
2012-05-03 13:12:08 +00:00
|
|
|
else
|
|
|
|
{
|
2013-10-11 04:55:42 +00:00
|
|
|
(*info->fprintf_func)(info->stream, " unhandled mode %s",
|
|
|
|
opcodePTR->constraints);
|
2012-05-03 13:12:08 +00:00
|
|
|
}
|
|
|
|
perviousBin = raw_code;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
(*info->fprintf_func)(info->stream,
|
2012-05-17 15:13:28 +00:00
|
|
|
" unable to find opcode match #0%x", raw_code);
|
2012-05-03 13:12:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
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
|
2012-05-17 15:13:28 +00:00
|
|
|
ripBits (unsigned int *operandBitsRemaining,
|
|
|
|
int numBitsRequested,
|
|
|
|
struct xgate_opcode *opcodePTR,
|
|
|
|
unsigned int memory)
|
2012-05-03 13:12:08 +00:00
|
|
|
{
|
|
|
|
unsigned int currentBit;
|
|
|
|
int operand;
|
|
|
|
int numBitsFound;
|
2012-05-17 15:13:28 +00:00
|
|
|
|
2012-05-03 13:12:08 +00:00
|
|
|
for (operand = 0, numBitsFound = 0, currentBit = 1
|
2012-05-17 15:13:28 +00:00
|
|
|
<< ((opcodePTR->size * 8) - 1);
|
|
|
|
(numBitsFound < numBitsRequested) && currentBit; currentBit >>= 1)
|
2012-05-03 13:12:08 +00:00
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
if (currentBit & *operandBitsRemaining)
|
|
|
|
{
|
|
|
|
*operandBitsRemaining &= ~(currentBit); /* Consume the current bit. */
|
|
|
|
operand <<= 1; /* Make room for our next bit. */
|
|
|
|
numBitsFound++;
|
|
|
|
operand |= (currentBit & memory) > 0;
|
|
|
|
}
|
|
|
|
}
|
2012-05-03 13:12:08 +00:00
|
|
|
return operand;
|
|
|
|
}
|
|
|
|
|
2012-05-17 15:13:28 +00:00
|
|
|
static int
|
|
|
|
macro_search (char *currentName, char *lastName)
|
2012-05-03 13:12:08 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int length = 0;
|
|
|
|
char *where;
|
2012-05-17 15:13:28 +00:00
|
|
|
|
2012-05-03 13:12:08 +00:00
|
|
|
for (i = 0; i < xgate_num_opcodes; i++)
|
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
where = strstr (xgate_opcodes[i].constraints, lastName);
|
|
|
|
|
2012-05-03 13:12:08 +00:00
|
|
|
if (where)
|
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
length = strlen (where);
|
2012-05-03 13:12:08 +00:00
|
|
|
}
|
|
|
|
if (length)
|
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
where = strstr (xgate_opcodes[i].constraints, currentName);
|
2012-05-03 13:12:08 +00:00
|
|
|
if (where)
|
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
length = strlen (where);
|
2012-05-03 13:12:08 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-05-17 15:13:28 +00:00
|
|
|
static struct decodeInfo *
|
|
|
|
find_match (unsigned int raw_code)
|
2012-05-03 13:12:08 +00:00
|
|
|
{
|
|
|
|
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)
|
|
|
|
{
|
2012-05-17 15:13:28 +00:00
|
|
|
/* Make sure we didn't run into a macro or alias. */
|
2012-05-03 13:12:08 +00:00
|
|
|
if (decodeTablePTR->opcodePTR->cycles_min != 0)
|
|
|
|
{
|
|
|
|
return decodeTablePTR;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
2012-05-17 15:13:28 +00:00
|
|
|
continue;
|
2012-05-03 13:12:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|