Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(952)

Unified Diff: opcodes/xgate-dis.c

Issue 11969036: Merge GDB 7.5.1 (Closed) Base URL: http://git.chromium.org/native_client/nacl-gdb.git@master
Patch Set: Created 7 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « opcodes/w65-dis.c ('k') | opcodes/xgate-opc.c » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: opcodes/xgate-dis.c
diff --git a/opcodes/xgate-dis.c b/opcodes/xgate-dis.c
new file mode 100644
index 0000000000000000000000000000000000000000..31f44d63f9e5de7000c267f9f2836f9efa519f18
--- /dev/null
+++ b/opcodes/xgate-dis.c
@@ -0,0 +1,396 @@
+/* xgate-dis.c -- Freescale XGATE disassembly
+ Copyright 2009, 2010, 2011, 2012
+ 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 "sysdep.h"
+#include <assert.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);
+static int macro_search (char *, char *);
+static struct decodeInfo * find_match (unsigned int);
+
+/* 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))
+ {
+ 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;
+}
+
+static 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;
+}
+
+static 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;
+}
« no previous file with comments | « opcodes/w65-dis.c ('k') | opcodes/xgate-opc.c » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698