2012-05-03 21:12:08 +08:00
|
|
|
/* tc-xgate.c -- Assembler code for Freescale XGATE
|
2024-01-04 19:52:08 +08:00
|
|
|
Copyright (C) 2010-2024 Free Software Foundation, Inc.
|
2012-05-03 21:12:08 +08:00
|
|
|
Contributed by Sean Keys <skeys@ipdatasys.com>
|
|
|
|
|
|
|
|
This file is part of GAS, the GNU Assembler.
|
|
|
|
|
|
|
|
GAS 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.
|
|
|
|
|
|
|
|
GAS 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 GAS; see the file COPYING. If not, write to
|
|
|
|
the Free Software Foundation, 51 Franklin Street - Fifth Floor,
|
|
|
|
Boston, MA 02110-1301, USA. */
|
|
|
|
|
|
|
|
#include "as.h"
|
|
|
|
#include "safe-ctype.h"
|
|
|
|
#include "subsegs.h"
|
|
|
|
#include "opcode/xgate.h"
|
|
|
|
#include "dwarf2dbg.h"
|
|
|
|
#include "elf/xgate.h"
|
|
|
|
|
|
|
|
const char comment_chars[] = ";!";
|
|
|
|
const char line_comment_chars[] = "#*";
|
|
|
|
const char line_separator_chars[] = "";
|
|
|
|
const char EXP_CHARS[] = "eE";
|
|
|
|
const char FLT_CHARS[] = "dD";
|
|
|
|
|
2013-10-11 12:55:42 +08:00
|
|
|
/* Max opcodes per opcode handle. */
|
|
|
|
#define MAX_OPCODES 0x05
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
#define SIXTEENTH_BIT 0x8000
|
|
|
|
#define N_BITS_IN_WORD 16
|
|
|
|
#define MAX_NUM_OPERANDS 3
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
/* #define STATE_CONDITIONAL_BRANCH (1) */
|
|
|
|
#define STATE_PC_RELATIVE (2)
|
|
|
|
#define REGISTER_P(ptr) (ptr == 'r')
|
|
|
|
#define INCREMENT 01
|
|
|
|
#define DECREMENT 02
|
|
|
|
#define MAXREGISTER 07
|
|
|
|
#define MINREGISTER 00
|
|
|
|
|
|
|
|
#define OPTION_MMCU 'm'
|
|
|
|
|
|
|
|
/* This macro has no side-effects. */
|
|
|
|
#define ENCODE_RELAX(what,length) (((what) << 2) + (length))
|
|
|
|
|
2012-06-01 06:10:53 +08:00
|
|
|
/* Each unique opcode name has a handle. That handle may
|
|
|
|
contain pointers to opcodes with the same name but
|
|
|
|
different address modes. */
|
2012-05-03 21:12:08 +08:00
|
|
|
struct xgate_opcode_handle
|
|
|
|
{
|
|
|
|
int number_of_modes;
|
|
|
|
char *name;
|
|
|
|
struct xgate_opcode *opc0[MAX_OPCODES];
|
|
|
|
};
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
/* XGATE's registers all are 16-bit general purpose.
|
|
|
|
They are numbered according to the specifications. */
|
2012-07-06 03:37:52 +08:00
|
|
|
typedef enum register_id
|
|
|
|
{
|
|
|
|
REG_NONE = -1,
|
|
|
|
REG_R0 = 0,
|
|
|
|
REG_R1 = 1,
|
|
|
|
REG_R2 = 2,
|
|
|
|
REG_R3 = 3,
|
|
|
|
REG_R4 = 4,
|
|
|
|
REG_R5 = 5,
|
|
|
|
REG_R6 = 6,
|
|
|
|
REG_R7 = 7,
|
|
|
|
REG_PC = 8,
|
|
|
|
REG_CCR = 9
|
|
|
|
} register_id;
|
|
|
|
|
|
|
|
/* Operand Modifiers */
|
|
|
|
typedef enum op_modifiers
|
|
|
|
{
|
|
|
|
MOD_NONE = -1,
|
|
|
|
MOD_POSTINC = 1,
|
|
|
|
MOD_PREDEC = 2,
|
|
|
|
MOD_CONSTANT = 3,
|
|
|
|
MOD_LOAD_HIGH = 4,
|
|
|
|
MOD_LOAD_LOW = 5
|
|
|
|
}op_modifiers;
|
|
|
|
|
|
|
|
typedef struct s_operand
|
|
|
|
{
|
|
|
|
expressionS exp;
|
|
|
|
register_id reg;
|
|
|
|
op_modifiers mod;
|
|
|
|
} s_operand;
|
|
|
|
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
/* Forward declarations. */
|
|
|
|
static inline char *skip_whitespace (char *);
|
|
|
|
static void get_default_target (void);
|
|
|
|
static char *extract_word (char *, char *, int);
|
|
|
|
static struct xgate_opcode *xgate_find_match (struct xgate_opcode_handle *,
|
2013-10-11 12:55:42 +08:00
|
|
|
int, s_operand [], unsigned int);
|
2012-11-06 18:03:32 +08:00
|
|
|
static int cmp_opcode (struct xgate_opcode *, struct xgate_opcode *);
|
|
|
|
static void xgate_print_table (void);
|
|
|
|
static unsigned int xgate_get_operands (char *, s_operand []);
|
|
|
|
static register_id reg_name_search (char *);
|
2016-08-05 18:26:13 +08:00
|
|
|
static op_modifiers xgate_determine_modifiers (char **);
|
2012-11-06 18:03:32 +08:00
|
|
|
static void xgate_scan_operands (struct xgate_opcode *opcode, s_operand []);
|
|
|
|
static unsigned int xgate_parse_operand (struct xgate_opcode *, int *, int,
|
|
|
|
char **, s_operand);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2020-08-18 16:57:21 +08:00
|
|
|
static htab_t xgate_hash;
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
/* Previous opcode. */
|
|
|
|
static unsigned int prev = 0;
|
|
|
|
|
|
|
|
static unsigned char fixup_required = 0;
|
|
|
|
|
|
|
|
/* Used to enable clipping of 16 bit operands into 8 bit constraints. */
|
2012-11-06 18:03:32 +08:00
|
|
|
static unsigned char autoHiLo = 0;
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
static char oper_check;
|
|
|
|
static char flag_print_insn_syntax = 0;
|
|
|
|
static char flag_print_opcodes = 0;
|
|
|
|
|
|
|
|
static int current_architecture;
|
|
|
|
static const char *default_cpu;
|
|
|
|
|
|
|
|
/* ELF flags to set in the output file header. */
|
|
|
|
static int elf_flags = E_XGATE_F64;
|
|
|
|
|
|
|
|
/* This table describes how you change sizes for the various types of variable
|
|
|
|
size expressions. This version only supports two kinds. */
|
|
|
|
|
|
|
|
/* The fields are:
|
|
|
|
How far Forward this mode will reach.
|
|
|
|
How far Backward this mode will reach.
|
|
|
|
How many bytes this mode will add to the size of the frag.
|
|
|
|
Which mode to go to if the offset won't fit in this one. */
|
|
|
|
|
|
|
|
relax_typeS md_relax_table[] =
|
|
|
|
{
|
|
|
|
{1, 1, 0, 0}, /* First entries aren't used. */
|
|
|
|
{1, 1, 0, 0}, /* For no good reason except. */
|
|
|
|
{1, 1, 0, 0}, /* that the VAX doesn't either. */
|
|
|
|
{1, 1, 0, 0},
|
|
|
|
/* XGATE 9 and 10 bit pc rel todo complete and test */
|
|
|
|
/*{(511), (-512), 0, ENCODE_RELAX (STATE_PC_RELATIVE, STATE_WORD)},
|
|
|
|
{(1023), (-1024), 0, ENCODE_RELAX (STATE_PC_RELATIVE, STATE_WORD)}, */
|
|
|
|
{0, 0, 0, 0}
|
|
|
|
};
|
|
|
|
|
|
|
|
/* This table describes all the machine specific pseudo-ops the assembler
|
|
|
|
has to support. The fields are: pseudo-op name without dot function to
|
|
|
|
call to execute this pseudo-op Integer arg to pass to the function. */
|
|
|
|
const pseudo_typeS md_pseudo_table[] =
|
|
|
|
{
|
|
|
|
/* The following pseudo-ops are supported for MRI compatibility. */
|
|
|
|
{0, 0, 0}
|
|
|
|
};
|
|
|
|
|
|
|
|
const char *md_shortopts = "m:";
|
|
|
|
|
|
|
|
struct option md_longopts[] =
|
|
|
|
{
|
|
|
|
#define OPTION_PRINT_INSN_SYNTAX (OPTION_MD_BASE + 0)
|
|
|
|
{ "print-insn-syntax", no_argument, NULL, OPTION_PRINT_INSN_SYNTAX },
|
|
|
|
|
|
|
|
#define OPTION_PRINT_OPCODES (OPTION_MD_BASE + 1)
|
|
|
|
{ "print-opcodes", no_argument, NULL, OPTION_PRINT_OPCODES },
|
|
|
|
|
|
|
|
#define OPTION_GENERATE_EXAMPLE (OPTION_MD_BASE + 2)
|
|
|
|
{ "generate-example", no_argument, NULL, OPTION_GENERATE_EXAMPLE },
|
|
|
|
|
|
|
|
#define OPTION_MSHORT (OPTION_MD_BASE + 3)
|
|
|
|
{ "mshort", no_argument, NULL, OPTION_MSHORT },
|
|
|
|
|
|
|
|
#define OPTION_MLONG (OPTION_MD_BASE + 4)
|
|
|
|
{ "mlong", no_argument, NULL, OPTION_MLONG },
|
|
|
|
|
|
|
|
#define OPTION_MSHORT_DOUBLE (OPTION_MD_BASE + 5)
|
|
|
|
{ "mshort-double", no_argument, NULL, OPTION_MSHORT_DOUBLE },
|
|
|
|
|
|
|
|
#define OPTION_MLONG_DOUBLE (OPTION_MD_BASE + 6)
|
|
|
|
{ "mlong-double", no_argument, NULL, OPTION_MLONG_DOUBLE },
|
|
|
|
|
|
|
|
{ NULL, no_argument, NULL, 0 }
|
|
|
|
};
|
|
|
|
|
2016-08-05 18:26:13 +08:00
|
|
|
size_t md_longopts_size = sizeof (md_longopts);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2016-04-01 20:07:50 +08:00
|
|
|
const char *
|
2012-05-03 21:12:08 +08:00
|
|
|
md_atof (int type, char *litP, int *sizeP)
|
|
|
|
{
|
Use bool in gas
* as.h (POISON_BFD_BOOLEAN): Define.
* as.c, * as.h, * atof-generic.c, * config/atof-ieee.c,
* config/bfin-aux.h, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-elf.h, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h,
* config/tc-avr.c, * config/tc-avr.h, * config/tc-bfin.c,
* config/tc-bfin.h, * config/tc-bpf.c, * config/tc-cris.c,
* config/tc-csky.c, * config/tc-csky.h, * config/tc-d10v.c,
* config/tc-d10v.h, * config/tc-d30v.c, * config/tc-d30v.h,
* config/tc-dlx.c, * config/tc-dlx.h, * config/tc-epiphany.c,
* config/tc-epiphany.h, * config/tc-fr30.c, * config/tc-fr30.h,
* config/tc-frv.c, * config/tc-frv.h, * config/tc-ft32.c,
* config/tc-ft32.h, * config/tc-h8300.c, * config/tc-hppa.c,
* config/tc-i386-intel.c, * config/tc-i386.c, * config/tc-ia64.c,
* config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-iq2000.h,
* config/tc-lm32.c, * config/tc-lm32.h, * config/tc-m32c.c,
* config/tc-m32c.h, * config/tc-m32r.c, * config/tc-m32r.h,
* config/tc-m68hc11.c, * config/tc-m68k.c, * config/tc-mcore.c,
* config/tc-mcore.h, * config/tc-mep.c, * config/tc-mep.h,
* config/tc-metag.c, * config/tc-metag.h,
* config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mips.h,
* config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c,
* config/tc-mn10300.h, * config/tc-moxie.c, * config/tc-msp430.c,
* config/tc-msp430.h, * config/tc-mt.c, * config/tc-mt.h,
* config/tc-nds32.c, * config/tc-nds32.h, * config/tc-nios2.c,
* config/tc-ns32k.c, * config/tc-or1k.c, * config/tc-or1k.h,
* config/tc-pdp11.c, * config/tc-ppc.c, * config/tc-pru.c,
* config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h,
* config/tc-rx.c, * config/tc-rx.h, * config/tc-s12z.c,
* config/tc-s12z.h, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score.h, * config/tc-score7.c, * config/tc-sh.c,
* config/tc-sh.h, * config/tc-spu.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-tilegx.c,
* config/tc-tilepro.c, * config/tc-v850.c, * config/tc-v850.h,
* config/tc-visium.c, * config/tc-visium.h, * config/tc-wasm32.c,
* config/tc-wasm32.h, * config/tc-xc16x.c, * config/tc-xgate.c,
* config/tc-xstormy16.c, * config/tc-xstormy16.h,
* config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c,
* config/tc-z8k.c, * config/xtensa-istack.h,
* config/xtensa-relax.c, * config/xtensa-relax.h, * dw2gencfi.c,
* dwarf2dbg.c, * dwarf2dbg.h, * expr.c, * expr.h, * frags.c,
* frags.h, * listing.c, * macro.c, * output-file.c, * read.c,
* read.h, * stabs.c, * symbols.c, * write.c: Replace bfd_boolean
with bool, FALSE with false, and TRUE with true.
2021-03-31 08:12:05 +08:00
|
|
|
return ieee_md_atof (type, litP, sizeP, true);
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2016-02-27 22:35:32 +08:00
|
|
|
md_parse_option (int c, const char *arg)
|
2012-05-03 21:12:08 +08:00
|
|
|
{
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case OPTION_MMCU:
|
|
|
|
if (strcasecmp (arg, "v1") == 0)
|
|
|
|
current_architecture = XGATE_V1;
|
|
|
|
else if (strcasecmp (arg, "v2") == 0)
|
|
|
|
current_architecture = XGATE_V2;
|
|
|
|
else if (strcasecmp (arg, "v3") == 0)
|
|
|
|
current_architecture = XGATE_V3;
|
|
|
|
else
|
2017-01-23 23:23:07 +08:00
|
|
|
as_bad (_("architecture variant invalid"));
|
2012-05-03 21:12:08 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_PRINT_INSN_SYNTAX:
|
|
|
|
flag_print_insn_syntax = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_PRINT_OPCODES:
|
|
|
|
flag_print_opcodes = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_GENERATE_EXAMPLE:
|
|
|
|
flag_print_opcodes = 2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_MSHORT:
|
|
|
|
elf_flags &= ~E_XGATE_I32;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_MLONG:
|
|
|
|
elf_flags |= E_XGATE_I32;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_MSHORT_DOUBLE:
|
|
|
|
elf_flags &= ~E_XGATE_F64;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OPTION_MLONG_DOUBLE:
|
|
|
|
elf_flags |= E_XGATE_F64;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
xgate_arch_format (void)
|
|
|
|
{
|
|
|
|
get_default_target ();
|
|
|
|
|
|
|
|
if (current_architecture & cpuxgate)
|
|
|
|
return "elf32-xgate";
|
|
|
|
|
|
|
|
return "error";
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
get_default_target (void)
|
|
|
|
{
|
|
|
|
const bfd_target *target;
|
|
|
|
bfd abfd;
|
|
|
|
|
|
|
|
if (current_architecture != 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
default_cpu = "unknown";
|
|
|
|
target = bfd_find_target (0, &abfd);
|
|
|
|
|
|
|
|
if (target && target->name)
|
|
|
|
{
|
|
|
|
if (strcmp (target->name, "elf32-xgate") == 0)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
current_architecture = cpuxgate;
|
|
|
|
default_cpu = "XGATE V1";
|
|
|
|
return;
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
as_bad (_("Default target `%s' is not supported."), target->name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
md_begin (void)
|
|
|
|
{
|
|
|
|
struct xgate_opcode *xgate_opcode_ptr = NULL;
|
|
|
|
struct xgate_opcode *xgate_op_table = NULL;
|
|
|
|
struct xgate_opcode_handle *op_handles = 0;
|
2016-02-26 00:55:21 +08:00
|
|
|
const char *prev_op_name = 0;
|
2012-05-03 21:12:08 +08:00
|
|
|
int handle_enum = 0;
|
2012-06-01 06:10:53 +08:00
|
|
|
int number_of_op_handles = 0;
|
2012-05-03 21:12:08 +08:00
|
|
|
int i, j = 0;
|
|
|
|
|
|
|
|
/* Create a local copy of our opcode table
|
|
|
|
including an extra line for NULL termination. */
|
2016-04-07 04:26:46 +08:00
|
|
|
xgate_op_table = XNEWVEC (struct xgate_opcode, xgate_num_opcodes);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
memset (xgate_op_table, 0,
|
2016-08-05 18:26:13 +08:00
|
|
|
sizeof (struct xgate_opcode) * (xgate_num_opcodes));
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
for (xgate_opcode_ptr = (struct xgate_opcode*) xgate_opcodes, i = 0;
|
2012-11-06 18:03:32 +08:00
|
|
|
i < xgate_num_opcodes; i++)
|
2012-05-03 21:12:08 +08:00
|
|
|
xgate_op_table[i] = xgate_opcode_ptr[i];
|
|
|
|
|
2016-08-05 18:26:13 +08:00
|
|
|
qsort (xgate_op_table, xgate_num_opcodes, sizeof (struct xgate_opcode),
|
2012-05-03 21:12:08 +08:00
|
|
|
(int (*)(const void *, const void *)) cmp_opcode);
|
|
|
|
|
|
|
|
/* Calculate number of handles since this will be
|
|
|
|
smaller than the raw number of opcodes in the table. */
|
2012-06-01 06:10:53 +08:00
|
|
|
prev_op_name = "";
|
|
|
|
for (xgate_opcode_ptr = xgate_op_table, i = 0; i < xgate_num_opcodes;
|
2012-11-06 18:03:32 +08:00
|
|
|
xgate_opcode_ptr++, i++)
|
2012-05-03 21:12:08 +08:00
|
|
|
{
|
2012-06-01 06:10:53 +08:00
|
|
|
if (strcmp (prev_op_name, xgate_opcode_ptr->name))
|
2012-11-06 18:03:32 +08:00
|
|
|
number_of_op_handles++;
|
2012-05-03 21:12:08 +08:00
|
|
|
prev_op_name = xgate_opcode_ptr->name;
|
|
|
|
}
|
|
|
|
|
2016-04-07 04:26:46 +08:00
|
|
|
op_handles = XNEWVEC (struct xgate_opcode_handle, number_of_op_handles);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2012-06-01 06:10:53 +08:00
|
|
|
/* Insert unique opcode names into hash table, aliasing duplicates. */
|
2020-08-18 16:57:21 +08:00
|
|
|
xgate_hash = str_htab_create ();
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2012-06-01 06:10:53 +08:00
|
|
|
prev_op_name = "";
|
2012-05-03 21:12:08 +08:00
|
|
|
for (xgate_opcode_ptr = xgate_op_table, i = 0, j = 0; i < xgate_num_opcodes;
|
2012-11-06 18:03:32 +08:00
|
|
|
i++, xgate_opcode_ptr++)
|
2012-05-03 21:12:08 +08:00
|
|
|
{
|
2012-06-01 06:10:53 +08:00
|
|
|
if (!strcmp (prev_op_name, xgate_opcode_ptr->name))
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
handle_enum++;
|
|
|
|
op_handles[j].opc0[handle_enum] = xgate_opcode_ptr;
|
|
|
|
}
|
2012-06-01 06:10:53 +08:00
|
|
|
else
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
handle_enum = 0;
|
|
|
|
if (i)
|
|
|
|
j++;
|
|
|
|
op_handles[j].name = xgate_opcode_ptr->name;
|
|
|
|
op_handles[j].opc0[0] = xgate_opcode_ptr;
|
PR26513, 629310abec breaks assembling PowerPC Linux kernels
Inserting with replacement is wrong for some gas hash table uses.
This patch implements an htab_insert that conditionally replaces, and
similarly for str_hash_insert. str_hash_insert with replace=0 is
roughly equivalent to the older hash_insert, and str_hash_insert with
replace=1 to the older hash_jam, but return values are different. I
found it useful to know whether the slot was occupied prior to
inserting/replacing. I've also reinstated the fatal errors on messing
up opcode tables with duplicates.
PR 26513
* hash.h (htab_insert): Update prototype and comment.
(struct string_tuple): Make "value" a const void*.
(string_tuple_alloc): Likewise.
(str_hash_find, str_hash_find_n): Cast returned value.
(str_hash_insert): Add "replace" parameter, and return slot pointer.
Free alloc'd element when not inserted.
* hash.c (htab_insert): Likewise. Return slot when element exists,
otherwise return NULL.
* read.c (pop_insert): Insert into hash table without first searching.
* config/tc-avr.c (md_begin): Likewise.
* config/tc-msp430.c (md_begin): Likewise.
* config/tc-nds32.c (nds32_init_nds32_pseudo_opcodes): Likewise.
* config/tc-v850.c (md_begin): Likewise.
* macro.c (do_formals, define_macro, macro_expand_body): Likewise.
(delete_macro): Delete from hash table.
* config/tc-tic54x.c (subsym_create_or_replace): Correct logic.
* symbols.c (local_symbol_make, symbol_table_insert): Allow
replacement of hash table entries.
* config/obj-coff-seh.c (seh_hash_insert): Likewise.
* config/obj-coff.c (tag_insert): Likewise.
* config/tc-iq2000.c (iq2000_add_macro): Likewise.
* config/tc-m68k.c (md_begin): Likewise for aliases.
* config/tc-tic4x.c (tic4x_asg): Likewise.
* config/tc-tic6x.c (md_begin): Likewise.
* dw2gencfi.c (dwcfi_hash_find_or_make): Disallow replacement of
hash table entries.
* ecoff.c (add_string, get_tag): Likewise.
* macro.c (expand_irp): Likewise.
* config/obj-elf.c (build_additional_section_info): Likewise.
* config/tc-aarch64.c (insert_reg_alias): Likewise.
(checked_hash_insert): Likewise.
* config/tc-alpha.c (get_alpha_reloc_tag, md_begin): Likewise.
* config/tc-arc.c (arc_insert_opcode, declare_register): Likewise.
(declare_addrtype, md_begin, arc_extcorereg): Likewise.
* config/tc-arm.c (insert_reg_alias): Likewise.
(arm_tc_equal_in_insn, md_begin): Likewise.
* config/tc-cr16.c (initialise_reg_hash_table, md_begin): Likewise.
* config/tc-cris.c (md_begin): Likewise.
* config/tc-crx.c (md_begin): Likewise.
* config/tc-csky.c (md_begin): Likewise.
* config/tc-d10v.c (md_begin): Likewise.
* config/tc-dlx.c (md_begin): Likewise.
* config/tc-ft32.c (md_begin): Likewise.
* config/tc-h8300.c (md_begin): Likewise.
* config/tc-hppa.c (md_begin): Likewise.
* config/tc-i386.c (md_begin): Likewise.
* config/tc-ia64.c (dot_rot, dot_entry, declare_register): Likewise.
(md_begin, dot_alias): Likewise.
* config/tc-m68hc11.c (md_begin): Likewise.
* config/tc-m68k.c (md_begin): Likewise.
* config/tc-mcore.c (md_begin): Likewise.
* config/tc-microblaze.c (md_begin): Likewise.
* config/tc-mips.c (md_begin): Likewise.
* config/tc-mmix.c (md_begin): Likewise.
* config/tc-mn10200.c (md_begin): Likewise.
* config/tc-mn10300.c (md_begin): Likewise.
* config/tc-moxie.c (md_begin): Likewise.
* config/tc-nds32.c (nds32_relax_hint, md_begin): Likewise.
* config/tc-nios2.c (md_begin): Likewise.
* config/tc-ns32k.c (md_begin): Likewise.
* config/tc-pdp11.c (md_begin): Likewise.
* config/tc-pj.c (fake_opcode, md_begin): Likewise.
* config/tc-ppc.c (ppc_setup_opcodes): Likewise.
* config/tc-pru.c (md_begin): Likewise.
* config/tc-riscv.c (init_ext_version_hash): Likewise.
(init_opcode_names_hash, hash_reg_name, init_opcode_hash): Likewise.
(riscv_init_csr_hash): Likewise.
* config/tc-s390.c (s390_setup_opcodes, md_begin): Likewise.
* config/tc-score.c (s3_insert_reg): Likewise.
(s3_build_score_ops_hsh, s3_build_dependency_insn_hsh): Likewise.
* config/tc-score7.c (s7_build_score_ops_hsh): Likewise.
(s7_build_dependency_insn_hsh, s7_insert_reg): Likewise.
* config/tc-sh.c (md_begin): Likewise.
* config/tc-sparc.c (md_begin): Likewise.
* config/tc-spu.c (md_begin): Likewise.
* config/tc-tic30.c (md_begin): Likewise.
* config/tc-tic4x.c (tic4x_inst_insert): Likewise.
* config/tc-tic54x.c (stag_add_field_symbols, md_begin): Likewise.
(tic54x_endstruct, tic54x_var, tic54x_macro_info): Likewise.
(subsym_substitute): Likewise.
* config/tc-tilegx.c (md_begin): Likewise.
* config/tc-tilepro.c (md_begin): Likewise.
* config/tc-vax.c (vip_begin): Likewise.
* config/tc-wasm32.c (md_begin): Likewise.
* config/tc-xgate.c (md_begin): Likewise.
* config/tc-z8k.c (md_begin): Likewise.
* testsuite/gas/ppc/dcbt.d,
* testsuite/gas/ppc/dcbt.s: New test.
* testsuite/gas/ppc/ppc.exp: Run it.
* ecoff.c (add_string): Report fatal error on duplicates.
* config/tc-alpha.c (md_begin): Likewise.
* config/tc-arc.c (arc_insert_opcode, declare_register): Likewise.
(declare_addrtype, md_begin, arc_extcorereg): Likewise.
* config/tc-cr16.c (initialise_reg_hash_table, md_begin): Likewise.
* config/tc-cris.c (md_begin): Likewise.
* config/tc-crx.c (md_begin): Likewise.
* config/tc-dlx.c (md_begin): Likewise.
* config/tc-hppa.c (md_begin): Likewise.
* config/tc-i386.c (md_begin): Likewise.
* config/tc-ia64.c (dot_rot, dot_entry, declare_register): Likewise.
(md_begin): Likewise.
* config/tc-m68k.c (md_begin): Likewise.
* config/tc-mips.c (md_begin): Likewise.
* config/tc-nios2.c (md_begin): Likewise.
* config/tc-ns32k.c (md_begin): Likewise.
* config/tc-ppc.c (ppc_setup_opcodes): Likewise.
* config/tc-pru.c (md_begin): Likewise.
* config/tc-riscv.c (init_ext_version_hash): Likewise.
(init_opcode_names_hash, hash_reg_name, init_opcode_hash): Likewise.
* config/tc-s390.c (s390_setup_opcodes, md_begin): Likewise.
* config/tc-sparc.c (md_begin): Likewise.
* config/tc-tic30.c (md_begin): Likewise.
* config/tc-tic4x.c (tic4x_inst_insert): Likewise.
* config/tc-tilegx.c (md_begin): Likewise.
* config/tc-tilepro.c (md_begin): Likewise.
* config/tc-vax.c (vip_begin): Likewise.
* config/tc-alpha.c,
* config/tc-arm.c,
* config/tc-avr.c,
* config/tc-cr16.c,
* config/tc-csky.c,
* config/tc-i386.c,
* config/tc-m68hc11.c,
* config/tc-m68k.c,
* config/tc-microblaze.c,
* config/tc-ns32k.c,
* config/tc-pj.c,
* config/tc-ppc.c,
* config/tc-score.c,
* config/tc-score7.c,
* config/tc-tic4x.c,
* config/tc-tic54x.c,
* config/tc-tilegx.c,
* config/tc-tilepro.c,
* config/tc-xgate.c: Formatting.
2020-08-22 16:29:57 +08:00
|
|
|
str_hash_insert (xgate_hash, op_handles[j].name, &op_handles[j], 0);
|
2012-11-06 18:03:32 +08:00
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
op_handles[j].number_of_modes = handle_enum;
|
|
|
|
prev_op_name = op_handles[j].name;
|
|
|
|
}
|
|
|
|
|
2013-01-11 06:15:05 +08:00
|
|
|
if (flag_print_opcodes)
|
|
|
|
{
|
2013-01-12 05:12:34 +08:00
|
|
|
xgate_print_table ();
|
2013-01-11 06:15:05 +08:00
|
|
|
exit (EXIT_SUCCESS);
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
xgate_init_after_args (void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
md_show_usage (FILE * stream)
|
|
|
|
{
|
|
|
|
get_default_target ();
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
fprintf (stream,
|
2012-05-03 21:12:08 +08:00
|
|
|
_("\
|
2012-11-06 18:03:32 +08:00
|
|
|
Freescale XGATE co-processor options:\n\
|
|
|
|
-mshort use 16-bit int ABI (default)\n\
|
|
|
|
-mlong use 32-bit int ABI\n\
|
|
|
|
-mshort-double use 32-bit double ABI\n\
|
|
|
|
-mlong-double use 64-bit double ABI (default)\n\
|
2017-01-23 23:23:07 +08:00
|
|
|
--mxgate specify the processor variant [default %s]\n\
|
2012-11-06 18:03:32 +08:00
|
|
|
--print-insn-syntax print the syntax of instruction in case of error\n\
|
|
|
|
--print-opcodes print the list of instructions with syntax\n\
|
|
|
|
--generate-example generate an example of each instruction"),
|
2012-05-03 21:12:08 +08:00
|
|
|
default_cpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
enum bfd_architecture
|
|
|
|
xgate_arch (void)
|
|
|
|
{
|
|
|
|
get_default_target ();
|
|
|
|
return bfd_arch_xgate;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
xgate_mach (void)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
static void
|
2012-05-03 21:12:08 +08:00
|
|
|
xgate_print_syntax (char *name)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < xgate_num_opcodes; i++)
|
|
|
|
{
|
|
|
|
if (!strcmp (xgate_opcodes[i].name, name))
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_IDR))
|
|
|
|
printf ("\tFormat is %s\tRx, Rx, Rx+|-Rx|Rx\n",
|
2012-05-03 21:12:08 +08:00
|
|
|
xgate_opcodes[i].name);
|
2012-11-06 18:03:32 +08:00
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_INH))
|
|
|
|
printf ("\tFormat is %s\n", xgate_opcodes[i].name);
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_TRI))
|
|
|
|
printf ("\tFormat is %s\tRx, Rx, Rx\n", xgate_opcodes[i].name);
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_DYA))
|
|
|
|
printf ("\tFormat is %s\tRx, Rx\n", xgate_opcodes[i].name);
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_IMM3))
|
|
|
|
printf ("\tFormat is %s\t<3-bit value>\n", xgate_opcodes[i].name);
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_IMM4))
|
|
|
|
printf ("\tFormat is %s\t<4 -bit value>\n", xgate_opcodes[i].name);
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_IMM8))
|
|
|
|
printf ("\tFormat is %s\tRx, <8-bit value>\n",
|
2012-05-03 21:12:08 +08:00
|
|
|
xgate_opcodes[i].name);
|
2012-11-06 18:03:32 +08:00
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_IMM16))
|
|
|
|
printf ("\tFormat is %s\tRx, <16-bit value>\n",
|
2012-05-03 21:12:08 +08:00
|
|
|
xgate_opcodes[i].name);
|
2012-11-06 18:03:32 +08:00
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_MON_R_C))
|
|
|
|
printf ("\tFormat is %s\tRx, CCR\n", xgate_opcodes[i].name);
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_MON_C_R))
|
|
|
|
printf ("\tFormat is %s\tCCR, Rx\n", xgate_opcodes[i].name);
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_MON_R_P))
|
|
|
|
printf ("\tFormat is %s\tRx, PC\n", xgate_opcodes[i].name);
|
|
|
|
if (!strcmp (xgate_opcodes[i].constraints, XGATE_OP_IMM16mLDW))
|
|
|
|
printf ("\tFormat is %s\tRx, <16-bit value>\n",
|
2012-05-03 21:12:08 +08:00
|
|
|
xgate_opcodes[i].name);
|
2012-11-06 18:03:32 +08:00
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
static void
|
2012-05-03 21:12:08 +08:00
|
|
|
xgate_print_table (void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < xgate_num_opcodes; i++)
|
|
|
|
xgate_print_syntax (xgate_opcodes[i].name);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
xgate_listing_header (void)
|
|
|
|
{
|
|
|
|
if (current_architecture & cpuxgate)
|
|
|
|
return "XGATE GAS ";
|
|
|
|
|
|
|
|
return "ERROR MC9S12X GAS ";
|
|
|
|
}
|
|
|
|
|
|
|
|
symbolS *
|
|
|
|
md_undefined_symbol (char *name ATTRIBUTE_UNUSED)
|
|
|
|
{
|
2016-08-05 18:26:13 +08:00
|
|
|
return NULL;
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* GAS will call this function for each section at the end of the assembly,
|
|
|
|
to permit the CPU backend to adjust the alignment of a section. */
|
|
|
|
|
|
|
|
valueT
|
|
|
|
md_section_align (asection * seg, valueT addr)
|
|
|
|
{
|
bfd_section_* macros
This large patch removes the unnecessary bfd parameter from various
bfd section macros and functions. The bfd is hardly ever used and if
needed for the bfd_set_section_* or bfd_rename_section functions can
be found via section->owner except for the com, und, abs, and ind
std_section special sections. Those sections shouldn't be modified
anyway.
The patch also removes various bfd_get_section_<field> macros,
replacing their use with bfd_section_<field>, and adds
bfd_set_section_lma. I've also fixed a minor bug in gas where
compressed section renaming was done directly rather than calling
bfd_rename_section. This would have broken bfd_get_section_by_name
and similar functions, but that hardly mattered at such a late stage
in gas processing.
bfd/
* bfd-in.h (bfd_get_section_name, bfd_get_section_vma),
(bfd_get_section_lma, bfd_get_section_alignment),
(bfd_get_section_size, bfd_get_section_flags),
(bfd_get_section_userdata): Delete.
(bfd_section_name, bfd_section_size, bfd_section_vma),
(bfd_section_lma, bfd_section_alignment): Lose bfd parameter.
(bfd_section_flags, bfd_section_userdata): New.
(bfd_is_com_section): Rename parameter.
* section.c (bfd_set_section_userdata, bfd_set_section_vma),
(bfd_set_section_alignment, bfd_set_section_flags, bfd_rename_section),
(bfd_set_section_size): Delete bfd parameter, rename section parameter.
(bfd_set_section_lma): New.
* bfd-in2.h: Regenerate.
* mach-o.c (bfd_mach_o_init_section_from_mach_o): Delete bfd param,
update callers.
* aoutx.h, * bfd.c, * coff-alpha.c, * coff-arm.c, * coff-mips.c,
* coff64-rs6000.c, * coffcode.h, * coffgen.c, * cofflink.c,
* compress.c, * ecoff.c, * elf-eh-frame.c, * elf-hppa.h,
* elf-ifunc.c, * elf-m10200.c, * elf-m10300.c, * elf-properties.c,
* elf-s390-common.c, * elf-vxworks.c, * elf.c, * elf32-arc.c,
* elf32-arm.c, * elf32-avr.c, * elf32-bfin.c, * elf32-cr16.c,
* elf32-cr16c.c, * elf32-cris.c, * elf32-crx.c, * elf32-csky.c,
* elf32-d10v.c, * elf32-epiphany.c, * elf32-fr30.c, * elf32-frv.c,
* elf32-ft32.c, * elf32-h8300.c, * elf32-hppa.c, * elf32-i386.c,
* elf32-ip2k.c, * elf32-iq2000.c, * elf32-lm32.c, * elf32-m32c.c,
* elf32-m32r.c, * elf32-m68hc1x.c, * elf32-m68k.c, * elf32-mcore.c,
* elf32-mep.c, * elf32-metag.c, * elf32-microblaze.c,
* elf32-moxie.c, * elf32-msp430.c, * elf32-mt.c, * elf32-nds32.c,
* elf32-nios2.c, * elf32-or1k.c, * elf32-ppc.c, * elf32-pru.c,
* elf32-rl78.c, * elf32-rx.c, * elf32-s390.c, * elf32-score.c,
* elf32-score7.c, * elf32-sh.c, * elf32-spu.c, * elf32-tic6x.c,
* elf32-tilepro.c, * elf32-v850.c, * elf32-vax.c, * elf32-visium.c,
* elf32-xstormy16.c, * elf32-xtensa.c, * elf64-alpha.c,
* elf64-bpf.c, * elf64-hppa.c, * elf64-ia64-vms.c, * elf64-mmix.c,
* elf64-ppc.c, * elf64-s390.c, * elf64-sparc.c, * elf64-x86-64.c,
* elflink.c, * elfnn-aarch64.c, * elfnn-ia64.c, * elfnn-riscv.c,
* elfxx-aarch64.c, * elfxx-mips.c, * elfxx-sparc.c,
* elfxx-tilegx.c, * elfxx-x86.c, * i386msdos.c, * linker.c,
* mach-o.c, * mmo.c, * opncls.c, * pdp11.c, * pei-x86_64.c,
* peicode.h, * reloc.c, * section.c, * syms.c, * vms-alpha.c,
* xcofflink.c: Update throughout for bfd section macro and function
changes.
binutils/
* addr2line.c, * bucomm.c, * coffgrok.c, * dlltool.c, * nm.c,
* objcopy.c, * objdump.c, * od-elf32_avr.c, * od-macho.c,
* od-xcoff.c, * prdbg.c, * rdcoff.c, * rddbg.c, * rescoff.c,
* resres.c, * size.c, * srconv.c, * strings.c, * windmc.c: Update
throughout for bfd section macro and function changes.
gas/
* as.c, * as.h, * dw2gencfi.c, * dwarf2dbg.c, * ecoff.c,
* read.c, * stabs.c, * subsegs.c, * subsegs.h, * write.c,
* config/obj-coff-seh.c, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-macho.c, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arm.c, * config/tc-avr.c, * config/tc-bfin.c,
* config/tc-bpf.c, * config/tc-d10v.c, * config/tc-d30v.c,
* config/tc-epiphany.c, * config/tc-fr30.c, * config/tc-frv.c,
* config/tc-h8300.c, * config/tc-hppa.c, * config/tc-i386.c,
* config/tc-ia64.c, * config/tc-ip2k.c, * config/tc-iq2000.c,
* config/tc-lm32.c, * config/tc-m32c.c, * config/tc-m32r.c,
* config/tc-m68hc11.c, * config/tc-mep.c, * config/tc-microblaze.c,
* config/tc-mips.c, * config/tc-mmix.c, * config/tc-mn10200.c,
* config/tc-mn10300.c, * config/tc-msp430.c, * config/tc-mt.c,
* config/tc-nds32.c, * config/tc-or1k.c, * config/tc-ppc.c,
* config/tc-pru.c, * config/tc-rl78.c, * config/tc-rx.c,
* config/tc-s12z.c, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score7.c, * config/tc-sh.c, * config/tc-sparc.c,
* config/tc-spu.c, * config/tc-tic4x.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tilegx.c, * config/tc-tilepro.c,
* config/tc-v850.c, * config/tc-visium.c, * config/tc-wasm32.c,
* config/tc-xc16x.c, * config/tc-xgate.c, * config/tc-xstormy16.c,
* config/tc-xtensa.c, * config/tc-z8k.c: Update throughout for
bfd section macro and function changes.
* write.c (compress_debug): Use bfd_rename_section.
gdb/
* aarch64-linux-tdep.c, * arm-tdep.c, * auto-load.c,
* coff-pe-read.c, * coffread.c, * corelow.c, * dbxread.c,
* dicos-tdep.c, * dwarf2-frame.c, * dwarf2read.c, * elfread.c,
* exec.c, * fbsd-tdep.c, * gcore.c, * gdb_bfd.c, * gdb_bfd.h,
* hppa-tdep.c, * i386-cygwin-tdep.c, * i386-fbsd-tdep.c,
* i386-linux-tdep.c, * jit.c, * linux-tdep.c, * machoread.c,
* maint.c, * mdebugread.c, * minidebug.c, * mips-linux-tdep.c,
* mips-sde-tdep.c, * mips-tdep.c, * mipsread.c, * nto-tdep.c,
* objfiles.c, * objfiles.h, * osabi.c, * ppc-linux-tdep.c,
* ppc64-tdep.c, * record-btrace.c, * record-full.c, * remote.c,
* rs6000-aix-tdep.c, * rs6000-tdep.c, * s390-linux-tdep.c,
* s390-tdep.c, * solib-aix.c, * solib-dsbt.c, * solib-frv.c,
* solib-spu.c, * solib-svr4.c, * solib-target.c,
* spu-linux-nat.c, * spu-tdep.c, * symfile-mem.c, * symfile.c,
* symmisc.c, * symtab.c, * target.c, * windows-nat.c,
* xcoffread.c, * cli/cli-dump.c, * compile/compile-object-load.c,
* mi/mi-interp.c: Update throughout for bfd section macro and
function changes.
* gcore (gcore_create_callback): Use bfd_set_section_lma.
* spu-tdep.c (spu_overlay_new_objfile): Likewise.
gprof/
* corefile.c, * symtab.c: Update throughout for bfd section
macro and function changes.
ld/
* ldcref.c, * ldctor.c, * ldelf.c, * ldlang.c, * pe-dll.c,
* emultempl/aarch64elf.em, * emultempl/aix.em,
* emultempl/armcoff.em, * emultempl/armelf.em,
* emultempl/cr16elf.em, * emultempl/cskyelf.em,
* emultempl/m68hc1xelf.em, * emultempl/m68kelf.em,
* emultempl/mipself.em, * emultempl/mmix-elfnmmo.em,
* emultempl/mmo.em, * emultempl/msp430.em,
* emultempl/nios2elf.em, * emultempl/pe.em, * emultempl/pep.em,
* emultempl/ppc64elf.em, * emultempl/xtensaelf.em: Update
throughout for bfd section macro and function changes.
libctf/
* ctf-open-bfd.c: Update throughout for bfd section macro changes.
opcodes/
* arc-ext.c: Update throughout for bfd section macro changes.
sim/
* common/sim-load.c, * common/sim-utils.c, * cris/sim-if.c,
* erc32/func.c, * lm32/sim-if.c, * m32c/load.c, * m32c/trace.c,
* m68hc11/interp.c, * ppc/hw_htab.c, * ppc/hw_init.c,
* rl78/load.c, * rl78/trace.c, * rx/gdb-if.c, * rx/load.c,
* rx/trace.c: Update throughout for bfd section macro changes.
2019-09-16 18:55:17 +08:00
|
|
|
int align = bfd_section_alignment (seg);
|
2015-11-10 00:12:57 +08:00
|
|
|
return ((addr + (1 << align) - 1) & -(1 << align));
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
md_assemble (char *input_line)
|
|
|
|
{
|
|
|
|
struct xgate_opcode *opcode = 0;
|
|
|
|
struct xgate_opcode *macro_opcode = 0;
|
|
|
|
struct xgate_opcode_handle *opcode_handle = 0;
|
|
|
|
/* Caller expects it to be returned as it was passed. */
|
|
|
|
char *saved_input_line = input_line;
|
|
|
|
char op_name[9] = { 0 };
|
2013-10-11 12:55:42 +08:00
|
|
|
unsigned int operandCount = 0;
|
2012-05-03 21:12:08 +08:00
|
|
|
char *p = 0;
|
|
|
|
|
2012-07-06 03:37:52 +08:00
|
|
|
s_operand new_operands[MAX_NUM_OPERANDS];
|
|
|
|
|
2012-05-03 21:12:08 +08:00
|
|
|
fixup_required = 0;
|
|
|
|
oper_check = 0; /* set error flags */
|
2016-08-05 18:26:13 +08:00
|
|
|
input_line = extract_word (input_line, op_name, sizeof (op_name));
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
/* Check to make sure we are not reading a bogus line. */
|
|
|
|
if (!op_name[0])
|
|
|
|
as_bad (_("opcode missing or not found on input line"));
|
|
|
|
|
PR26513, 629310abec breaks assembling PowerPC Linux kernels
Inserting with replacement is wrong for some gas hash table uses.
This patch implements an htab_insert that conditionally replaces, and
similarly for str_hash_insert. str_hash_insert with replace=0 is
roughly equivalent to the older hash_insert, and str_hash_insert with
replace=1 to the older hash_jam, but return values are different. I
found it useful to know whether the slot was occupied prior to
inserting/replacing. I've also reinstated the fatal errors on messing
up opcode tables with duplicates.
PR 26513
* hash.h (htab_insert): Update prototype and comment.
(struct string_tuple): Make "value" a const void*.
(string_tuple_alloc): Likewise.
(str_hash_find, str_hash_find_n): Cast returned value.
(str_hash_insert): Add "replace" parameter, and return slot pointer.
Free alloc'd element when not inserted.
* hash.c (htab_insert): Likewise. Return slot when element exists,
otherwise return NULL.
* read.c (pop_insert): Insert into hash table without first searching.
* config/tc-avr.c (md_begin): Likewise.
* config/tc-msp430.c (md_begin): Likewise.
* config/tc-nds32.c (nds32_init_nds32_pseudo_opcodes): Likewise.
* config/tc-v850.c (md_begin): Likewise.
* macro.c (do_formals, define_macro, macro_expand_body): Likewise.
(delete_macro): Delete from hash table.
* config/tc-tic54x.c (subsym_create_or_replace): Correct logic.
* symbols.c (local_symbol_make, symbol_table_insert): Allow
replacement of hash table entries.
* config/obj-coff-seh.c (seh_hash_insert): Likewise.
* config/obj-coff.c (tag_insert): Likewise.
* config/tc-iq2000.c (iq2000_add_macro): Likewise.
* config/tc-m68k.c (md_begin): Likewise for aliases.
* config/tc-tic4x.c (tic4x_asg): Likewise.
* config/tc-tic6x.c (md_begin): Likewise.
* dw2gencfi.c (dwcfi_hash_find_or_make): Disallow replacement of
hash table entries.
* ecoff.c (add_string, get_tag): Likewise.
* macro.c (expand_irp): Likewise.
* config/obj-elf.c (build_additional_section_info): Likewise.
* config/tc-aarch64.c (insert_reg_alias): Likewise.
(checked_hash_insert): Likewise.
* config/tc-alpha.c (get_alpha_reloc_tag, md_begin): Likewise.
* config/tc-arc.c (arc_insert_opcode, declare_register): Likewise.
(declare_addrtype, md_begin, arc_extcorereg): Likewise.
* config/tc-arm.c (insert_reg_alias): Likewise.
(arm_tc_equal_in_insn, md_begin): Likewise.
* config/tc-cr16.c (initialise_reg_hash_table, md_begin): Likewise.
* config/tc-cris.c (md_begin): Likewise.
* config/tc-crx.c (md_begin): Likewise.
* config/tc-csky.c (md_begin): Likewise.
* config/tc-d10v.c (md_begin): Likewise.
* config/tc-dlx.c (md_begin): Likewise.
* config/tc-ft32.c (md_begin): Likewise.
* config/tc-h8300.c (md_begin): Likewise.
* config/tc-hppa.c (md_begin): Likewise.
* config/tc-i386.c (md_begin): Likewise.
* config/tc-ia64.c (dot_rot, dot_entry, declare_register): Likewise.
(md_begin, dot_alias): Likewise.
* config/tc-m68hc11.c (md_begin): Likewise.
* config/tc-m68k.c (md_begin): Likewise.
* config/tc-mcore.c (md_begin): Likewise.
* config/tc-microblaze.c (md_begin): Likewise.
* config/tc-mips.c (md_begin): Likewise.
* config/tc-mmix.c (md_begin): Likewise.
* config/tc-mn10200.c (md_begin): Likewise.
* config/tc-mn10300.c (md_begin): Likewise.
* config/tc-moxie.c (md_begin): Likewise.
* config/tc-nds32.c (nds32_relax_hint, md_begin): Likewise.
* config/tc-nios2.c (md_begin): Likewise.
* config/tc-ns32k.c (md_begin): Likewise.
* config/tc-pdp11.c (md_begin): Likewise.
* config/tc-pj.c (fake_opcode, md_begin): Likewise.
* config/tc-ppc.c (ppc_setup_opcodes): Likewise.
* config/tc-pru.c (md_begin): Likewise.
* config/tc-riscv.c (init_ext_version_hash): Likewise.
(init_opcode_names_hash, hash_reg_name, init_opcode_hash): Likewise.
(riscv_init_csr_hash): Likewise.
* config/tc-s390.c (s390_setup_opcodes, md_begin): Likewise.
* config/tc-score.c (s3_insert_reg): Likewise.
(s3_build_score_ops_hsh, s3_build_dependency_insn_hsh): Likewise.
* config/tc-score7.c (s7_build_score_ops_hsh): Likewise.
(s7_build_dependency_insn_hsh, s7_insert_reg): Likewise.
* config/tc-sh.c (md_begin): Likewise.
* config/tc-sparc.c (md_begin): Likewise.
* config/tc-spu.c (md_begin): Likewise.
* config/tc-tic30.c (md_begin): Likewise.
* config/tc-tic4x.c (tic4x_inst_insert): Likewise.
* config/tc-tic54x.c (stag_add_field_symbols, md_begin): Likewise.
(tic54x_endstruct, tic54x_var, tic54x_macro_info): Likewise.
(subsym_substitute): Likewise.
* config/tc-tilegx.c (md_begin): Likewise.
* config/tc-tilepro.c (md_begin): Likewise.
* config/tc-vax.c (vip_begin): Likewise.
* config/tc-wasm32.c (md_begin): Likewise.
* config/tc-xgate.c (md_begin): Likewise.
* config/tc-z8k.c (md_begin): Likewise.
* testsuite/gas/ppc/dcbt.d,
* testsuite/gas/ppc/dcbt.s: New test.
* testsuite/gas/ppc/ppc.exp: Run it.
* ecoff.c (add_string): Report fatal error on duplicates.
* config/tc-alpha.c (md_begin): Likewise.
* config/tc-arc.c (arc_insert_opcode, declare_register): Likewise.
(declare_addrtype, md_begin, arc_extcorereg): Likewise.
* config/tc-cr16.c (initialise_reg_hash_table, md_begin): Likewise.
* config/tc-cris.c (md_begin): Likewise.
* config/tc-crx.c (md_begin): Likewise.
* config/tc-dlx.c (md_begin): Likewise.
* config/tc-hppa.c (md_begin): Likewise.
* config/tc-i386.c (md_begin): Likewise.
* config/tc-ia64.c (dot_rot, dot_entry, declare_register): Likewise.
(md_begin): Likewise.
* config/tc-m68k.c (md_begin): Likewise.
* config/tc-mips.c (md_begin): Likewise.
* config/tc-nios2.c (md_begin): Likewise.
* config/tc-ns32k.c (md_begin): Likewise.
* config/tc-ppc.c (ppc_setup_opcodes): Likewise.
* config/tc-pru.c (md_begin): Likewise.
* config/tc-riscv.c (init_ext_version_hash): Likewise.
(init_opcode_names_hash, hash_reg_name, init_opcode_hash): Likewise.
* config/tc-s390.c (s390_setup_opcodes, md_begin): Likewise.
* config/tc-sparc.c (md_begin): Likewise.
* config/tc-tic30.c (md_begin): Likewise.
* config/tc-tic4x.c (tic4x_inst_insert): Likewise.
* config/tc-tilegx.c (md_begin): Likewise.
* config/tc-tilepro.c (md_begin): Likewise.
* config/tc-vax.c (vip_begin): Likewise.
* config/tc-alpha.c,
* config/tc-arm.c,
* config/tc-avr.c,
* config/tc-cr16.c,
* config/tc-csky.c,
* config/tc-i386.c,
* config/tc-m68hc11.c,
* config/tc-m68k.c,
* config/tc-microblaze.c,
* config/tc-ns32k.c,
* config/tc-pj.c,
* config/tc-ppc.c,
* config/tc-score.c,
* config/tc-score7.c,
* config/tc-tic4x.c,
* config/tc-tic54x.c,
* config/tc-tilegx.c,
* config/tc-tilepro.c,
* config/tc-xgate.c: Formatting.
2020-08-22 16:29:57 +08:00
|
|
|
opcode_handle = (struct xgate_opcode_handle *) str_hash_find (xgate_hash,
|
|
|
|
op_name);
|
|
|
|
if (!opcode_handle)
|
|
|
|
as_bad (_("opcode %s not found in opcode hash table"), op_name);
|
2012-05-03 21:12:08 +08:00
|
|
|
else
|
|
|
|
{
|
2012-07-06 03:37:52 +08:00
|
|
|
/* Parse operands so we can find the proper opcode bin. */
|
|
|
|
|
2013-10-11 12:55:42 +08:00
|
|
|
operandCount = xgate_get_operands (input_line, new_operands);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
opcode = xgate_find_match (opcode_handle, opcode_handle->number_of_modes,
|
2013-10-11 12:55:42 +08:00
|
|
|
new_operands, operandCount);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
if (!opcode)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
2017-01-23 23:23:07 +08:00
|
|
|
as_bad (_("matching operands to opcode"));
|
2012-11-06 18:03:32 +08:00
|
|
|
xgate_print_syntax (opcode_handle->opc0[0]->name);
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
else if (opcode->size == 2)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
2012-05-03 21:12:08 +08:00
|
|
|
/* Size is one word - assemble that native insn. */
|
2012-11-06 18:03:32 +08:00
|
|
|
xgate_scan_operands (opcode, new_operands);
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
else
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
2012-05-03 21:12:08 +08:00
|
|
|
/* Insn is a simplified instruction - expand it out. */
|
2012-11-06 18:03:32 +08:00
|
|
|
autoHiLo = 1;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
/* skip past our ';' separator. */
|
|
|
|
for (i = strlen (opcode->constraints), p = opcode->constraints; i > 0;
|
|
|
|
i--, p++)
|
|
|
|
{
|
|
|
|
if (*p == ';')
|
|
|
|
{
|
|
|
|
p++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
input_line = skip_whitespace (input_line);
|
|
|
|
char *macro_inline = input_line;
|
|
|
|
|
|
|
|
/* Loop though the macro's opcode list and apply operands to
|
|
|
|
each real opcode. */
|
|
|
|
for (i = 0; *p && i < (opcode->size / 2); i++)
|
|
|
|
{
|
2012-05-03 21:12:08 +08:00
|
|
|
/* Loop though macro operand list. */
|
2012-11-06 18:03:32 +08:00
|
|
|
input_line = macro_inline; /* Rewind. */
|
|
|
|
p = extract_word (p, op_name, 10);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
PR26513, 629310abec breaks assembling PowerPC Linux kernels
Inserting with replacement is wrong for some gas hash table uses.
This patch implements an htab_insert that conditionally replaces, and
similarly for str_hash_insert. str_hash_insert with replace=0 is
roughly equivalent to the older hash_insert, and str_hash_insert with
replace=1 to the older hash_jam, but return values are different. I
found it useful to know whether the slot was occupied prior to
inserting/replacing. I've also reinstated the fatal errors on messing
up opcode tables with duplicates.
PR 26513
* hash.h (htab_insert): Update prototype and comment.
(struct string_tuple): Make "value" a const void*.
(string_tuple_alloc): Likewise.
(str_hash_find, str_hash_find_n): Cast returned value.
(str_hash_insert): Add "replace" parameter, and return slot pointer.
Free alloc'd element when not inserted.
* hash.c (htab_insert): Likewise. Return slot when element exists,
otherwise return NULL.
* read.c (pop_insert): Insert into hash table without first searching.
* config/tc-avr.c (md_begin): Likewise.
* config/tc-msp430.c (md_begin): Likewise.
* config/tc-nds32.c (nds32_init_nds32_pseudo_opcodes): Likewise.
* config/tc-v850.c (md_begin): Likewise.
* macro.c (do_formals, define_macro, macro_expand_body): Likewise.
(delete_macro): Delete from hash table.
* config/tc-tic54x.c (subsym_create_or_replace): Correct logic.
* symbols.c (local_symbol_make, symbol_table_insert): Allow
replacement of hash table entries.
* config/obj-coff-seh.c (seh_hash_insert): Likewise.
* config/obj-coff.c (tag_insert): Likewise.
* config/tc-iq2000.c (iq2000_add_macro): Likewise.
* config/tc-m68k.c (md_begin): Likewise for aliases.
* config/tc-tic4x.c (tic4x_asg): Likewise.
* config/tc-tic6x.c (md_begin): Likewise.
* dw2gencfi.c (dwcfi_hash_find_or_make): Disallow replacement of
hash table entries.
* ecoff.c (add_string, get_tag): Likewise.
* macro.c (expand_irp): Likewise.
* config/obj-elf.c (build_additional_section_info): Likewise.
* config/tc-aarch64.c (insert_reg_alias): Likewise.
(checked_hash_insert): Likewise.
* config/tc-alpha.c (get_alpha_reloc_tag, md_begin): Likewise.
* config/tc-arc.c (arc_insert_opcode, declare_register): Likewise.
(declare_addrtype, md_begin, arc_extcorereg): Likewise.
* config/tc-arm.c (insert_reg_alias): Likewise.
(arm_tc_equal_in_insn, md_begin): Likewise.
* config/tc-cr16.c (initialise_reg_hash_table, md_begin): Likewise.
* config/tc-cris.c (md_begin): Likewise.
* config/tc-crx.c (md_begin): Likewise.
* config/tc-csky.c (md_begin): Likewise.
* config/tc-d10v.c (md_begin): Likewise.
* config/tc-dlx.c (md_begin): Likewise.
* config/tc-ft32.c (md_begin): Likewise.
* config/tc-h8300.c (md_begin): Likewise.
* config/tc-hppa.c (md_begin): Likewise.
* config/tc-i386.c (md_begin): Likewise.
* config/tc-ia64.c (dot_rot, dot_entry, declare_register): Likewise.
(md_begin, dot_alias): Likewise.
* config/tc-m68hc11.c (md_begin): Likewise.
* config/tc-m68k.c (md_begin): Likewise.
* config/tc-mcore.c (md_begin): Likewise.
* config/tc-microblaze.c (md_begin): Likewise.
* config/tc-mips.c (md_begin): Likewise.
* config/tc-mmix.c (md_begin): Likewise.
* config/tc-mn10200.c (md_begin): Likewise.
* config/tc-mn10300.c (md_begin): Likewise.
* config/tc-moxie.c (md_begin): Likewise.
* config/tc-nds32.c (nds32_relax_hint, md_begin): Likewise.
* config/tc-nios2.c (md_begin): Likewise.
* config/tc-ns32k.c (md_begin): Likewise.
* config/tc-pdp11.c (md_begin): Likewise.
* config/tc-pj.c (fake_opcode, md_begin): Likewise.
* config/tc-ppc.c (ppc_setup_opcodes): Likewise.
* config/tc-pru.c (md_begin): Likewise.
* config/tc-riscv.c (init_ext_version_hash): Likewise.
(init_opcode_names_hash, hash_reg_name, init_opcode_hash): Likewise.
(riscv_init_csr_hash): Likewise.
* config/tc-s390.c (s390_setup_opcodes, md_begin): Likewise.
* config/tc-score.c (s3_insert_reg): Likewise.
(s3_build_score_ops_hsh, s3_build_dependency_insn_hsh): Likewise.
* config/tc-score7.c (s7_build_score_ops_hsh): Likewise.
(s7_build_dependency_insn_hsh, s7_insert_reg): Likewise.
* config/tc-sh.c (md_begin): Likewise.
* config/tc-sparc.c (md_begin): Likewise.
* config/tc-spu.c (md_begin): Likewise.
* config/tc-tic30.c (md_begin): Likewise.
* config/tc-tic4x.c (tic4x_inst_insert): Likewise.
* config/tc-tic54x.c (stag_add_field_symbols, md_begin): Likewise.
(tic54x_endstruct, tic54x_var, tic54x_macro_info): Likewise.
(subsym_substitute): Likewise.
* config/tc-tilegx.c (md_begin): Likewise.
* config/tc-tilepro.c (md_begin): Likewise.
* config/tc-vax.c (vip_begin): Likewise.
* config/tc-wasm32.c (md_begin): Likewise.
* config/tc-xgate.c (md_begin): Likewise.
* config/tc-z8k.c (md_begin): Likewise.
* testsuite/gas/ppc/dcbt.d,
* testsuite/gas/ppc/dcbt.s: New test.
* testsuite/gas/ppc/ppc.exp: Run it.
* ecoff.c (add_string): Report fatal error on duplicates.
* config/tc-alpha.c (md_begin): Likewise.
* config/tc-arc.c (arc_insert_opcode, declare_register): Likewise.
(declare_addrtype, md_begin, arc_extcorereg): Likewise.
* config/tc-cr16.c (initialise_reg_hash_table, md_begin): Likewise.
* config/tc-cris.c (md_begin): Likewise.
* config/tc-crx.c (md_begin): Likewise.
* config/tc-dlx.c (md_begin): Likewise.
* config/tc-hppa.c (md_begin): Likewise.
* config/tc-i386.c (md_begin): Likewise.
* config/tc-ia64.c (dot_rot, dot_entry, declare_register): Likewise.
(md_begin): Likewise.
* config/tc-m68k.c (md_begin): Likewise.
* config/tc-mips.c (md_begin): Likewise.
* config/tc-nios2.c (md_begin): Likewise.
* config/tc-ns32k.c (md_begin): Likewise.
* config/tc-ppc.c (ppc_setup_opcodes): Likewise.
* config/tc-pru.c (md_begin): Likewise.
* config/tc-riscv.c (init_ext_version_hash): Likewise.
(init_opcode_names_hash, hash_reg_name, init_opcode_hash): Likewise.
* config/tc-s390.c (s390_setup_opcodes, md_begin): Likewise.
* config/tc-sparc.c (md_begin): Likewise.
* config/tc-tic30.c (md_begin): Likewise.
* config/tc-tic4x.c (tic4x_inst_insert): Likewise.
* config/tc-tilegx.c (md_begin): Likewise.
* config/tc-tilepro.c (md_begin): Likewise.
* config/tc-vax.c (vip_begin): Likewise.
* config/tc-alpha.c,
* config/tc-arm.c,
* config/tc-avr.c,
* config/tc-cr16.c,
* config/tc-csky.c,
* config/tc-i386.c,
* config/tc-m68hc11.c,
* config/tc-m68k.c,
* config/tc-microblaze.c,
* config/tc-ns32k.c,
* config/tc-pj.c,
* config/tc-ppc.c,
* config/tc-score.c,
* config/tc-score7.c,
* config/tc-tic4x.c,
* config/tc-tic54x.c,
* config/tc-tilegx.c,
* config/tc-tilepro.c,
* config/tc-xgate.c: Formatting.
2020-08-22 16:29:57 +08:00
|
|
|
opcode_handle
|
|
|
|
= (struct xgate_opcode_handle *) str_hash_find (xgate_hash,
|
|
|
|
op_name);
|
|
|
|
if (!opcode_handle)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
as_bad (_(": processing macro, real opcode handle"
|
|
|
|
" not found in hash"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-05 18:26:13 +08:00
|
|
|
operandCount = xgate_get_operands (input_line, new_operands);
|
2013-10-11 12:55:42 +08:00
|
|
|
macro_opcode = xgate_find_match (opcode_handle,
|
|
|
|
opcode_handle->number_of_modes, new_operands,
|
2016-08-05 18:26:13 +08:00
|
|
|
operandCount);
|
2012-11-06 18:03:32 +08:00
|
|
|
xgate_scan_operands (macro_opcode, new_operands);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
2012-07-28 06:33:22 +08:00
|
|
|
autoHiLo = 0;
|
2012-05-03 21:12:08 +08:00
|
|
|
input_line = saved_input_line;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Force truly undefined symbols to their maximum size, and generally set up
|
|
|
|
the frag list to be relaxed. */
|
|
|
|
|
|
|
|
int
|
|
|
|
md_estimate_size_before_relax (fragS *fragp, asection *seg)
|
|
|
|
{
|
|
|
|
/* If symbol is undefined or located in a different section,
|
|
|
|
select the largest supported relocation. */
|
|
|
|
relax_substateT subtype;
|
2012-11-06 18:03:32 +08:00
|
|
|
relax_substateT rlx_state[] = { 0, 2 };
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
for (subtype = 0; subtype < ARRAY_SIZE (rlx_state); subtype += 2)
|
|
|
|
{
|
|
|
|
if (fragp->fr_subtype == rlx_state[subtype]
|
2012-11-06 18:03:32 +08:00
|
|
|
&& (!S_IS_DEFINED (fragp->fr_symbol)
|
|
|
|
|| seg != S_GET_SEGMENT (fragp->fr_symbol)))
|
|
|
|
{
|
|
|
|
fragp->fr_subtype = rlx_state[subtype + 1];
|
|
|
|
break;
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (fragp->fr_subtype >= ARRAY_SIZE (md_relax_table))
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
return md_relax_table[fragp->fr_subtype].rlx_length;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Relocation, relaxation and frag conversions. */
|
|
|
|
|
|
|
|
/* PC-relative offsets are relative to the start of the
|
|
|
|
next instruction. That is, the address of the offset, plus its
|
|
|
|
size, since the offset is always the last part of the insn. */
|
|
|
|
|
|
|
|
long
|
|
|
|
md_pcrel_from (fixS * fixP)
|
|
|
|
{
|
|
|
|
return fixP->fx_size + fixP->fx_where + fixP->fx_frag->fr_address;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If while processing a fixup, a reloc really needs to be created
|
|
|
|
then it is done here. */
|
|
|
|
|
|
|
|
arelent *
|
|
|
|
tc_gen_reloc (asection * section ATTRIBUTE_UNUSED, fixS * fixp)
|
|
|
|
{
|
|
|
|
arelent * reloc;
|
|
|
|
|
2016-04-07 04:26:46 +08:00
|
|
|
reloc = XNEW (arelent);
|
|
|
|
reloc->sym_ptr_ptr = XNEW (asymbol *);
|
2012-05-03 21:12:08 +08:00
|
|
|
*reloc->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
|
|
|
|
reloc->address = fixp->fx_frag->fr_address + fixp->fx_where;
|
|
|
|
|
|
|
|
if (fixp->fx_r_type == 0)
|
2012-11-06 18:03:32 +08:00
|
|
|
reloc->howto = bfd_reloc_type_lookup (stdoutput, BFD_RELOC_16);
|
2012-05-03 21:12:08 +08:00
|
|
|
else
|
2012-11-06 18:03:32 +08:00
|
|
|
reloc->howto = bfd_reloc_type_lookup (stdoutput, fixp->fx_r_type);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
if (reloc->howto == (reloc_howto_type *) NULL)
|
|
|
|
{
|
|
|
|
as_bad_where (fixp->fx_file, fixp->fx_line, _
|
2012-11-06 18:03:32 +08:00
|
|
|
("Relocation %d is not supported by object file format."),
|
|
|
|
(int) fixp->fx_r_type);
|
2012-05-03 21:12:08 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Since we use Rel instead of Rela, encode the vtable entry to be
|
|
|
|
used in the relocation's section offset. */
|
|
|
|
if (fixp->fx_r_type == BFD_RELOC_VTABLE_ENTRY)
|
|
|
|
reloc->address = fixp->fx_offset;
|
|
|
|
reloc->addend = 0;
|
|
|
|
return reloc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Patch the instruction with the resolved operand. Elf relocation
|
|
|
|
info will also be generated to take care of linker/loader fixups.
|
|
|
|
The XGATE addresses only 16-bit addresses.The BFD_RELOC_32 is necessary
|
|
|
|
for the support of --gstabs. */
|
|
|
|
|
|
|
|
void
|
|
|
|
md_apply_fix (fixS * fixP, valueT * valP, segT seg ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
char *where;
|
|
|
|
long value = *valP;
|
|
|
|
int opcode = 0;
|
|
|
|
ldiv_t result;
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
/* If the fixup is done mark it done so no further symbol resolution
|
|
|
|
will take place. */
|
2012-05-03 21:12:08 +08:00
|
|
|
if (fixP->fx_addsy == (symbolS *) NULL)
|
2012-11-06 18:03:32 +08:00
|
|
|
fixP->fx_done = 1;
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
/* We don't actually support subtracting a symbol. */
|
|
|
|
if (fixP->fx_subsy != (symbolS *) NULL)
|
2021-07-21 13:09:29 +08:00
|
|
|
as_bad_subtract (fixP);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
where = fixP->fx_frag->fr_literal + fixP->fx_where;
|
|
|
|
opcode = bfd_getl16 (where);
|
|
|
|
int mask = 0;
|
|
|
|
|
|
|
|
switch (fixP->fx_r_type)
|
|
|
|
{
|
2020-05-27 00:33:48 +08:00
|
|
|
case BFD_RELOC_XGATE_PCREL_9:
|
2012-05-03 21:12:08 +08:00
|
|
|
if (value < -512 || value > 511)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
2012-11-06 18:03:32 +08:00
|
|
|
_("Value %ld too large for 9-bit PC-relative branch."),
|
|
|
|
value);
|
2012-05-03 21:12:08 +08:00
|
|
|
result = ldiv (value, 2); /* from bytes to words */
|
|
|
|
value = result.quot;
|
|
|
|
if (result.rem)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line, _
|
2012-11-06 18:03:32 +08:00
|
|
|
("Value %ld not aligned by 2 for 9-bit"
|
|
|
|
" PC-relative branch."), value);
|
|
|
|
/* Clip into 8-bit field.
|
|
|
|
FIXME I'm sure there is a more proper place for this. */
|
|
|
|
mask = 0x1FF;
|
2012-05-03 21:12:08 +08:00
|
|
|
value &= mask;
|
|
|
|
number_to_chars_bigendian (where, (opcode | value), 2);
|
|
|
|
break;
|
2020-05-27 00:33:48 +08:00
|
|
|
case BFD_RELOC_XGATE_PCREL_10:
|
2012-05-03 21:12:08 +08:00
|
|
|
if (value < -1024 || value > 1023)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
2012-11-06 18:03:32 +08:00
|
|
|
_("Value %ld too large for 10-bit PC-relative branch."),
|
|
|
|
value);
|
2012-05-03 21:12:08 +08:00
|
|
|
result = ldiv (value, 2); /* from bytes to words */
|
|
|
|
value = result.quot;
|
|
|
|
if (result.rem)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line, _
|
2012-11-06 18:03:32 +08:00
|
|
|
("Value %ld not aligned by 2 for 10-bit"
|
|
|
|
" PC-relative branch."), value);
|
|
|
|
/* Clip into 9-bit field.
|
|
|
|
FIXME I'm sure there is a more proper place for this. */
|
|
|
|
mask = 0x3FF;
|
2012-05-03 21:12:08 +08:00
|
|
|
value &= mask;
|
|
|
|
number_to_chars_bigendian (where, (opcode | value), 2);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_XGATE_IMM8_HI:
|
|
|
|
if (value < -65537 || value > 65535)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
|
|
|
_("Value out of 16-bit range."));
|
|
|
|
value >>= 8;
|
|
|
|
value &= 0x00ff;
|
|
|
|
bfd_putb16 ((bfd_vma) value | opcode, (void *) where);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_XGATE_24:
|
|
|
|
case BFD_RELOC_XGATE_IMM8_LO:
|
|
|
|
if (value < -65537 || value > 65535)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
|
|
|
_("Value out of 16-bit range."));
|
|
|
|
value &= 0x00ff;
|
|
|
|
bfd_putb16 ((bfd_vma) value | opcode, (void *) where);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_XGATE_IMM3:
|
|
|
|
if (value < 0 || value > 7)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
|
|
|
_("Value out of 3-bit range."));
|
|
|
|
value <<= 8; /* make big endian */
|
|
|
|
number_to_chars_bigendian (where, (opcode | value), 2);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_XGATE_IMM4:
|
|
|
|
if (value < 0 || value > 15)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
|
|
|
_("Value out of 4-bit range."));
|
|
|
|
value <<= 4; /* align the operand bits */
|
|
|
|
number_to_chars_bigendian (where, (opcode | value), 2);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_XGATE_IMM5:
|
|
|
|
if (value < 0 || value > 31)
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
|
|
|
_("Value out of 5-bit range."));
|
|
|
|
value <<= 5; /* align the operand bits */
|
|
|
|
number_to_chars_bigendian (where, (opcode | value), 2);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_8:
|
|
|
|
((bfd_byte *) where)[0] = (bfd_byte) value;
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_32:
|
|
|
|
bfd_putb32 ((bfd_vma) value, (unsigned char *) where);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_16:
|
|
|
|
bfd_putb16 ((bfd_vma) value, (unsigned char *) where);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
as_fatal (_("Line %d: unknown relocation type: 0x%x."), fixP->fx_line,
|
|
|
|
fixP->fx_r_type);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* See whether we need to force a relocation into the output file. */
|
|
|
|
|
|
|
|
int
|
|
|
|
tc_xgate_force_relocation (fixS * fixP)
|
|
|
|
{
|
|
|
|
if (fixP->fx_r_type == BFD_RELOC_XGATE_RL_GROUP)
|
|
|
|
return 1;
|
|
|
|
return generic_force_reloc (fixP);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Here we decide which fixups can be adjusted to make them relative
|
|
|
|
to the beginning of the section instead of the symbol. Basically
|
|
|
|
we need to make sure that the linker relaxation is done
|
|
|
|
correctly, so in some cases we force the original symbol to be
|
|
|
|
used. */
|
|
|
|
|
|
|
|
int
|
|
|
|
tc_xgate_fix_adjustable (fixS * fixP)
|
|
|
|
{
|
|
|
|
switch (fixP->fx_r_type)
|
|
|
|
{
|
|
|
|
/* For the linker relaxation to work correctly, these relocs
|
|
|
|
need to be on the symbol itself. */
|
|
|
|
case BFD_RELOC_16:
|
|
|
|
case BFD_RELOC_XGATE_RL_JUMP:
|
|
|
|
case BFD_RELOC_XGATE_RL_GROUP:
|
|
|
|
case BFD_RELOC_VTABLE_INHERIT:
|
|
|
|
case BFD_RELOC_VTABLE_ENTRY:
|
|
|
|
case BFD_RELOC_32:
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
md_convert_frag (bfd * abfd ATTRIBUTE_UNUSED,
|
|
|
|
asection * sec ATTRIBUTE_UNUSED,
|
|
|
|
fragS * fragP ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
as_bad (("md_convert_frag not implemented yet"));
|
|
|
|
abort ();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the ELF specific flags. */
|
|
|
|
|
|
|
|
void
|
|
|
|
xgate_elf_final_processing (void)
|
|
|
|
{
|
|
|
|
elf_flags |= EF_XGATE_MACH;
|
|
|
|
elf_elfheader (stdoutput)->e_flags &= ~EF_XGATE_ABI;
|
|
|
|
elf_elfheader (stdoutput)->e_flags |= elf_flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline char *
|
|
|
|
skip_whitespace (char *s)
|
|
|
|
{
|
|
|
|
while (*s == ' ' || *s == '\t' || *s == '(' || *s == ')')
|
|
|
|
s++;
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Extract a word (continuous alpha-numeric chars) from the input line. */
|
|
|
|
|
|
|
|
static char *
|
|
|
|
extract_word (char *from, char *to, int limit)
|
|
|
|
{
|
|
|
|
char *op_end;
|
|
|
|
int size = 0;
|
|
|
|
|
|
|
|
/* Drop leading whitespace. */
|
|
|
|
from = skip_whitespace (from);
|
|
|
|
*to = 0;
|
|
|
|
/* Find the op code end. */
|
|
|
|
for (op_end = from; *op_end != 0 && is_part_of_name (*op_end);)
|
|
|
|
{
|
|
|
|
to[size++] = *op_end++;
|
|
|
|
if (size + 1 >= limit)
|
2012-11-06 18:03:32 +08:00
|
|
|
break;
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
to[size] = 0;
|
|
|
|
return op_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *
|
|
|
|
xgate_new_instruction (int size)
|
|
|
|
{
|
|
|
|
char *f = frag_more (size);
|
|
|
|
dwarf2_emit_insn (size);
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
static unsigned short
|
2012-05-03 21:12:08 +08:00
|
|
|
xgate_apply_operand (unsigned short new_mask,
|
|
|
|
unsigned short *availiable_mask_bits,
|
|
|
|
unsigned short mask,
|
|
|
|
unsigned char n_bits)
|
|
|
|
{
|
|
|
|
unsigned short n_shifts;
|
|
|
|
unsigned int n_drop_bits;
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
/* Shift until you find an available operand bit "1" and record
|
|
|
|
the number of shifts. */
|
2012-05-03 21:12:08 +08:00
|
|
|
for (n_shifts = 0;
|
|
|
|
!(*availiable_mask_bits & SIXTEENTH_BIT) && n_shifts < 16;
|
|
|
|
n_shifts++)
|
|
|
|
*availiable_mask_bits <<= 1;
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
/* Shift for the number of bits your operand requires while bits
|
|
|
|
are available. */
|
2012-05-03 21:12:08 +08:00
|
|
|
for (n_drop_bits = n_bits;
|
|
|
|
n_drop_bits && (*availiable_mask_bits & SIXTEENTH_BIT);
|
|
|
|
--n_drop_bits)
|
|
|
|
*availiable_mask_bits <<= 1;
|
|
|
|
|
|
|
|
if (n_drop_bits)
|
|
|
|
as_bad (_(":operand has too many bits"));
|
|
|
|
*availiable_mask_bits >>= n_shifts + n_bits;
|
|
|
|
if ((n_drop_bits == 0) && (*availiable_mask_bits == 0))
|
|
|
|
{
|
|
|
|
oper_check = 1; /* flag operand check as good */
|
|
|
|
}
|
|
|
|
new_mask <<= N_BITS_IN_WORD - (n_shifts + n_bits);
|
|
|
|
mask |= new_mask;
|
|
|
|
return mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Parse ordinary expression. */
|
|
|
|
|
|
|
|
static char *
|
|
|
|
xgate_parse_exp (char *s, expressionS * op)
|
|
|
|
{
|
|
|
|
input_line_pointer = s;
|
2016-08-05 18:26:13 +08:00
|
|
|
|
|
|
|
expression (op);
|
2012-05-03 21:12:08 +08:00
|
|
|
if (op->X_op == O_absent)
|
|
|
|
as_bad (_("missing operand"));
|
gas: equates of registers
There are two problems: symbol_equated_p() doesn't recognize equates of
registers, and S_CAN_BE_REDEFINED() goes by section rather than by
expression type. Both together undermine .eqv and .equiv clearly meaning
to guard the involved symbols against re-definition (both ways).
To compensate pseudo_set() now using O_symbol and S_CAN_BE_REDEFINED()
now checking for O_register,
- for targets creating register symbols through symbol_{new,create}() ->
symbol_init() -> S_SET_VALUE() (alpha, arc, dlx, ia64, m68k, mips,
mmix, tic4x, tic54x, plus anything using cgen or itbl-ops), have
symbol_init() set their expressions to O_register,
- x86'es parse_register() also can't go by section anymore when
trying to "look through" equates; probably symbol_equated_p() should
have been used there from the beginning, if only that had worked for
equates of registers,
- various targets need to "look through" equates when parsing insn
operands (which also helps transitive forward equates); perhaps even
more ought to, but many don't look to consider the possibility of
register equates in the first place.
This was uncovered by code reported in PR gas/30274 (duplicating
PR gas/30272), except that there .eqv was used when really .equ was
meant. Therefore that bug report is addressed here only in so far as
gas wouldn't crash anymore; the code there still won't assemble
successfully, just that now the issues there are properly diagnosed.
2023-05-12 14:55:48 +08:00
|
|
|
else
|
|
|
|
resolve_register (op);
|
2012-05-03 21:12:08 +08:00
|
|
|
return input_line_pointer;
|
|
|
|
}
|
|
|
|
|
2012-07-06 03:37:52 +08:00
|
|
|
static int
|
|
|
|
cmp_opcode (struct xgate_opcode *op1, struct xgate_opcode *op2)
|
|
|
|
{
|
|
|
|
return strcmp (op1->name, op2->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct xgate_opcode *
|
|
|
|
xgate_find_match (struct xgate_opcode_handle *opcode_handle,
|
2013-10-11 12:55:42 +08:00
|
|
|
int numberOfModes, s_operand oprs[], unsigned int operandCount)
|
2012-07-06 03:37:52 +08:00
|
|
|
{
|
|
|
|
int i;
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2012-07-06 03:37:52 +08:00
|
|
|
if (numberOfModes == 0)
|
|
|
|
return opcode_handle->opc0[0];
|
|
|
|
|
|
|
|
for (i = 0; i <= numberOfModes; i++)
|
2013-10-11 12:55:42 +08:00
|
|
|
{
|
|
|
|
switch (operandCount)
|
|
|
|
{
|
|
|
|
case 0:
|
2016-08-05 18:26:13 +08:00
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_INH))
|
2013-10-11 12:55:42 +08:00
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
if (oprs[0].reg >= REG_R0 && oprs[0].reg <= REG_R7)
|
2016-08-05 18:26:13 +08:00
|
|
|
{
|
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_MON))
|
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_DYA_MON))
|
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
}
|
2013-10-11 12:55:42 +08:00
|
|
|
if (oprs[0].reg == REG_NONE)
|
2016-08-05 18:26:13 +08:00
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_IMM3))
|
2013-10-11 12:55:42 +08:00
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
if (oprs[0].reg >= REG_R0 && oprs[0].reg <= REG_R7)
|
|
|
|
{
|
|
|
|
if (oprs[1].reg >= REG_R0 && oprs[1].reg <= REG_R7)
|
2016-08-05 18:26:13 +08:00
|
|
|
{
|
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_DYA))
|
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
}
|
2013-10-11 12:55:42 +08:00
|
|
|
if (oprs[1].reg == REG_CCR)
|
2016-08-05 18:26:13 +08:00
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_MON_R_C))
|
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
if (oprs[1].reg == REG_PC)
|
2016-08-05 18:26:13 +08:00
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_MON_R_P))
|
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
if (oprs[1].reg == REG_NONE)
|
2016-08-05 18:26:13 +08:00
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_IMM16)
|
|
|
|
|| !strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_IMM8)
|
|
|
|
|| !strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_IMM4)
|
|
|
|
|| !strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_IMM16mADD)
|
2016-08-05 18:26:13 +08:00
|
|
|
|| !strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_IMM16mAND)
|
2016-08-05 18:26:13 +08:00
|
|
|
|| !strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_IMM16mCPC)
|
2016-08-05 18:26:13 +08:00
|
|
|
|| !strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_IMM16mSUB)
|
2016-08-05 18:26:13 +08:00
|
|
|
|| !strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_IMM16mLDW))
|
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
}
|
|
|
|
if (oprs[0].reg == REG_CCR)
|
2016-08-05 18:26:13 +08:00
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints, XGATE_OP_MON_C_R))
|
2013-10-11 12:55:42 +08:00
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
if (oprs[0].reg >= REG_R0 && oprs[0].reg <= REG_R7)
|
|
|
|
{
|
|
|
|
if (oprs[1].reg >= REG_R0 && oprs[1].reg <= REG_R7)
|
|
|
|
{
|
|
|
|
if (oprs[2].reg >= REG_R0 && oprs[2].reg <= REG_R7)
|
|
|
|
{
|
2016-08-05 18:26:13 +08:00
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_IDR)
|
2016-08-05 18:26:13 +08:00
|
|
|
|| !strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_TRI))
|
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oprs[2].reg == REG_NONE)
|
2016-08-05 18:26:13 +08:00
|
|
|
if (!strcmp (opcode_handle->opc0[i]->constraints,
|
2013-10-11 12:55:42 +08:00
|
|
|
XGATE_OP_IDO5))
|
|
|
|
return opcode_handle->opc0[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2016-08-05 18:26:13 +08:00
|
|
|
as_bad (_("unknown operand count"));
|
2013-10-11 12:55:42 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL ;
|
2012-07-06 03:37:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Because we are dealing with two different core that view the system
|
|
|
|
memory with different offsets, we must differentiate what core a
|
|
|
|
symbol belongs to, in order for the linker to cross-link. */
|
|
|
|
|
|
|
|
int
|
|
|
|
xgate_frob_symbol (symbolS *sym)
|
2012-05-03 21:12:08 +08:00
|
|
|
{
|
2012-07-06 03:37:52 +08:00
|
|
|
asymbol *bfdsym;
|
|
|
|
elf_symbol_type *elfsym;
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2012-07-06 03:37:52 +08:00
|
|
|
bfdsym = symbol_get_bfdsym (sym);
|
2020-09-16 07:55:56 +08:00
|
|
|
elfsym = elf_symbol_from (bfdsym);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2016-08-05 18:26:13 +08:00
|
|
|
gas_assert (elfsym);
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2012-07-06 03:37:52 +08:00
|
|
|
/* Mark the symbol as being *from XGATE */
|
|
|
|
elfsym->internal_elf_sym.st_target_internal = 1;
|
2012-05-03 21:12:08 +08:00
|
|
|
|
2012-07-06 03:37:52 +08:00
|
|
|
return 0;
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
static unsigned int
|
2012-07-06 03:37:52 +08:00
|
|
|
xgate_get_operands (char *line, s_operand oprs[])
|
2012-05-03 21:12:08 +08:00
|
|
|
{
|
2012-07-06 03:37:52 +08:00
|
|
|
int num_operands;
|
|
|
|
|
|
|
|
/* If there are no operands, then it must be inherent. */
|
|
|
|
if (*line == 0 || *line == '\n' || *line == '\r')
|
2013-10-11 12:55:42 +08:00
|
|
|
return 0;
|
2012-07-06 03:37:52 +08:00
|
|
|
|
|
|
|
for (num_operands = 0; strlen (line) && (num_operands < MAX_NUM_OPERANDS);
|
2012-11-06 18:03:32 +08:00
|
|
|
num_operands++)
|
2012-07-06 03:37:52 +08:00
|
|
|
{
|
|
|
|
line = skip_whitespace (line);
|
|
|
|
if (*line == '#')
|
2012-11-06 18:03:32 +08:00
|
|
|
line++;
|
2012-07-06 03:37:52 +08:00
|
|
|
|
2012-07-28 06:33:22 +08:00
|
|
|
oprs[num_operands].mod = xgate_determine_modifiers (&line);
|
2012-07-06 03:37:52 +08:00
|
|
|
|
|
|
|
if ((oprs[num_operands].reg = reg_name_search (line)) == REG_NONE)
|
2012-11-06 18:03:32 +08:00
|
|
|
line = xgate_parse_exp (line, &oprs[num_operands].exp);
|
2012-07-06 03:37:52 +08:00
|
|
|
|
|
|
|
/* skip to next operand */
|
|
|
|
while (*line != 0)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
if (*line == ',')
|
|
|
|
{
|
|
|
|
line++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
line++;
|
|
|
|
}
|
2012-07-06 03:37:52 +08:00
|
|
|
}
|
|
|
|
if (num_operands > MAX_NUM_OPERANDS)
|
|
|
|
return 0;
|
2013-10-11 12:55:42 +08:00
|
|
|
return num_operands;
|
2012-07-06 03:37:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* reg_name_search() finds the register number given its name.
|
|
|
|
Returns the register number or REG_NONE on failure. */
|
2016-08-05 18:26:13 +08:00
|
|
|
|
2012-07-06 03:37:52 +08:00
|
|
|
static register_id
|
|
|
|
reg_name_search (char *name)
|
|
|
|
{
|
|
|
|
if (strncasecmp (name, "r0", 2) == 0)
|
|
|
|
return REG_R0;
|
|
|
|
if (strncasecmp (name, "r1", 2) == 0)
|
|
|
|
return REG_R1;
|
|
|
|
if (strncasecmp (name, "r2", 2) == 0)
|
|
|
|
return REG_R2;
|
|
|
|
if (strncasecmp (name, "r3", 2) == 0)
|
|
|
|
return REG_R3;
|
|
|
|
if (strncasecmp (name, "r4", 2) == 0)
|
|
|
|
return REG_R4;
|
|
|
|
if (strncasecmp (name, "r5", 2) == 0)
|
|
|
|
return REG_R5;
|
|
|
|
if (strncasecmp (name, "r6", 2) == 0)
|
|
|
|
return REG_R6;
|
|
|
|
if (strncasecmp (name, "r7", 2) == 0)
|
|
|
|
return REG_R7;
|
|
|
|
if (strncasecmp (name, "pc", 2) == 0)
|
|
|
|
return REG_PC;
|
|
|
|
if (strncasecmp (name, "ccr", 3) == 0)
|
|
|
|
return REG_CCR;
|
|
|
|
return REG_NONE;
|
|
|
|
}
|
|
|
|
|
2012-07-28 06:33:22 +08:00
|
|
|
/* Parse operand modifiers such as inc/dec/hi/low. */
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
static op_modifiers
|
2016-08-05 18:26:13 +08:00
|
|
|
xgate_determine_modifiers (char **line)
|
2012-07-06 03:37:52 +08:00
|
|
|
{
|
|
|
|
char *local_line = line[0];
|
|
|
|
|
|
|
|
if (strncasecmp (local_line, "%hi", 3) == 0)
|
|
|
|
{
|
|
|
|
*line += 3;
|
|
|
|
return MOD_LOAD_HIGH;
|
|
|
|
}
|
|
|
|
if (strncasecmp (local_line, "%lo", 3) == 0)
|
|
|
|
{
|
|
|
|
*line += 3;
|
|
|
|
return MOD_LOAD_LOW;
|
|
|
|
}
|
|
|
|
if (*(local_line + 2) == '+')
|
2012-11-06 18:03:32 +08:00
|
|
|
return MOD_POSTINC;
|
2012-07-06 03:37:52 +08:00
|
|
|
if (strncasecmp (local_line, "-r", 2) == 0)
|
|
|
|
{
|
|
|
|
*line += 1;
|
|
|
|
return MOD_PREDEC;
|
2012-11-06 18:03:32 +08:00
|
|
|
}
|
2012-07-06 03:37:52 +08:00
|
|
|
return MOD_NONE;
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Parse instruction operands. */
|
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
static void
|
2012-07-06 03:37:52 +08:00
|
|
|
xgate_scan_operands (struct xgate_opcode *opcode, s_operand oprs[])
|
2012-05-03 21:12:08 +08:00
|
|
|
{
|
|
|
|
char *frag = xgate_new_instruction (opcode->size);
|
|
|
|
int where = frag - frag_now->fr_literal;
|
|
|
|
char *op = opcode->constraints;
|
|
|
|
unsigned int bin = (int) opcode->bin_opcode;
|
|
|
|
unsigned short oper_mask = 0;
|
|
|
|
int operand_bit_length = 0;
|
|
|
|
unsigned int operand = 0;
|
|
|
|
char n_operand_bits = 0;
|
|
|
|
char first_operand_equals_second = 0;
|
|
|
|
int i = 0;
|
|
|
|
char c = 0;
|
|
|
|
|
|
|
|
/* Generate available operand bits mask. */
|
|
|
|
for (i = 0; (c = opcode->format[i]); i++)
|
|
|
|
{
|
|
|
|
if (ISDIGIT (c) || (c == 's'))
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
oper_mask <<= 1;
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
else
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
oper_mask <<= 1;
|
|
|
|
oper_mask += 1;
|
|
|
|
n_operand_bits++;
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Parse first operand. */
|
|
|
|
if (*op)
|
|
|
|
{
|
|
|
|
if (*op == '=')
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
first_operand_equals_second = 1;
|
|
|
|
++op;
|
|
|
|
}
|
|
|
|
operand = xgate_parse_operand (opcode, &operand_bit_length, where,
|
|
|
|
&op, oprs[0]);
|
2012-05-03 21:12:08 +08:00
|
|
|
++op;
|
|
|
|
bin = xgate_apply_operand (operand, &oper_mask, bin, operand_bit_length);
|
2012-07-06 03:37:52 +08:00
|
|
|
|
2016-08-05 18:26:13 +08:00
|
|
|
if (first_operand_equals_second)
|
2012-11-06 18:03:32 +08:00
|
|
|
bin = xgate_apply_operand (operand, &oper_mask, bin,
|
|
|
|
operand_bit_length);
|
2012-05-03 21:12:08 +08:00
|
|
|
/* Parse second operand. */
|
|
|
|
if (*op)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
if (*op == ',')
|
|
|
|
++op;
|
|
|
|
if (first_operand_equals_second)
|
|
|
|
{
|
|
|
|
bin = xgate_apply_operand (operand, &oper_mask, bin,
|
|
|
|
operand_bit_length);
|
|
|
|
++op;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
operand = xgate_parse_operand (opcode, &operand_bit_length, where,
|
|
|
|
&op, oprs[1]);
|
|
|
|
bin = xgate_apply_operand (operand, &oper_mask, bin,
|
|
|
|
operand_bit_length);
|
|
|
|
++op;
|
|
|
|
}
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
/* Parse the third register. */
|
|
|
|
if (*op)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
if (*op == ',')
|
|
|
|
++op;
|
|
|
|
operand = xgate_parse_operand (opcode, &operand_bit_length, where,
|
|
|
|
&op, oprs[2]);
|
|
|
|
bin = xgate_apply_operand (operand, &oper_mask, bin,
|
|
|
|
operand_bit_length);
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
if (opcode->size == 2 && fixup_required)
|
|
|
|
{
|
|
|
|
bfd_putl16 (bin, frag);
|
|
|
|
}
|
2013-10-11 12:55:42 +08:00
|
|
|
else if ( !strcmp (opcode->constraints, XGATE_OP_REL9)
|
|
|
|
|| !strcmp (opcode->constraints, XGATE_OP_REL10))
|
2012-05-03 21:12:08 +08:00
|
|
|
{
|
|
|
|
/* Write our data to a frag for further processing. */
|
2012-07-06 03:37:52 +08:00
|
|
|
bfd_putl16 (opcode->bin_opcode, frag);
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Apply operand mask(s)to bin opcode and write the output. */
|
|
|
|
/* Since we are done write this frag in xgate BE format. */
|
2012-07-06 03:37:52 +08:00
|
|
|
number_to_chars_bigendian (frag, bin, opcode->size);
|
2012-05-03 21:12:08 +08:00
|
|
|
}
|
|
|
|
prev = bin;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int
|
2012-07-06 03:37:52 +08:00
|
|
|
xgate_parse_operand (struct xgate_opcode *opcode,
|
2012-11-06 18:03:32 +08:00
|
|
|
int *bit_width,
|
|
|
|
int where,
|
|
|
|
char **op_con,
|
|
|
|
s_operand operand)
|
2012-05-03 21:12:08 +08:00
|
|
|
{
|
|
|
|
char *op_constraint = *op_con;
|
|
|
|
unsigned int op_mask = 0;
|
|
|
|
unsigned int pp_fix = 0;
|
|
|
|
unsigned short max_size = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
*bit_width = 0;
|
|
|
|
/* Reset. */
|
|
|
|
|
|
|
|
switch (*op_constraint)
|
|
|
|
{
|
|
|
|
case '+': /* Indexed register operand +/- or plain r. */
|
|
|
|
/* Default to neither inc or dec. */
|
|
|
|
pp_fix = 0;
|
|
|
|
*bit_width = 5;
|
2012-07-06 03:37:52 +08:00
|
|
|
|
2012-07-28 06:33:22 +08:00
|
|
|
if (operand.reg == REG_NONE)
|
2012-11-06 18:03:32 +08:00
|
|
|
as_bad (_(": expected register name r0-r7 ") );
|
2012-07-28 06:33:22 +08:00
|
|
|
op_mask = operand.reg;
|
2016-08-05 18:26:13 +08:00
|
|
|
if (operand.mod == MOD_POSTINC)
|
2012-11-06 18:03:32 +08:00
|
|
|
pp_fix = INCREMENT;
|
2016-08-05 18:26:13 +08:00
|
|
|
if (operand.mod == MOD_PREDEC)
|
2012-11-06 18:03:32 +08:00
|
|
|
pp_fix = DECREMENT;
|
2012-05-03 21:12:08 +08:00
|
|
|
op_mask <<= 2;
|
|
|
|
op_mask |= pp_fix;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'r': /* Register operand. */
|
2012-11-06 18:03:32 +08:00
|
|
|
if (operand.reg == REG_NONE)
|
|
|
|
as_bad (_(": expected register name r0-r7 "));
|
2012-07-06 03:37:52 +08:00
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
*bit_width = 3;
|
2012-07-06 03:37:52 +08:00
|
|
|
|
2012-11-06 18:03:32 +08:00
|
|
|
op_mask = operand.reg;
|
2012-05-03 21:12:08 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'i': /* Immediate value or expression expected. */
|
|
|
|
/* Advance the original format pointer. */
|
|
|
|
(*op_con)++;
|
|
|
|
op_constraint++;
|
|
|
|
if (ISDIGIT (*op_constraint))
|
2012-11-06 18:03:32 +08:00
|
|
|
*bit_width = (int) *op_constraint - '0';
|
2012-05-03 21:12:08 +08:00
|
|
|
else if (*op_constraint == 'a')
|
2012-11-06 18:03:32 +08:00
|
|
|
*bit_width = 0x0A;
|
2012-05-03 21:12:08 +08:00
|
|
|
else if (*op_constraint == 'f')
|
2012-11-06 18:03:32 +08:00
|
|
|
*bit_width = 0x0F;
|
|
|
|
|
2012-05-03 21:12:08 +08:00
|
|
|
/* http://tigcc.ticalc.org/doc/gnuasm.html#SEC31 */
|
2012-07-28 06:33:22 +08:00
|
|
|
if (operand.exp.X_op == O_constant)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
op_mask = operand.exp.X_add_number;
|
|
|
|
if (((opcode->name[strlen (opcode->name) - 1] == 'l') && autoHiLo)
|
|
|
|
|| operand.mod == MOD_LOAD_LOW)
|
|
|
|
op_mask &= 0x00FF;
|
|
|
|
else if (((opcode->name[strlen (opcode->name) - 1]) == 'h'
|
|
|
|
&& autoHiLo) || operand.mod == MOD_LOAD_HIGH)
|
|
|
|
op_mask >>= 8;
|
|
|
|
|
|
|
|
/* Make sure it fits. */
|
|
|
|
for (i = *bit_width; i; i--)
|
|
|
|
{
|
|
|
|
max_size <<= 1;
|
|
|
|
max_size += 1;
|
|
|
|
}
|
|
|
|
if (op_mask > max_size)
|
|
|
|
as_bad (_(":operand value(%d) too big for constraint"), op_mask);
|
|
|
|
}
|
2012-05-03 21:12:08 +08:00
|
|
|
else
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
/* Should be BFD_RELOC_XGATE_IMM8_LO instead of BFD_RELOC_XGATE_24
|
|
|
|
TODO fix. */
|
|
|
|
fixup_required = 1;
|
|
|
|
if (*op_constraint == '8')
|
|
|
|
{
|
|
|
|
if (((opcode->name[strlen (opcode->name) - 1] == 'l')
|
|
|
|
&& autoHiLo) || operand.mod == MOD_LOAD_LOW)
|
Use bool in gas
* as.h (POISON_BFD_BOOLEAN): Define.
* as.c, * as.h, * atof-generic.c, * config/atof-ieee.c,
* config/bfin-aux.h, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-elf.h, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h,
* config/tc-avr.c, * config/tc-avr.h, * config/tc-bfin.c,
* config/tc-bfin.h, * config/tc-bpf.c, * config/tc-cris.c,
* config/tc-csky.c, * config/tc-csky.h, * config/tc-d10v.c,
* config/tc-d10v.h, * config/tc-d30v.c, * config/tc-d30v.h,
* config/tc-dlx.c, * config/tc-dlx.h, * config/tc-epiphany.c,
* config/tc-epiphany.h, * config/tc-fr30.c, * config/tc-fr30.h,
* config/tc-frv.c, * config/tc-frv.h, * config/tc-ft32.c,
* config/tc-ft32.h, * config/tc-h8300.c, * config/tc-hppa.c,
* config/tc-i386-intel.c, * config/tc-i386.c, * config/tc-ia64.c,
* config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-iq2000.h,
* config/tc-lm32.c, * config/tc-lm32.h, * config/tc-m32c.c,
* config/tc-m32c.h, * config/tc-m32r.c, * config/tc-m32r.h,
* config/tc-m68hc11.c, * config/tc-m68k.c, * config/tc-mcore.c,
* config/tc-mcore.h, * config/tc-mep.c, * config/tc-mep.h,
* config/tc-metag.c, * config/tc-metag.h,
* config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mips.h,
* config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c,
* config/tc-mn10300.h, * config/tc-moxie.c, * config/tc-msp430.c,
* config/tc-msp430.h, * config/tc-mt.c, * config/tc-mt.h,
* config/tc-nds32.c, * config/tc-nds32.h, * config/tc-nios2.c,
* config/tc-ns32k.c, * config/tc-or1k.c, * config/tc-or1k.h,
* config/tc-pdp11.c, * config/tc-ppc.c, * config/tc-pru.c,
* config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h,
* config/tc-rx.c, * config/tc-rx.h, * config/tc-s12z.c,
* config/tc-s12z.h, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score.h, * config/tc-score7.c, * config/tc-sh.c,
* config/tc-sh.h, * config/tc-spu.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-tilegx.c,
* config/tc-tilepro.c, * config/tc-v850.c, * config/tc-v850.h,
* config/tc-visium.c, * config/tc-visium.h, * config/tc-wasm32.c,
* config/tc-wasm32.h, * config/tc-xc16x.c, * config/tc-xgate.c,
* config/tc-xstormy16.c, * config/tc-xstormy16.h,
* config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c,
* config/tc-z8k.c, * config/xtensa-istack.h,
* config/xtensa-relax.c, * config/xtensa-relax.h, * dw2gencfi.c,
* dwarf2dbg.c, * dwarf2dbg.h, * expr.c, * expr.h, * frags.c,
* frags.h, * listing.c, * macro.c, * output-file.c, * read.c,
* read.h, * stabs.c, * symbols.c, * write.c: Replace bfd_boolean
with bool, FALSE with false, and TRUE with true.
2021-03-31 08:12:05 +08:00
|
|
|
fix_new_exp (frag_now, where, 2, &operand.exp, false,
|
2012-11-06 18:03:32 +08:00
|
|
|
BFD_RELOC_XGATE_24);
|
|
|
|
else if (((opcode->name[strlen (opcode->name) - 1]) == 'h'
|
|
|
|
&& autoHiLo) || operand.mod == MOD_LOAD_HIGH )
|
Use bool in gas
* as.h (POISON_BFD_BOOLEAN): Define.
* as.c, * as.h, * atof-generic.c, * config/atof-ieee.c,
* config/bfin-aux.h, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-elf.h, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h,
* config/tc-avr.c, * config/tc-avr.h, * config/tc-bfin.c,
* config/tc-bfin.h, * config/tc-bpf.c, * config/tc-cris.c,
* config/tc-csky.c, * config/tc-csky.h, * config/tc-d10v.c,
* config/tc-d10v.h, * config/tc-d30v.c, * config/tc-d30v.h,
* config/tc-dlx.c, * config/tc-dlx.h, * config/tc-epiphany.c,
* config/tc-epiphany.h, * config/tc-fr30.c, * config/tc-fr30.h,
* config/tc-frv.c, * config/tc-frv.h, * config/tc-ft32.c,
* config/tc-ft32.h, * config/tc-h8300.c, * config/tc-hppa.c,
* config/tc-i386-intel.c, * config/tc-i386.c, * config/tc-ia64.c,
* config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-iq2000.h,
* config/tc-lm32.c, * config/tc-lm32.h, * config/tc-m32c.c,
* config/tc-m32c.h, * config/tc-m32r.c, * config/tc-m32r.h,
* config/tc-m68hc11.c, * config/tc-m68k.c, * config/tc-mcore.c,
* config/tc-mcore.h, * config/tc-mep.c, * config/tc-mep.h,
* config/tc-metag.c, * config/tc-metag.h,
* config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mips.h,
* config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c,
* config/tc-mn10300.h, * config/tc-moxie.c, * config/tc-msp430.c,
* config/tc-msp430.h, * config/tc-mt.c, * config/tc-mt.h,
* config/tc-nds32.c, * config/tc-nds32.h, * config/tc-nios2.c,
* config/tc-ns32k.c, * config/tc-or1k.c, * config/tc-or1k.h,
* config/tc-pdp11.c, * config/tc-ppc.c, * config/tc-pru.c,
* config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h,
* config/tc-rx.c, * config/tc-rx.h, * config/tc-s12z.c,
* config/tc-s12z.h, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score.h, * config/tc-score7.c, * config/tc-sh.c,
* config/tc-sh.h, * config/tc-spu.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-tilegx.c,
* config/tc-tilepro.c, * config/tc-v850.c, * config/tc-v850.h,
* config/tc-visium.c, * config/tc-visium.h, * config/tc-wasm32.c,
* config/tc-wasm32.h, * config/tc-xc16x.c, * config/tc-xgate.c,
* config/tc-xstormy16.c, * config/tc-xstormy16.h,
* config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c,
* config/tc-z8k.c, * config/xtensa-istack.h,
* config/xtensa-relax.c, * config/xtensa-relax.h, * dw2gencfi.c,
* dwarf2dbg.c, * dwarf2dbg.h, * expr.c, * expr.h, * frags.c,
* frags.h, * listing.c, * macro.c, * output-file.c, * read.c,
* read.h, * stabs.c, * symbols.c, * write.c: Replace bfd_boolean
with bool, FALSE with false, and TRUE with true.
2021-03-31 08:12:05 +08:00
|
|
|
fix_new_exp (frag_now, where, 2, &operand.exp, false,
|
2012-11-06 18:03:32 +08:00
|
|
|
BFD_RELOC_XGATE_IMM8_HI);
|
|
|
|
else
|
|
|
|
as_bad (_("you must use a hi/lo directive or 16-bit macro "
|
|
|
|
"to load a 16-bit value."));
|
|
|
|
}
|
|
|
|
else if (*op_constraint == '5')
|
Use bool in gas
* as.h (POISON_BFD_BOOLEAN): Define.
* as.c, * as.h, * atof-generic.c, * config/atof-ieee.c,
* config/bfin-aux.h, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-elf.h, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h,
* config/tc-avr.c, * config/tc-avr.h, * config/tc-bfin.c,
* config/tc-bfin.h, * config/tc-bpf.c, * config/tc-cris.c,
* config/tc-csky.c, * config/tc-csky.h, * config/tc-d10v.c,
* config/tc-d10v.h, * config/tc-d30v.c, * config/tc-d30v.h,
* config/tc-dlx.c, * config/tc-dlx.h, * config/tc-epiphany.c,
* config/tc-epiphany.h, * config/tc-fr30.c, * config/tc-fr30.h,
* config/tc-frv.c, * config/tc-frv.h, * config/tc-ft32.c,
* config/tc-ft32.h, * config/tc-h8300.c, * config/tc-hppa.c,
* config/tc-i386-intel.c, * config/tc-i386.c, * config/tc-ia64.c,
* config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-iq2000.h,
* config/tc-lm32.c, * config/tc-lm32.h, * config/tc-m32c.c,
* config/tc-m32c.h, * config/tc-m32r.c, * config/tc-m32r.h,
* config/tc-m68hc11.c, * config/tc-m68k.c, * config/tc-mcore.c,
* config/tc-mcore.h, * config/tc-mep.c, * config/tc-mep.h,
* config/tc-metag.c, * config/tc-metag.h,
* config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mips.h,
* config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c,
* config/tc-mn10300.h, * config/tc-moxie.c, * config/tc-msp430.c,
* config/tc-msp430.h, * config/tc-mt.c, * config/tc-mt.h,
* config/tc-nds32.c, * config/tc-nds32.h, * config/tc-nios2.c,
* config/tc-ns32k.c, * config/tc-or1k.c, * config/tc-or1k.h,
* config/tc-pdp11.c, * config/tc-ppc.c, * config/tc-pru.c,
* config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h,
* config/tc-rx.c, * config/tc-rx.h, * config/tc-s12z.c,
* config/tc-s12z.h, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score.h, * config/tc-score7.c, * config/tc-sh.c,
* config/tc-sh.h, * config/tc-spu.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-tilegx.c,
* config/tc-tilepro.c, * config/tc-v850.c, * config/tc-v850.h,
* config/tc-visium.c, * config/tc-visium.h, * config/tc-wasm32.c,
* config/tc-wasm32.h, * config/tc-xc16x.c, * config/tc-xgate.c,
* config/tc-xstormy16.c, * config/tc-xstormy16.h,
* config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c,
* config/tc-z8k.c, * config/xtensa-istack.h,
* config/xtensa-relax.c, * config/xtensa-relax.h, * dw2gencfi.c,
* dwarf2dbg.c, * dwarf2dbg.h, * expr.c, * expr.h, * frags.c,
* frags.h, * listing.c, * macro.c, * output-file.c, * read.c,
* read.h, * stabs.c, * symbols.c, * write.c: Replace bfd_boolean
with bool, FALSE with false, and TRUE with true.
2021-03-31 08:12:05 +08:00
|
|
|
fix_new_exp (frag_now, where, 2, &operand.exp, false,
|
2012-11-06 18:03:32 +08:00
|
|
|
BFD_RELOC_XGATE_IMM5);
|
|
|
|
else if (*op_constraint == '4')
|
Use bool in gas
* as.h (POISON_BFD_BOOLEAN): Define.
* as.c, * as.h, * atof-generic.c, * config/atof-ieee.c,
* config/bfin-aux.h, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-elf.h, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h,
* config/tc-avr.c, * config/tc-avr.h, * config/tc-bfin.c,
* config/tc-bfin.h, * config/tc-bpf.c, * config/tc-cris.c,
* config/tc-csky.c, * config/tc-csky.h, * config/tc-d10v.c,
* config/tc-d10v.h, * config/tc-d30v.c, * config/tc-d30v.h,
* config/tc-dlx.c, * config/tc-dlx.h, * config/tc-epiphany.c,
* config/tc-epiphany.h, * config/tc-fr30.c, * config/tc-fr30.h,
* config/tc-frv.c, * config/tc-frv.h, * config/tc-ft32.c,
* config/tc-ft32.h, * config/tc-h8300.c, * config/tc-hppa.c,
* config/tc-i386-intel.c, * config/tc-i386.c, * config/tc-ia64.c,
* config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-iq2000.h,
* config/tc-lm32.c, * config/tc-lm32.h, * config/tc-m32c.c,
* config/tc-m32c.h, * config/tc-m32r.c, * config/tc-m32r.h,
* config/tc-m68hc11.c, * config/tc-m68k.c, * config/tc-mcore.c,
* config/tc-mcore.h, * config/tc-mep.c, * config/tc-mep.h,
* config/tc-metag.c, * config/tc-metag.h,
* config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mips.h,
* config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c,
* config/tc-mn10300.h, * config/tc-moxie.c, * config/tc-msp430.c,
* config/tc-msp430.h, * config/tc-mt.c, * config/tc-mt.h,
* config/tc-nds32.c, * config/tc-nds32.h, * config/tc-nios2.c,
* config/tc-ns32k.c, * config/tc-or1k.c, * config/tc-or1k.h,
* config/tc-pdp11.c, * config/tc-ppc.c, * config/tc-pru.c,
* config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h,
* config/tc-rx.c, * config/tc-rx.h, * config/tc-s12z.c,
* config/tc-s12z.h, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score.h, * config/tc-score7.c, * config/tc-sh.c,
* config/tc-sh.h, * config/tc-spu.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-tilegx.c,
* config/tc-tilepro.c, * config/tc-v850.c, * config/tc-v850.h,
* config/tc-visium.c, * config/tc-visium.h, * config/tc-wasm32.c,
* config/tc-wasm32.h, * config/tc-xc16x.c, * config/tc-xgate.c,
* config/tc-xstormy16.c, * config/tc-xstormy16.h,
* config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c,
* config/tc-z8k.c, * config/xtensa-istack.h,
* config/xtensa-relax.c, * config/xtensa-relax.h, * dw2gencfi.c,
* dwarf2dbg.c, * dwarf2dbg.h, * expr.c, * expr.h, * frags.c,
* frags.h, * listing.c, * macro.c, * output-file.c, * read.c,
* read.h, * stabs.c, * symbols.c, * write.c: Replace bfd_boolean
with bool, FALSE with false, and TRUE with true.
2021-03-31 08:12:05 +08:00
|
|
|
fix_new_exp (frag_now, where, 2, &operand.exp, false,
|
2012-11-06 18:03:32 +08:00
|
|
|
BFD_RELOC_XGATE_IMM4);
|
|
|
|
else if (*op_constraint == '3')
|
Use bool in gas
* as.h (POISON_BFD_BOOLEAN): Define.
* as.c, * as.h, * atof-generic.c, * config/atof-ieee.c,
* config/bfin-aux.h, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-elf.h, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h,
* config/tc-avr.c, * config/tc-avr.h, * config/tc-bfin.c,
* config/tc-bfin.h, * config/tc-bpf.c, * config/tc-cris.c,
* config/tc-csky.c, * config/tc-csky.h, * config/tc-d10v.c,
* config/tc-d10v.h, * config/tc-d30v.c, * config/tc-d30v.h,
* config/tc-dlx.c, * config/tc-dlx.h, * config/tc-epiphany.c,
* config/tc-epiphany.h, * config/tc-fr30.c, * config/tc-fr30.h,
* config/tc-frv.c, * config/tc-frv.h, * config/tc-ft32.c,
* config/tc-ft32.h, * config/tc-h8300.c, * config/tc-hppa.c,
* config/tc-i386-intel.c, * config/tc-i386.c, * config/tc-ia64.c,
* config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-iq2000.h,
* config/tc-lm32.c, * config/tc-lm32.h, * config/tc-m32c.c,
* config/tc-m32c.h, * config/tc-m32r.c, * config/tc-m32r.h,
* config/tc-m68hc11.c, * config/tc-m68k.c, * config/tc-mcore.c,
* config/tc-mcore.h, * config/tc-mep.c, * config/tc-mep.h,
* config/tc-metag.c, * config/tc-metag.h,
* config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mips.h,
* config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c,
* config/tc-mn10300.h, * config/tc-moxie.c, * config/tc-msp430.c,
* config/tc-msp430.h, * config/tc-mt.c, * config/tc-mt.h,
* config/tc-nds32.c, * config/tc-nds32.h, * config/tc-nios2.c,
* config/tc-ns32k.c, * config/tc-or1k.c, * config/tc-or1k.h,
* config/tc-pdp11.c, * config/tc-ppc.c, * config/tc-pru.c,
* config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h,
* config/tc-rx.c, * config/tc-rx.h, * config/tc-s12z.c,
* config/tc-s12z.h, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score.h, * config/tc-score7.c, * config/tc-sh.c,
* config/tc-sh.h, * config/tc-spu.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-tilegx.c,
* config/tc-tilepro.c, * config/tc-v850.c, * config/tc-v850.h,
* config/tc-visium.c, * config/tc-visium.h, * config/tc-wasm32.c,
* config/tc-wasm32.h, * config/tc-xc16x.c, * config/tc-xgate.c,
* config/tc-xstormy16.c, * config/tc-xstormy16.h,
* config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c,
* config/tc-z8k.c, * config/xtensa-istack.h,
* config/xtensa-relax.c, * config/xtensa-relax.h, * dw2gencfi.c,
* dwarf2dbg.c, * dwarf2dbg.h, * expr.c, * expr.h, * frags.c,
* frags.h, * listing.c, * macro.c, * output-file.c, * read.c,
* read.h, * stabs.c, * symbols.c, * write.c: Replace bfd_boolean
with bool, FALSE with false, and TRUE with true.
2021-03-31 08:12:05 +08:00
|
|
|
fix_new_exp (frag_now, where, 2, &operand.exp, false,
|
2012-11-06 18:03:32 +08:00
|
|
|
BFD_RELOC_XGATE_IMM3);
|
|
|
|
else
|
|
|
|
as_bad (_(":unknown relocation constraint size"));
|
|
|
|
}
|
2012-07-28 06:33:22 +08:00
|
|
|
break;
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
case 'c': /* CCR register expected. */
|
2012-07-06 03:37:52 +08:00
|
|
|
*bit_width = 0;
|
2012-07-28 06:33:22 +08:00
|
|
|
if (operand.reg != REG_CCR)
|
2012-11-06 18:03:32 +08:00
|
|
|
as_bad (_(": expected register name ccr "));
|
2012-07-28 06:33:22 +08:00
|
|
|
break;
|
2012-05-03 21:12:08 +08:00
|
|
|
|
|
|
|
case 'p': /* PC register expected. */
|
2012-07-06 03:37:52 +08:00
|
|
|
*bit_width = 0;
|
2012-11-06 18:03:32 +08:00
|
|
|
if (operand.reg != REG_PC)
|
|
|
|
as_bad (_(": expected register name pc "));
|
2012-05-03 21:12:08 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'b': /* Branch expected. */
|
|
|
|
(*op_con)++;
|
|
|
|
op_constraint++;
|
|
|
|
|
2012-07-28 06:33:22 +08:00
|
|
|
if (operand.exp.X_op != O_register)
|
2012-11-06 18:03:32 +08:00
|
|
|
{
|
|
|
|
if (*op_constraint == '9')
|
Use bool in gas
* as.h (POISON_BFD_BOOLEAN): Define.
* as.c, * as.h, * atof-generic.c, * config/atof-ieee.c,
* config/bfin-aux.h, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-elf.h, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h,
* config/tc-avr.c, * config/tc-avr.h, * config/tc-bfin.c,
* config/tc-bfin.h, * config/tc-bpf.c, * config/tc-cris.c,
* config/tc-csky.c, * config/tc-csky.h, * config/tc-d10v.c,
* config/tc-d10v.h, * config/tc-d30v.c, * config/tc-d30v.h,
* config/tc-dlx.c, * config/tc-dlx.h, * config/tc-epiphany.c,
* config/tc-epiphany.h, * config/tc-fr30.c, * config/tc-fr30.h,
* config/tc-frv.c, * config/tc-frv.h, * config/tc-ft32.c,
* config/tc-ft32.h, * config/tc-h8300.c, * config/tc-hppa.c,
* config/tc-i386-intel.c, * config/tc-i386.c, * config/tc-ia64.c,
* config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-iq2000.h,
* config/tc-lm32.c, * config/tc-lm32.h, * config/tc-m32c.c,
* config/tc-m32c.h, * config/tc-m32r.c, * config/tc-m32r.h,
* config/tc-m68hc11.c, * config/tc-m68k.c, * config/tc-mcore.c,
* config/tc-mcore.h, * config/tc-mep.c, * config/tc-mep.h,
* config/tc-metag.c, * config/tc-metag.h,
* config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mips.h,
* config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c,
* config/tc-mn10300.h, * config/tc-moxie.c, * config/tc-msp430.c,
* config/tc-msp430.h, * config/tc-mt.c, * config/tc-mt.h,
* config/tc-nds32.c, * config/tc-nds32.h, * config/tc-nios2.c,
* config/tc-ns32k.c, * config/tc-or1k.c, * config/tc-or1k.h,
* config/tc-pdp11.c, * config/tc-ppc.c, * config/tc-pru.c,
* config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h,
* config/tc-rx.c, * config/tc-rx.h, * config/tc-s12z.c,
* config/tc-s12z.h, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score.h, * config/tc-score7.c, * config/tc-sh.c,
* config/tc-sh.h, * config/tc-spu.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-tilegx.c,
* config/tc-tilepro.c, * config/tc-v850.c, * config/tc-v850.h,
* config/tc-visium.c, * config/tc-visium.h, * config/tc-wasm32.c,
* config/tc-wasm32.h, * config/tc-xc16x.c, * config/tc-xgate.c,
* config/tc-xstormy16.c, * config/tc-xstormy16.h,
* config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c,
* config/tc-z8k.c, * config/xtensa-istack.h,
* config/xtensa-relax.c, * config/xtensa-relax.h, * dw2gencfi.c,
* dwarf2dbg.c, * dwarf2dbg.h, * expr.c, * expr.h, * frags.c,
* frags.h, * listing.c, * macro.c, * output-file.c, * read.c,
* read.h, * stabs.c, * symbols.c, * write.c: Replace bfd_boolean
with bool, FALSE with false, and TRUE with true.
2021-03-31 08:12:05 +08:00
|
|
|
fix_new_exp (frag_now, where, 2, &operand.exp, true,
|
2020-05-27 00:33:48 +08:00
|
|
|
BFD_RELOC_XGATE_PCREL_9);
|
2012-11-06 18:03:32 +08:00
|
|
|
else if (*op_constraint == 'a')
|
Use bool in gas
* as.h (POISON_BFD_BOOLEAN): Define.
* as.c, * as.h, * atof-generic.c, * config/atof-ieee.c,
* config/bfin-aux.h, * config/obj-coff.c, * config/obj-ecoff.c,
* config/obj-elf.c, * config/obj-elf.h, * config/obj-som.c,
* config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c,
* config/tc-arc.h, * config/tc-arm.c, * config/tc-arm.h,
* config/tc-avr.c, * config/tc-avr.h, * config/tc-bfin.c,
* config/tc-bfin.h, * config/tc-bpf.c, * config/tc-cris.c,
* config/tc-csky.c, * config/tc-csky.h, * config/tc-d10v.c,
* config/tc-d10v.h, * config/tc-d30v.c, * config/tc-d30v.h,
* config/tc-dlx.c, * config/tc-dlx.h, * config/tc-epiphany.c,
* config/tc-epiphany.h, * config/tc-fr30.c, * config/tc-fr30.h,
* config/tc-frv.c, * config/tc-frv.h, * config/tc-ft32.c,
* config/tc-ft32.h, * config/tc-h8300.c, * config/tc-hppa.c,
* config/tc-i386-intel.c, * config/tc-i386.c, * config/tc-ia64.c,
* config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-iq2000.h,
* config/tc-lm32.c, * config/tc-lm32.h, * config/tc-m32c.c,
* config/tc-m32c.h, * config/tc-m32r.c, * config/tc-m32r.h,
* config/tc-m68hc11.c, * config/tc-m68k.c, * config/tc-mcore.c,
* config/tc-mcore.h, * config/tc-mep.c, * config/tc-mep.h,
* config/tc-metag.c, * config/tc-metag.h,
* config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mips.h,
* config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c,
* config/tc-mn10300.h, * config/tc-moxie.c, * config/tc-msp430.c,
* config/tc-msp430.h, * config/tc-mt.c, * config/tc-mt.h,
* config/tc-nds32.c, * config/tc-nds32.h, * config/tc-nios2.c,
* config/tc-ns32k.c, * config/tc-or1k.c, * config/tc-or1k.h,
* config/tc-pdp11.c, * config/tc-ppc.c, * config/tc-pru.c,
* config/tc-pru.h, * config/tc-riscv.c, * config/tc-riscv.h,
* config/tc-rx.c, * config/tc-rx.h, * config/tc-s12z.c,
* config/tc-s12z.h, * config/tc-s390.c, * config/tc-score.c,
* config/tc-score.h, * config/tc-score7.c, * config/tc-sh.c,
* config/tc-sh.h, * config/tc-spu.c, * config/tc-tic54x.c,
* config/tc-tic6x.c, * config/tc-tic6x.h, * config/tc-tilegx.c,
* config/tc-tilepro.c, * config/tc-v850.c, * config/tc-v850.h,
* config/tc-visium.c, * config/tc-visium.h, * config/tc-wasm32.c,
* config/tc-wasm32.h, * config/tc-xc16x.c, * config/tc-xgate.c,
* config/tc-xstormy16.c, * config/tc-xstormy16.h,
* config/tc-xtensa.c, * config/tc-xtensa.h, * config/tc-z80.c,
* config/tc-z8k.c, * config/xtensa-istack.h,
* config/xtensa-relax.c, * config/xtensa-relax.h, * dw2gencfi.c,
* dwarf2dbg.c, * dwarf2dbg.h, * expr.c, * expr.h, * frags.c,
* frags.h, * listing.c, * macro.c, * output-file.c, * read.c,
* read.h, * stabs.c, * symbols.c, * write.c: Replace bfd_boolean
with bool, FALSE with false, and TRUE with true.
2021-03-31 08:12:05 +08:00
|
|
|
fix_new_exp (frag_now, where, 2, &operand.exp, true,
|
2020-05-27 00:33:48 +08:00
|
|
|
BFD_RELOC_XGATE_PCREL_10);
|
2012-11-06 18:03:32 +08:00
|
|
|
}
|
2012-07-06 03:37:52 +08:00
|
|
|
else
|
2012-11-06 18:03:32 +08:00
|
|
|
as_fatal (_("Operand `%x' not recognized in fixup8."),
|
|
|
|
operand.exp.X_op);
|
2012-07-06 03:37:52 +08:00
|
|
|
break;
|
|
|
|
case '?':
|
|
|
|
break;
|
2012-06-29 05:36:35 +08:00
|
|
|
|
2012-07-06 03:37:52 +08:00
|
|
|
default:
|
|
|
|
as_bad (_("unknown constraint `%c'"), *op_constraint);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return op_mask;
|
2012-06-29 05:36:35 +08:00
|
|
|
}
|