2018-05-18 22:26:18 +08:00
|
|
|
|
/* tc-s12z.c -- Assembler code for the Freescale S12Z
|
2020-01-01 15:57:01 +08:00
|
|
|
|
Copyright (C) 2018-2020 Free Software Foundation, Inc.
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
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 "dwarf2dbg.h"
|
2018-07-11 16:42:01 +08:00
|
|
|
|
#include "opcode/s12z.h"
|
2018-05-18 22:26:18 +08:00
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
|
|
|
|
|
const char comment_chars[] = ";";
|
|
|
|
|
|
|
|
|
|
const char line_comment_chars[] = "#*";
|
|
|
|
|
const char line_separator_chars[] = "";
|
|
|
|
|
|
2019-05-15 20:16:33 +08:00
|
|
|
|
static char * register_prefix = NULL;
|
|
|
|
|
|
2018-05-18 22:26:18 +08:00
|
|
|
|
const char EXP_CHARS[] = "eE";
|
|
|
|
|
const char FLT_CHARS[] = "dD";
|
|
|
|
|
|
|
|
|
|
static char *fail_line_pointer;
|
|
|
|
|
|
2019-05-22 13:16:14 +08:00
|
|
|
|
/* A wrapper around the standard library's strtol.
|
|
|
|
|
It converts STR into an integral value.
|
|
|
|
|
This wrapper deals with literal_prefix_dollar_hex. */
|
|
|
|
|
static long
|
|
|
|
|
s12z_strtol (const char *str, char ** endptr)
|
|
|
|
|
{
|
|
|
|
|
int base = 0;
|
|
|
|
|
bool negative = false;
|
|
|
|
|
|
|
|
|
|
long result = 0;
|
|
|
|
|
|
|
|
|
|
char *start = (char *) str;
|
|
|
|
|
|
|
|
|
|
/* In the case where literal_prefix_dollar_hex is TRUE the sign has
|
|
|
|
|
to be handled explicitly. Otherwise the string will not be
|
|
|
|
|
recognised as an integer. */
|
|
|
|
|
if (str[0] == '-')
|
|
|
|
|
{
|
|
|
|
|
negative = true;
|
|
|
|
|
++str;
|
|
|
|
|
}
|
|
|
|
|
else if (str[0] == '+')
|
|
|
|
|
{
|
|
|
|
|
++str;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (literal_prefix_dollar_hex && (str[0] == '$'))
|
|
|
|
|
{
|
|
|
|
|
base = 16;
|
|
|
|
|
str++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = strtol (str, endptr, base);
|
|
|
|
|
if (*endptr == str)
|
|
|
|
|
{
|
|
|
|
|
*endptr = start;
|
|
|
|
|
}
|
|
|
|
|
if (negative)
|
|
|
|
|
result = -result;
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
/* Options and initialization. */
|
|
|
|
|
|
2019-05-15 20:16:33 +08:00
|
|
|
|
const char *md_shortopts = "";
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
struct option md_longopts[] =
|
|
|
|
|
{
|
2019-05-22 13:16:14 +08:00
|
|
|
|
#define OPTION_REG_PREFIX (OPTION_MD_BASE)
|
|
|
|
|
{"mreg-prefix", required_argument, NULL, OPTION_REG_PREFIX},
|
|
|
|
|
#define OPTION_DOLLAR_HEX (OPTION_MD_BASE + 1)
|
|
|
|
|
{"mdollar-hex", no_argument, NULL, OPTION_DOLLAR_HEX},
|
2019-05-15 20:16:33 +08:00
|
|
|
|
{NULL, no_argument, NULL, 0}
|
2018-05-18 22:26:18 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
size_t md_longopts_size = sizeof (md_longopts);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
relax_typeS md_relax_table[] =
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* 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[] =
|
|
|
|
|
{
|
|
|
|
|
{0, 0, 0}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Get the target cpu for the assembler. */
|
|
|
|
|
const char *
|
|
|
|
|
s12z_arch_format (void)
|
|
|
|
|
{
|
|
|
|
|
return "elf32-s12z";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
enum bfd_architecture
|
|
|
|
|
s12z_arch (void)
|
|
|
|
|
{
|
|
|
|
|
return bfd_arch_s12z;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
s12z_mach (void)
|
|
|
|
|
{
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Listing header selected according to cpu. */
|
|
|
|
|
const char *
|
|
|
|
|
s12z_listing_header (void)
|
|
|
|
|
{
|
|
|
|
|
return "S12Z GAS ";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2019-05-15 20:16:33 +08:00
|
|
|
|
md_show_usage (FILE *stream)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
2019-05-22 13:16:14 +08:00
|
|
|
|
fputs (_("\ns12z options:\n"), stream);
|
|
|
|
|
fputs (_(" -mreg-prefix=PREFIX set a prefix used to indicate register names (default none)\n"), stream);
|
|
|
|
|
fputs (_(" -mdollar-hex the prefix '$' instead of '0x' is used to indicate literal hexadecimal constants\n"), stream);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
s12z_print_statistics (FILE *file ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
2019-05-15 20:16:33 +08:00
|
|
|
|
md_parse_option (int c, const char *arg)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
2019-05-15 20:16:33 +08:00
|
|
|
|
switch (c)
|
|
|
|
|
{
|
2019-05-22 13:16:14 +08:00
|
|
|
|
case OPTION_REG_PREFIX:
|
2019-05-15 20:16:33 +08:00
|
|
|
|
register_prefix = xstrdup (arg);
|
|
|
|
|
break;
|
2019-05-22 13:16:14 +08:00
|
|
|
|
case OPTION_DOLLAR_HEX:
|
|
|
|
|
literal_prefix_dollar_hex = TRUE;
|
|
|
|
|
break;
|
2019-05-15 20:16:33 +08:00
|
|
|
|
default:
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
return 1;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
symbolS *
|
|
|
|
|
md_undefined_symbol (char *name ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
|
md_atof (int type, char *litP, int *sizeP)
|
|
|
|
|
{
|
|
|
|
|
return ieee_md_atof (type, litP, sizeP, TRUE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
return ((addr + (1 << align) - 1) & -(1 << align));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
md_begin (void)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
s12z_init_after_args (void)
|
|
|
|
|
{
|
2019-05-22 13:16:14 +08:00
|
|
|
|
if (flag_traditional_format)
|
|
|
|
|
literal_prefix_dollar_hex = TRUE;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Builtin help. */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
skip_whites (char *p)
|
|
|
|
|
{
|
|
|
|
|
while (*p == ' ' || *p == '\t')
|
|
|
|
|
p++;
|
|
|
|
|
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Start a new insn that contains at least 'size' bytes. Record the
|
|
|
|
|
line information of that insn in the dwarf2 debug sections. */
|
|
|
|
|
static char *
|
|
|
|
|
s12z_new_insn (int size)
|
|
|
|
|
{
|
|
|
|
|
char *f = frag_more (size);
|
|
|
|
|
|
|
|
|
|
dwarf2_emit_insn (size);
|
|
|
|
|
|
|
|
|
|
return f;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean lex_reg_name (uint16_t which, int *reg);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_constant (long *v)
|
|
|
|
|
{
|
|
|
|
|
char *end = NULL;
|
|
|
|
|
char *p = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
/* A constant may not have the same value as a register
|
|
|
|
|
eg: "d6" */
|
|
|
|
|
int dummy;
|
|
|
|
|
if (lex_reg_name (~0, &dummy))
|
|
|
|
|
{
|
|
|
|
|
input_line_pointer = p;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
errno = 0;
|
2019-05-22 13:16:14 +08:00
|
|
|
|
*v = s12z_strtol (p, &end);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
if (errno == 0 && end != p)
|
|
|
|
|
{
|
|
|
|
|
input_line_pointer = end;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_match (char x)
|
|
|
|
|
{
|
|
|
|
|
char *p = input_line_pointer;
|
|
|
|
|
if (*p != x)
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
input_line_pointer++;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_expression (expressionS *exp)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int dummy;
|
|
|
|
|
exp->X_op = O_absent;
|
|
|
|
|
|
|
|
|
|
if (lex_match ('#'))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (lex_reg_name (~0, &dummy))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
expression (exp);
|
|
|
|
|
if (exp->X_op != O_absent)
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-02 00:42:54 +08:00
|
|
|
|
/* Immediate operand.
|
|
|
|
|
If EXP_O is non-null, then a symbolic expression is permitted,
|
|
|
|
|
in which case, EXP_O will be populated with the parsed expression.
|
|
|
|
|
*/
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2019-02-02 00:42:54 +08:00
|
|
|
|
lex_imm (long *v, expressionS *exp_o)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
if (*input_line_pointer != '#')
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
input_line_pointer++;
|
|
|
|
|
expressionS exp;
|
|
|
|
|
if (!lex_expression (&exp))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (exp.X_op != O_constant)
|
2019-02-02 00:42:54 +08:00
|
|
|
|
{
|
|
|
|
|
if (!exp_o)
|
|
|
|
|
as_bad (_("A non-constant expression is not permitted here"));
|
|
|
|
|
else
|
|
|
|
|
*exp_o = exp;
|
|
|
|
|
}
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
*v = exp.X_add_number;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
2020-02-25 13:04:46 +08:00
|
|
|
|
fail:
|
2018-05-18 22:26:18 +08:00
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Short mmediate operand */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_imm_e4 (long *val)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if ((lex_imm (val, NULL)))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
if ((*val == -1) || (*val > 0 && *val <= 15))
|
|
|
|
|
{
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_match_string (const char *s)
|
|
|
|
|
{
|
|
|
|
|
char *p = input_line_pointer;
|
|
|
|
|
while (p != 0 && *p != '\t' && *p != ' ' && *p != '\0')
|
|
|
|
|
{
|
|
|
|
|
p++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t len = p - input_line_pointer;
|
|
|
|
|
if (len != strlen (s))
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
if (0 == strncasecmp (s, input_line_pointer, len))
|
|
|
|
|
{
|
|
|
|
|
input_line_pointer = p;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Parse a register name.
|
|
|
|
|
WHICH is a ORwise combination of the registers which are accepted.
|
|
|
|
|
~0 accepts all.
|
|
|
|
|
On success, REG will be filled with the index of the register which
|
|
|
|
|
was successfully scanned.
|
|
|
|
|
*/
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_reg_name (uint16_t which, int *reg)
|
|
|
|
|
{
|
|
|
|
|
char *p = input_line_pointer;
|
2019-05-15 20:16:33 +08:00
|
|
|
|
|
|
|
|
|
if (p == 0)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
/* Scan (and ignore) the register prefix. */
|
|
|
|
|
if (register_prefix)
|
|
|
|
|
{
|
|
|
|
|
int len = strlen (register_prefix);
|
|
|
|
|
if (0 == strncmp (register_prefix, p, len))
|
|
|
|
|
p += len;
|
|
|
|
|
else
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *start_of_reg_name = p;
|
|
|
|
|
|
|
|
|
|
while ((*p >= 'a' && *p <='z')
|
|
|
|
|
|| (*p >= '0' && *p <= '9')
|
|
|
|
|
|| (*p >= 'A' && *p <='Z'))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
p++;
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-15 20:16:33 +08:00
|
|
|
|
size_t len = p - start_of_reg_name;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
if (len <= 0)
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i < S12Z_N_REGISTERS; ++i)
|
|
|
|
|
{
|
|
|
|
|
gas_assert (registers[i].name);
|
|
|
|
|
|
2019-01-14 23:55:17 +08:00
|
|
|
|
if (len == strlen (registers[i].name)
|
2019-05-15 20:16:33 +08:00
|
|
|
|
&& 0 == strncasecmp (registers[i].name, start_of_reg_name, len))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
if ((0x1U << i) & which)
|
|
|
|
|
{
|
|
|
|
|
input_line_pointer = p;
|
|
|
|
|
*reg = i;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
lex_force_match (char x)
|
|
|
|
|
{
|
|
|
|
|
char *p = input_line_pointer;
|
|
|
|
|
if (*p != x)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Expecting '%c'"), x);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
input_line_pointer++;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2019-02-02 00:42:54 +08:00
|
|
|
|
lex_opr (uint8_t *buffer, int *n_bytes, expressionS *exp,
|
|
|
|
|
bool immediate_ok)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
uint8_t *xb = buffer;
|
|
|
|
|
int reg;
|
|
|
|
|
long imm;
|
|
|
|
|
exp->X_op = O_absent;
|
|
|
|
|
*n_bytes = 0;
|
|
|
|
|
*xb = 0;
|
|
|
|
|
if (lex_imm_e4 (&imm))
|
|
|
|
|
{
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!immediate_ok)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("An immediate value in a source operand is inappropriate"));
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
}
|
2018-05-18 22:26:18 +08:00
|
|
|
|
if (imm > 0)
|
|
|
|
|
*xb = imm;
|
|
|
|
|
else
|
|
|
|
|
*xb = 0;
|
|
|
|
|
*xb |= 0x70;
|
|
|
|
|
*n_bytes = 1;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
else if (lex_reg_name (REG_BIT_Dn, ®))
|
|
|
|
|
{
|
|
|
|
|
*xb = reg;
|
|
|
|
|
*xb |= 0xb8;
|
|
|
|
|
*n_bytes = 1;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
else if (lex_match ('['))
|
|
|
|
|
{
|
|
|
|
|
if (lex_expression (exp))
|
|
|
|
|
{
|
|
|
|
|
long c = exp->X_add_number;
|
|
|
|
|
if (lex_match (','))
|
|
|
|
|
{
|
|
|
|
|
if (lex_reg_name (REG_BIT_XYSP, ®))
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
if (c <= 255 && c >= -256)
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 2;
|
|
|
|
|
*xb |= 0xc4;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 4;
|
|
|
|
|
*xb |= 0xc6;
|
|
|
|
|
}
|
|
|
|
|
*xb |= (reg - REG_X) << 4;
|
|
|
|
|
|
|
|
|
|
if (c < 0)
|
|
|
|
|
*xb |= 0x01;
|
|
|
|
|
for (i = 1; i < *n_bytes ; ++i)
|
|
|
|
|
{
|
|
|
|
|
buffer[i] = c >> (8 * (*n_bytes - i - 1));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Bad operand for constant offset"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*xb = 0xfe;
|
|
|
|
|
*n_bytes = 4;
|
|
|
|
|
buffer[1] = c >> 16;
|
|
|
|
|
buffer[2] = c >> 8;
|
|
|
|
|
buffer[3] = c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (lex_reg_name (REG_BIT_Dn, ®))
|
|
|
|
|
{
|
|
|
|
|
if (!lex_force_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int reg2;
|
|
|
|
|
if (lex_reg_name (REG_BIT_XY, ®2))
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 1;
|
|
|
|
|
*xb = reg;
|
|
|
|
|
*xb |= (reg2 - REG_X) << 4;
|
|
|
|
|
*xb |= 0xc8;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid operand for register offset"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
if (!lex_force_match (']'))
|
|
|
|
|
goto fail;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
else if (lex_match ('('))
|
|
|
|
|
{
|
|
|
|
|
long c;
|
|
|
|
|
if (lex_constant (&c))
|
|
|
|
|
{
|
|
|
|
|
if (!lex_force_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
int reg2;
|
|
|
|
|
if (lex_reg_name (REG_BIT_XYSP, ®2))
|
|
|
|
|
{
|
|
|
|
|
if (reg2 != REG_P && c >= 0 && c <= 15)
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 1;
|
|
|
|
|
*xb = 0x40;
|
|
|
|
|
*xb |= (reg2 - REG_X) << 4;
|
|
|
|
|
*xb |= c;
|
|
|
|
|
}
|
|
|
|
|
else if (c >= -256 && c <= 255)
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 2;
|
|
|
|
|
*xb = 0xc0;
|
|
|
|
|
*xb |= (reg2 - REG_X) << 4;
|
|
|
|
|
if (c < 0)
|
|
|
|
|
*xb |= 0x01;
|
|
|
|
|
buffer[1] = c;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 4;
|
|
|
|
|
*xb = 0xc2;
|
|
|
|
|
*xb |= (reg2 - REG_X) << 4;
|
|
|
|
|
buffer[1] = c >> 16;
|
|
|
|
|
buffer[2] = c >> 8;
|
|
|
|
|
buffer[3] = c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (lex_reg_name (REG_BIT_Dn, ®2))
|
|
|
|
|
{
|
|
|
|
|
if (c >= -1 * (long) (0x1u << 17)
|
|
|
|
|
&&
|
|
|
|
|
c < (long) (0x1u << 17) - 1)
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 3;
|
|
|
|
|
*xb = 0x80;
|
|
|
|
|
*xb |= reg2;
|
|
|
|
|
*xb |= ((c >> 16) & 0x03) << 4;
|
|
|
|
|
buffer[1] = c >> 8;
|
|
|
|
|
buffer[2] = c;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 4;
|
|
|
|
|
*xb = 0xe8;
|
|
|
|
|
*xb |= reg2;
|
|
|
|
|
buffer[1] = c >> 16;
|
|
|
|
|
buffer[2] = c >> 8;
|
|
|
|
|
buffer[3] = c;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Bad operand for constant offset"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (lex_reg_name (REG_BIT_Dn, ®))
|
|
|
|
|
{
|
|
|
|
|
if (lex_match (','))
|
|
|
|
|
{
|
|
|
|
|
int reg2;
|
|
|
|
|
if (lex_reg_name (REG_BIT_XYS, ®2))
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 1;
|
|
|
|
|
*xb = 0x88;
|
|
|
|
|
*xb |= (reg2 - REG_X) << 4;
|
|
|
|
|
*xb |= reg;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid operand for register offset"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (lex_reg_name (REG_BIT_XYS, ®))
|
|
|
|
|
{
|
|
|
|
|
if (lex_match ('-'))
|
|
|
|
|
{
|
|
|
|
|
if (reg == REG_S)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid register for postdecrement operation"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
*n_bytes = 1;
|
|
|
|
|
if (reg == REG_X)
|
|
|
|
|
*xb = 0xc7;
|
|
|
|
|
else if (reg == REG_Y)
|
|
|
|
|
*xb = 0xd7;
|
|
|
|
|
}
|
|
|
|
|
else if (lex_match ('+'))
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 1;
|
|
|
|
|
if (reg == REG_X)
|
|
|
|
|
*xb = 0xe7;
|
|
|
|
|
else if (reg == REG_Y)
|
|
|
|
|
*xb = 0xf7;
|
|
|
|
|
else if (reg == REG_S)
|
|
|
|
|
*xb = 0xff;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (lex_match ('+'))
|
|
|
|
|
{
|
|
|
|
|
if (lex_reg_name (REG_BIT_XY, ®))
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 1;
|
|
|
|
|
if (reg == REG_X)
|
|
|
|
|
*xb = 0xe3;
|
|
|
|
|
else if (reg == REG_Y)
|
|
|
|
|
*xb = 0xf3;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid register for preincrement operation"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (lex_match ('-'))
|
|
|
|
|
{
|
|
|
|
|
if (lex_reg_name (REG_BIT_XYS, ®))
|
|
|
|
|
{
|
|
|
|
|
*n_bytes = 1;
|
|
|
|
|
if (reg == REG_X)
|
|
|
|
|
*xb = 0xc3;
|
|
|
|
|
else if (reg == REG_Y)
|
|
|
|
|
*xb = 0xd3;
|
|
|
|
|
else if (reg == REG_S)
|
|
|
|
|
*xb = 0xfb;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid register for predecrement operation"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (! lex_match (')'))
|
|
|
|
|
goto fail;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
else if (lex_expression (exp))
|
|
|
|
|
{
|
|
|
|
|
*xb = 0xfa;
|
|
|
|
|
*n_bytes = 4;
|
|
|
|
|
buffer[1] = 0;
|
|
|
|
|
buffer[2] = 0;
|
|
|
|
|
buffer[3] = 0;
|
|
|
|
|
if (exp->X_op == O_constant)
|
|
|
|
|
{
|
2018-09-20 22:49:01 +08:00
|
|
|
|
valueT value = exp->X_add_number;
|
|
|
|
|
|
|
|
|
|
if (value < (0x1U << 14))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
*xb = 0x00;
|
|
|
|
|
*n_bytes = 2;
|
2018-09-20 22:49:01 +08:00
|
|
|
|
*xb |= value >> 8;
|
|
|
|
|
buffer[1] = value;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
2018-09-20 22:49:01 +08:00
|
|
|
|
else if (value < (0x1U << 19))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
*xb = 0xf8;
|
2018-09-20 22:49:01 +08:00
|
|
|
|
if (value & (0x1U << 17))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
*xb |= 0x04;
|
2018-09-20 22:49:01 +08:00
|
|
|
|
if (value & (0x1U << 16))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
*xb |= 0x01;
|
|
|
|
|
*n_bytes = 3;
|
2018-09-20 22:49:01 +08:00
|
|
|
|
buffer[1] = value >> 8;
|
|
|
|
|
buffer[2] = value;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*xb = 0xfa;
|
|
|
|
|
*n_bytes = 4;
|
2018-09-20 22:49:01 +08:00
|
|
|
|
buffer[1] = value >> 16;
|
|
|
|
|
buffer[2] = value >> 8;
|
|
|
|
|
buffer[3] = value;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_offset (long *val)
|
|
|
|
|
{
|
|
|
|
|
char *end = NULL;
|
|
|
|
|
char *p = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
if (*p++ != '*')
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
if (*p != '+' && *p != '-')
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
bool negative = (*p == '-');
|
|
|
|
|
p++;
|
|
|
|
|
|
|
|
|
|
errno = 0;
|
2019-05-22 13:16:14 +08:00
|
|
|
|
*val = s12z_strtol (p, &end);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
if (errno == 0)
|
|
|
|
|
{
|
|
|
|
|
if (negative)
|
|
|
|
|
*val *= -1;
|
|
|
|
|
input_line_pointer = end;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct instruction;
|
|
|
|
|
|
|
|
|
|
typedef int (*parse_operand_func) (const struct instruction *);
|
|
|
|
|
|
|
|
|
|
struct instruction
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
|
|
/* The "page" to which the instruction belongs.
|
|
|
|
|
This is also only a hint. Some instructions might have modes in both
|
|
|
|
|
pages... */
|
|
|
|
|
char page;
|
|
|
|
|
|
|
|
|
|
/* This is a hint - and only a hint - about the opcode of the instruction.
|
|
|
|
|
The parse_operand_func is free to ignore it.
|
|
|
|
|
*/
|
|
|
|
|
uint8_t opc;
|
|
|
|
|
|
|
|
|
|
parse_operand_func parse_operands;
|
|
|
|
|
|
|
|
|
|
/* Some instructions can be encoded with a different opcode */
|
|
|
|
|
uint8_t alt_opc;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
no_operands (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
if (*input_line_pointer != '\0')
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Garbage at end of instruction"));
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (insn->page);
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-02 00:42:54 +08:00
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
emit_reloc (expressionS *exp, char *f, int size, enum bfd_reloc_code_real reloc)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
if (exp->X_op != O_absent && exp->X_op != O_constant)
|
|
|
|
|
{
|
2019-01-16 21:34:50 +08:00
|
|
|
|
fixS *fix = fix_new_exp (frag_now,
|
|
|
|
|
f - frag_now->fr_literal,
|
2019-02-02 00:42:54 +08:00
|
|
|
|
size,
|
2019-01-16 21:34:50 +08:00
|
|
|
|
exp,
|
|
|
|
|
FALSE,
|
2019-02-02 00:42:54 +08:00
|
|
|
|
reloc);
|
2019-01-16 21:34:50 +08:00
|
|
|
|
/* Some third party tools seem to use the lower bits
|
2019-02-02 00:42:54 +08:00
|
|
|
|
of this addend for flags. They don't get added
|
|
|
|
|
to the final location. The purpose of these flags
|
|
|
|
|
is not known. We simply set it to zero. */
|
2019-01-16 21:34:50 +08:00
|
|
|
|
fix->fx_addnumber = 0x00;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
2019-02-02 00:42:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Emit the code for an OPR address mode operand */
|
|
|
|
|
static char *
|
|
|
|
|
emit_opr (char *f, const uint8_t *buffer, int n_bytes, expressionS *exp)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
number_to_chars_bigendian (f++, buffer[0], 1);
|
|
|
|
|
|
|
|
|
|
emit_reloc (exp, f, 3, BFD_RELOC_S12Z_OPR);
|
|
|
|
|
|
2018-05-18 22:26:18 +08:00
|
|
|
|
for (i = 1; i < n_bytes; ++i)
|
|
|
|
|
number_to_chars_bigendian (f++, buffer[i], 1);
|
|
|
|
|
|
|
|
|
|
return f;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Emit the code for a 24 bit direct address operand */
|
|
|
|
|
static char *
|
|
|
|
|
emit_ext24 (char *f, long v)
|
|
|
|
|
{
|
|
|
|
|
number_to_chars_bigendian (f, v, 3);
|
|
|
|
|
|
|
|
|
|
return f + 3;
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
opr (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
/* Large constant direct values are more efficiently encoded as ext24 mode.
|
|
|
|
|
Otherwise a decision has to be deferred to a relax. */
|
|
|
|
|
if (exp.X_op == O_constant
|
|
|
|
|
&& buffer[0] == 0xFA
|
|
|
|
|
&& insn->alt_opc != 0)
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (4);
|
|
|
|
|
|
|
|
|
|
/* I don't think there are any instances of page 2 opcodes in this case */
|
|
|
|
|
gas_assert (insn->page == 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->alt_opc, 1);
|
|
|
|
|
|
|
|
|
|
emit_ext24 (f, exp.X_add_number);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (n_bytes + 1);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
}
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Parse a 15 bit offset, as an expression.
|
|
|
|
|
LONG_DISPLACEMENT will be set to true if the offset is wider than 7 bits.
|
|
|
|
|
*/
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_15_bit_offset (bool *long_displacement, expressionS *exp)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
long val;
|
|
|
|
|
if (lex_offset (&val))
|
|
|
|
|
{
|
|
|
|
|
exp->X_op = O_absent;
|
|
|
|
|
exp->X_add_number = val;
|
|
|
|
|
}
|
|
|
|
|
else if (lex_expression (exp))
|
|
|
|
|
{
|
|
|
|
|
if (exp->X_op == O_constant)
|
|
|
|
|
{
|
|
|
|
|
val = exp->X_add_number;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* If a symbol was parsed we don't know the displacement.
|
|
|
|
|
We have to assume it is long, and relax it later if possible. */
|
|
|
|
|
*long_displacement = true;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
exp->X_op = O_absent;
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (val > 0x3FFF || val < -0x4000)
|
|
|
|
|
{
|
|
|
|
|
as_fatal (_("Offset is outside of 15 bit range"));
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*long_displacement = (val > 63 || val < -64);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
emit_15_bit_offset (char *f, int where, expressionS *exp)
|
|
|
|
|
{
|
|
|
|
|
gas_assert (exp);
|
|
|
|
|
if (exp->X_op != O_absent && exp->X_op != O_constant)
|
|
|
|
|
{
|
|
|
|
|
exp->X_add_number += where;
|
|
|
|
|
fixS *fix = fix_new_exp (frag_now,
|
|
|
|
|
f - frag_now->fr_literal,
|
|
|
|
|
2,
|
|
|
|
|
exp,
|
|
|
|
|
TRUE,
|
|
|
|
|
BFD_RELOC_16_PCREL);
|
|
|
|
|
fix->fx_addnumber = where - 2;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
long val = exp->X_add_number;
|
|
|
|
|
bool long_displacement = (val > 63 || val < -64);
|
|
|
|
|
if (long_displacement)
|
|
|
|
|
val |= 0x8000;
|
|
|
|
|
else
|
|
|
|
|
val &= 0x7F;
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, val, long_displacement ? 2 : 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
rel (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
bool long_displacement;
|
|
|
|
|
|
|
|
|
|
expressionS exp;
|
|
|
|
|
if (! lex_15_bit_offset (&long_displacement, &exp))
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (long_displacement ? 3 : 2);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
emit_15_bit_offset (f, 3, &exp);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
reg_inh (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
int reg;
|
|
|
|
|
if (lex_reg_name (REG_BIT_Dn, ®))
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (insn->page);
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc + reg, 1);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Special case for CLR X and CLR Y */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
clr_xy (const struct instruction *insn ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
int reg;
|
|
|
|
|
if (lex_reg_name (REG_BIT_XY, ®))
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (1);
|
|
|
|
|
number_to_chars_bigendian (f, 0x9a + reg - REG_X, 1);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Some instructions have a suffix like ".l", ".b", ".w" etc
|
|
|
|
|
which indicates the size of the operands. */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
size_from_suffix (const struct instruction *insn, int idx)
|
|
|
|
|
{
|
|
|
|
|
const char *dot = strchr (insn->name, '.');
|
|
|
|
|
|
|
|
|
|
if (dot == NULL)
|
|
|
|
|
return -3;
|
|
|
|
|
|
|
|
|
|
int size = -2;
|
|
|
|
|
switch (dot[1 + idx])
|
|
|
|
|
{
|
|
|
|
|
case 'b':
|
|
|
|
|
size = 1;
|
|
|
|
|
break;
|
|
|
|
|
case 'w':
|
|
|
|
|
size = 2;
|
|
|
|
|
break;
|
|
|
|
|
case 'p':
|
|
|
|
|
size = 3;
|
|
|
|
|
break;
|
|
|
|
|
case 'l':
|
|
|
|
|
size = 4;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
as_fatal (_("Bad size"));
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return size;
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
mul_reg_reg_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
int Dd;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dj;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dj))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dk;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dk))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (insn->page + 1);
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc + Dd, 1);
|
|
|
|
|
const char *dot = strchrnul (insn->name, '.');
|
|
|
|
|
uint8_t mb ;
|
|
|
|
|
switch (dot[-1])
|
|
|
|
|
{
|
|
|
|
|
case 's':
|
|
|
|
|
mb = 0x80;
|
|
|
|
|
break;
|
|
|
|
|
case 'u':
|
|
|
|
|
mb = 0x00;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
as_fatal (_("BAD MUL"));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mb |= Dj << 3;
|
|
|
|
|
mb |= Dk;
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, mb, 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
mul_reg_reg_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
int Dd;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dj;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dj))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long imm;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (insn->page + 1 + size);
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc + Dd, 1);
|
|
|
|
|
uint8_t mb = 0x44;
|
|
|
|
|
const char *dot = strchrnul (insn->name, '.');
|
|
|
|
|
switch (dot[-1])
|
|
|
|
|
{
|
|
|
|
|
case 's':
|
|
|
|
|
mb |= 0x80;
|
|
|
|
|
break;
|
|
|
|
|
case 'u':
|
|
|
|
|
mb |= 0x00;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
as_fatal (_("BAD MUL"));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mb |= Dj << 3;
|
|
|
|
|
mb |= size - 1;
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, mb, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, imm, size);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
mul_reg_reg_opr (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
int Dd;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dj;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dj))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, true))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (insn->page + 1 + n_bytes);
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc + Dd, 1);
|
|
|
|
|
uint8_t mb = 0x40;
|
|
|
|
|
const char *dot = strchrnul (insn->name, '.');
|
|
|
|
|
switch (dot[-1])
|
|
|
|
|
{
|
|
|
|
|
case 's':
|
|
|
|
|
mb |= 0x80;
|
|
|
|
|
break;
|
|
|
|
|
case 'u':
|
|
|
|
|
mb |= 0x00;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
as_fatal (_("BAD MUL"));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mb |= Dj << 3;
|
|
|
|
|
mb |= size - 1;
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, mb, 1);
|
|
|
|
|
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
mul_reg_opr_opr (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
int Dd;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer1[4];
|
|
|
|
|
int n_bytes1;
|
|
|
|
|
expressionS exp1;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer1, &n_bytes1, &exp1, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer2[4];
|
|
|
|
|
int n_bytes2;
|
|
|
|
|
expressionS exp2;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer2, &n_bytes2, &exp2, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int size1 = size_from_suffix (insn, 0);
|
|
|
|
|
int size2 = size_from_suffix (insn, 1);
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (insn->page + 1 + n_bytes1 + n_bytes2);
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc + Dd, 1);
|
|
|
|
|
uint8_t mb = 0x42;
|
|
|
|
|
const char *dot = strchrnul (insn->name, '.');
|
|
|
|
|
switch (dot[-1])
|
|
|
|
|
{
|
|
|
|
|
case 's':
|
|
|
|
|
mb |= 0x80;
|
|
|
|
|
break;
|
|
|
|
|
case 'u':
|
|
|
|
|
mb |= 0x00;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
as_fatal (_("BAD MUL"));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mb |= (size1 - 1) << 4;
|
|
|
|
|
mb |= (size2 - 1) << 2;
|
|
|
|
|
number_to_chars_bigendian (f++, mb, 1);
|
|
|
|
|
|
|
|
|
|
f = emit_opr (f, buffer1, n_bytes1, &exp1);
|
|
|
|
|
f = emit_opr (f, buffer2, n_bytes2, &exp2);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define REG_BIT_GRP0 \
|
|
|
|
|
((0x1U << REG_D2) | \
|
|
|
|
|
(0x1U << REG_D3) | \
|
|
|
|
|
(0x1U << REG_CCH) | \
|
|
|
|
|
(0x1U << REG_CCL) | \
|
|
|
|
|
(0x1U << REG_D0) | \
|
|
|
|
|
(0x1U << REG_D1))
|
|
|
|
|
|
|
|
|
|
#define REG_BIT_GRP1 \
|
|
|
|
|
((0x1U << REG_D4) | \
|
|
|
|
|
(0x1U << REG_D5) | \
|
|
|
|
|
(0x1U << REG_D6) | \
|
|
|
|
|
(0x1U << REG_D7) | \
|
|
|
|
|
(0x1U << REG_X) | \
|
|
|
|
|
(0x1U << REG_Y))
|
|
|
|
|
|
|
|
|
|
static const uint8_t reg_map [] =
|
|
|
|
|
{
|
2020-06-29 08:37:56 +08:00
|
|
|
|
0x02, /* D2 */
|
|
|
|
|
0x01, /* D3 */
|
2018-05-18 22:26:18 +08:00
|
|
|
|
0x20,
|
2020-06-29 08:37:56 +08:00
|
|
|
|
0x10, /* D5 */
|
|
|
|
|
0x08, /* D0 */
|
|
|
|
|
0x04, /* D1 */
|
|
|
|
|
0x08, /* D6 */
|
|
|
|
|
0x04, /* D7 */
|
2018-05-18 22:26:18 +08:00
|
|
|
|
0x02,
|
2020-06-29 08:37:56 +08:00
|
|
|
|
0x01, /* Y */
|
2018-05-18 22:26:18 +08:00
|
|
|
|
0x00,
|
|
|
|
|
0x00,
|
2020-06-29 08:37:56 +08:00
|
|
|
|
0x20, /* CCH */
|
|
|
|
|
0x10, /* CCL */
|
2018-05-18 22:26:18 +08:00
|
|
|
|
0x00
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
lex_reg_list (uint16_t grp, uint16_t *reg_bits)
|
|
|
|
|
{
|
|
|
|
|
if (lex_match (','))
|
|
|
|
|
{
|
|
|
|
|
int reg;
|
|
|
|
|
if (!lex_reg_name (grp, ®))
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
*reg_bits |= 0x1u << reg;
|
|
|
|
|
lex_reg_list (grp, reg_bits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Empty list */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
psh_pull (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
uint8_t pb =
|
|
|
|
|
(0 == strcmp ("pul", insn->name)) ? 0x80: 0x00;
|
|
|
|
|
|
|
|
|
|
if (lex_match_string ("all16b"))
|
|
|
|
|
{
|
|
|
|
|
pb |= 0x40;
|
|
|
|
|
}
|
|
|
|
|
else if (lex_match_string ("all"))
|
|
|
|
|
{
|
|
|
|
|
/* Nothing to do */
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
int reg1;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_GRP1 | REG_BIT_GRP0, ®1))
|
|
|
|
|
goto fail;
|
|
|
|
|
uint16_t admitted_group = 0;
|
|
|
|
|
|
|
|
|
|
if ((0x1U << reg1) & REG_BIT_GRP1)
|
|
|
|
|
admitted_group = REG_BIT_GRP1;
|
|
|
|
|
else if ((0x1U << reg1) & REG_BIT_GRP0)
|
|
|
|
|
admitted_group = REG_BIT_GRP0;
|
|
|
|
|
|
|
|
|
|
uint16_t reg_bits = 0x1 << reg1;
|
|
|
|
|
if (!lex_reg_list (admitted_group, ®_bits))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (reg_bits & REG_BIT_GRP1)
|
|
|
|
|
pb |= 0x40;
|
|
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i < 16; ++i)
|
|
|
|
|
{
|
|
|
|
|
if (reg_bits & (0x1u << i))
|
|
|
|
|
pb |= reg_map[i];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (2);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, pb, 1);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
tfr (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
int reg1;
|
|
|
|
|
if (!lex_reg_name (~0, ®1))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int reg2;
|
|
|
|
|
if (!lex_reg_name (~0, ®2))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
2019-01-16 21:34:50 +08:00
|
|
|
|
if ( ((0 == strcasecmp ("sex", insn->name))
|
|
|
|
|
|| (0 == strcasecmp ("zex", insn->name)))
|
|
|
|
|
&& (registers[reg2].bytes <= registers[reg1].bytes))
|
|
|
|
|
as_warn (_("Source register for %s is no larger than the destination register"),
|
|
|
|
|
insn->name);
|
2019-02-02 00:42:54 +08:00
|
|
|
|
else if (reg1 == reg2)
|
|
|
|
|
as_warn (_("The destination and source registers are identical"));
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (1 + insn->page);
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, reg1 << 4 | reg2, 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
imm8 (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
long imm;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (! lex_imm (&imm, NULL))
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
if (imm > 127 || imm < -128)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Immediate value %ld is out of range for instruction %s"),
|
|
|
|
|
imm, insn->name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (2);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, imm, 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
reg_imm (const struct instruction *insn, int allowed_reg)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int reg;
|
|
|
|
|
if (lex_reg_name (allowed_reg, ®))
|
|
|
|
|
{
|
|
|
|
|
if (!lex_force_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
long imm;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (! lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
short size = registers[reg].bytes;
|
|
|
|
|
char *f = s12z_new_insn (insn->page + size);
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc + reg, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, imm, size);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
regd_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return reg_imm (insn, REG_BIT_Dn);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
regdxy_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return reg_imm (insn, REG_BIT_Dn | REG_BIT_XY);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
regs_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return reg_imm (insn, 0x1U << REG_S);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
trap_imm (const struct instruction *insn ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
long imm = -1;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (! lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (imm < 0x92 || imm > 0xFF ||
|
|
|
|
|
(imm >= 0xA0 && imm <= 0xA7) ||
|
|
|
|
|
(imm >= 0xB0 && imm <= 0xB7))
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("trap value %ld is not valid"), imm);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (2);
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, imm & 0xFF, 1);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Special one byte instruction CMP X, Y */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
regx_regy (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
int reg;
|
|
|
|
|
if (lex_reg_name (0x1U << REG_X, ®))
|
|
|
|
|
{
|
|
|
|
|
if (lex_force_match (','))
|
|
|
|
|
{
|
|
|
|
|
if (lex_reg_name (0x1U << REG_Y, ®))
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (1);
|
|
|
|
|
number_to_chars_bigendian (f, insn->opc, 1);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Special one byte instruction SUB D6, X, Y */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
regd6_regx_regy (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int reg;
|
|
|
|
|
if (!lex_reg_name (0x1U << REG_D6, ®))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_reg_name (0x1U << REG_X, ®))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_reg_name (0x1U << REG_Y, ®))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (1);
|
|
|
|
|
number_to_chars_bigendian (f, insn->opc, 1);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Special one byte instruction SUB D6, Y, X */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
regd6_regy_regx (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int reg;
|
|
|
|
|
if (!lex_reg_name (0x1U << REG_D6, ®))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_reg_name (0x1U << REG_Y, ®))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_reg_name (0x1U << REG_X, ®))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (1);
|
|
|
|
|
number_to_chars_bigendian (f, insn->opc, 1);
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2019-02-02 00:42:54 +08:00
|
|
|
|
reg_opr (const struct instruction *insn, int allowed_regs,
|
|
|
|
|
bool immediate_ok)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int reg;
|
|
|
|
|
if (lex_reg_name (allowed_regs, ®))
|
|
|
|
|
{
|
|
|
|
|
if (!lex_force_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (lex_opr (buffer, &n_bytes, &exp, immediate_ok))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
/* Large constant direct values are more efficiently encoded as ext24 mode.
|
|
|
|
|
Otherwise a decision has to be deferred to a relax. */
|
|
|
|
|
if (exp.X_op == O_constant
|
|
|
|
|
&& buffer[0] == 0xFA
|
|
|
|
|
&& insn->alt_opc != 0)
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (4);
|
|
|
|
|
|
|
|
|
|
/* I don't think there are any instances of page 2 opcodes in this case */
|
|
|
|
|
gas_assert (insn->page == 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->alt_opc + reg, 1);
|
|
|
|
|
|
|
|
|
|
emit_ext24 (f, exp.X_add_number);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (n_bytes + insn->page);
|
|
|
|
|
|
|
|
|
|
if (insn->page == 2)
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc + reg, 1);
|
|
|
|
|
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2019-02-02 00:42:54 +08:00
|
|
|
|
regdxy_opr_dest (const struct instruction *insn)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
2019-02-02 00:42:54 +08:00
|
|
|
|
return reg_opr (insn, REG_BIT_Dn | REG_BIT_XY, false);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2019-02-02 00:42:54 +08:00
|
|
|
|
regdxy_opr_src (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return reg_opr (insn, REG_BIT_Dn | REG_BIT_XY, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
regd_opr (const struct instruction *insn)
|
|
|
|
|
{
|
2019-02-02 00:42:54 +08:00
|
|
|
|
return reg_opr (insn, REG_BIT_Dn, true);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-02-02 00:42:54 +08:00
|
|
|
|
/* OP0: S; OP1: destination OPR */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2019-02-02 00:42:54 +08:00
|
|
|
|
regs_opr_dest (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return reg_opr (insn, 0x1U << REG_S, false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* OP0: S; OP1: source OPR */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2019-02-02 00:42:54 +08:00
|
|
|
|
regs_opr_src (const struct instruction *insn)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
2019-02-02 00:42:54 +08:00
|
|
|
|
return reg_opr (insn, 0x1U << REG_S, true);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
imm_opr (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
long imm;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
expressionS exp0;
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
exp0.X_op = O_absent;
|
|
|
|
|
|
|
|
|
|
/* Note: The ternary expression below means that "MOV.x #symbol,
|
|
|
|
|
mem-expr" is accepted when x is a member of {'w', 'p', 'l'} but
|
|
|
|
|
not when it is 'b'.
|
|
|
|
|
The Freescale assembler accepts "MOV.b #symbol, mem-expr" but
|
|
|
|
|
produces obviously incorrect code. Since such an instruction
|
|
|
|
|
would require an 8-bit reloc (which we don't have) and some
|
|
|
|
|
non-optimal kludges in the OPR encoding, it seems sensible that
|
|
|
|
|
such instructions should be rejected. */
|
|
|
|
|
if (!lex_imm (&imm, size > 1 ? &exp0 : NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
expressionS exp1;
|
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp1, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (1 + n_bytes + size);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
|
2019-02-02 00:42:54 +08:00
|
|
|
|
emit_reloc (&exp0, f, size, size == 4 ? BFD_RELOC_32 : BFD_RELOC_S12Z_OPR);
|
|
|
|
|
|
2018-05-18 22:26:18 +08:00
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i < size; ++i)
|
|
|
|
|
number_to_chars_bigendian (f++, imm >> (CHAR_BIT * (size - i - 1)), 1);
|
|
|
|
|
|
2019-02-02 00:42:54 +08:00
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp1);
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
opr_opr (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer1[4];
|
|
|
|
|
int n_bytes1;
|
|
|
|
|
expressionS exp1;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer1, &n_bytes1, &exp1, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer2[4];
|
|
|
|
|
int n_bytes2;
|
|
|
|
|
expressionS exp2;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer2, &n_bytes2, &exp2, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (1 + n_bytes1 + n_bytes2);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
|
|
|
|
|
f = emit_opr (f, buffer1, n_bytes1, &exp1);
|
|
|
|
|
f = emit_opr (f, buffer2, n_bytes2, &exp2);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
reg67sxy_opr (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
int reg;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_XYS | (0x1U << REG_D6) | (0x1U << REG_D7), ®))
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (1 + n_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc + reg - REG_D6, 1);
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
rotate (const struct instruction *insn, short dir)
|
|
|
|
|
{
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (n_bytes + 2);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
if (size < 0)
|
|
|
|
|
size = 1;
|
|
|
|
|
uint8_t sb = 0x24;
|
|
|
|
|
sb |= size - 1;
|
|
|
|
|
if (dir)
|
|
|
|
|
sb |= 0x40;
|
|
|
|
|
number_to_chars_bigendian (f++, sb, 1);
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
rol (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return rotate (insn, 1);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
ror (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return rotate (insn, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Shift instruction with a register operand and an immediate #1 or #2
|
|
|
|
|
left = 1; right = 0;
|
|
|
|
|
logical = 0; arithmetic = 1;
|
|
|
|
|
*/
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_shift_reg_imm1 (const struct instruction *insn, short type, short dir)
|
|
|
|
|
{
|
|
|
|
|
/*
|
|
|
|
|
This function is highly unusual and a bit wierd!
|
|
|
|
|
It first matches the input against a register {d0, d1, ... d7} followed by an immediate
|
|
|
|
|
{#1, #2}.
|
|
|
|
|
Then, it rewinds the input and parses it again as a OPR.
|
|
|
|
|
*/
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
int Dd;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
{
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long imm = -1;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (imm != 1 && imm != 2)
|
|
|
|
|
goto fail;
|
|
|
|
|
input_line_pointer = ilp;
|
|
|
|
|
|
|
|
|
|
/* Now parse the first operand again */
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
gas_assert (n_bytes == 1);
|
|
|
|
|
|
|
|
|
|
uint8_t sb = 0x34;
|
|
|
|
|
sb |= dir << 6;
|
|
|
|
|
sb |= type << 7;
|
|
|
|
|
if (imm == 2)
|
|
|
|
|
sb |= 0x08;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (3);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, sb, 1);
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Shift instruction with a register operand.
|
|
|
|
|
left = 1; right = 0;
|
|
|
|
|
logical = 0; arithmetic = 1; */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
lex_shift_reg (const struct instruction *insn, short type, short dir)
|
|
|
|
|
{
|
|
|
|
|
int Dd, Ds, Dn;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
{
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Ds))
|
|
|
|
|
{
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t sb = 0x10;
|
|
|
|
|
sb |= Ds;
|
|
|
|
|
sb |= dir << 6;
|
|
|
|
|
sb |= type << 7;
|
|
|
|
|
long imm;
|
|
|
|
|
if (lex_reg_name (REG_BIT_Dn, &Dn))
|
|
|
|
|
{
|
|
|
|
|
char *f = s12z_new_insn (3);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc | Dd, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, sb, 1);
|
|
|
|
|
uint8_t xb = 0xb8;
|
|
|
|
|
xb |= Dn;
|
|
|
|
|
number_to_chars_bigendian (f++, xb, 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
2019-02-02 00:42:54 +08:00
|
|
|
|
else if (lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
if (imm < 0 || imm > 31)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Shift value should be in the range [0,31]"));
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int n_bytes = 3;
|
|
|
|
|
if (imm == 1 || imm == 2)
|
|
|
|
|
{
|
|
|
|
|
n_bytes = 2;
|
|
|
|
|
sb &= ~0x10;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
sb |= (imm & 0x01) << 3;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (n_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc | Dd, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, sb, 1);
|
|
|
|
|
if (n_bytes > 2)
|
|
|
|
|
{
|
|
|
|
|
uint8_t xb = 0x70;
|
|
|
|
|
xb |= imm >> 1;
|
|
|
|
|
number_to_chars_bigendian (f++, xb, 1);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
impute_shift_dir_and_type (const struct instruction *insn, short *type, short *dir)
|
|
|
|
|
{
|
|
|
|
|
*dir = -1;
|
|
|
|
|
*type = -1;
|
|
|
|
|
switch (insn->name[0])
|
|
|
|
|
{
|
|
|
|
|
case 'l':
|
|
|
|
|
*type = 0;
|
|
|
|
|
break;
|
|
|
|
|
case 'a':
|
|
|
|
|
*type = 1;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
as_fatal (_("Bad shift mode"));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (insn->name[2])
|
|
|
|
|
{
|
|
|
|
|
case 'l':
|
|
|
|
|
*dir = 1;
|
|
|
|
|
break;
|
|
|
|
|
case 'r':
|
|
|
|
|
*dir = 0;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
as_fatal (_("Bad shift *direction"));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Shift instruction with a OPR operand */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
shift_two_operand (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
uint8_t sb = 0x34;
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
short dir = -1;
|
|
|
|
|
short type = -1;
|
|
|
|
|
impute_shift_dir_and_type (insn, &type, &dir);
|
|
|
|
|
sb |= dir << 6;
|
|
|
|
|
sb |= type << 7;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
sb |= size - 1;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_opr_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_opr_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long imm = -1;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (imm != 1 && imm != 2)
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (imm == 2)
|
|
|
|
|
sb |= 0x08;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (2 + n_opr_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, sb, 1);
|
|
|
|
|
emit_opr (f, buffer, n_opr_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Shift instruction with a OPR operand */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
shift_opr_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
short dir = -1;
|
|
|
|
|
short type = -1;
|
|
|
|
|
impute_shift_dir_and_type (insn, &type, &dir);
|
|
|
|
|
|
|
|
|
|
int Dd = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int n_bytes = 2;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer1[4];
|
|
|
|
|
int n_opr_bytes1;
|
|
|
|
|
|
|
|
|
|
expressionS exp1;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer1, &n_opr_bytes1, &exp1, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
n_bytes += n_opr_bytes1;
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer2[4];
|
|
|
|
|
int n_opr_bytes2 = 0;
|
|
|
|
|
expressionS exp2;
|
|
|
|
|
long imm;
|
|
|
|
|
bool immediate = false;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{
|
|
|
|
|
immediate = true;
|
|
|
|
|
}
|
2019-02-02 00:42:54 +08:00
|
|
|
|
else if (!lex_opr (buffer2, &n_opr_bytes2, &exp2, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t sb = 0x20;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
|
|
|
|
|
if (size != -1)
|
|
|
|
|
sb |= size - 1;
|
|
|
|
|
|
|
|
|
|
sb |= dir << 6;
|
|
|
|
|
sb |= type << 7;
|
|
|
|
|
|
|
|
|
|
if (immediate)
|
|
|
|
|
{
|
|
|
|
|
if (imm == 2 || imm == 1)
|
|
|
|
|
{
|
|
|
|
|
if (imm == 2)
|
|
|
|
|
sb |= 0x08;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
n_bytes++;
|
|
|
|
|
sb |= 0x10;
|
|
|
|
|
if (imm % 2)
|
|
|
|
|
sb |= 0x08;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
n_bytes += n_opr_bytes2;
|
|
|
|
|
sb |= 0x10;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (n_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc | Dd, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, sb, 1);
|
|
|
|
|
f = emit_opr (f, buffer1, n_opr_bytes1, &exp1);
|
|
|
|
|
if (immediate)
|
|
|
|
|
{
|
|
|
|
|
if (imm != 1 && imm != 2)
|
|
|
|
|
{
|
|
|
|
|
number_to_chars_bigendian (f++, 0x70 | (imm >> 1), 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
f = emit_opr (f, buffer2, n_opr_bytes2, &exp2);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Shift instruction with a register operand */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
shift_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
short dir = -1;
|
|
|
|
|
short type = -1;
|
|
|
|
|
impute_shift_dir_and_type (insn, &type, &dir);
|
|
|
|
|
|
|
|
|
|
if (lex_shift_reg_imm1 (insn, type, dir))
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
return lex_shift_reg (insn, type, dir);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bm_regd_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int Di = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Di))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long imm;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t bm = imm << 3;
|
|
|
|
|
bm |= Di;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (2);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bm, 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bm_opr_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_opr_bytes;
|
|
|
|
|
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_opr_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dn = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dn))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t bm = Dn << 4;
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
bm |= (size - 1) << 2;
|
|
|
|
|
bm |= 0x81;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (2 + n_opr_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bm, 1);
|
|
|
|
|
|
|
|
|
|
emit_opr (f, buffer, n_opr_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bm_opr_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_opr_bytes;
|
|
|
|
|
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_opr_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
long imm;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
|
|
|
|
|
if (imm < 0 || imm >= size * 8)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Immediate operand %ld is inappropriate for size of instruction"), imm);
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t bm = 0x80;
|
|
|
|
|
if (size == 2)
|
|
|
|
|
bm |= 0x02;
|
|
|
|
|
else if (size == 4)
|
|
|
|
|
bm |= 0x08;
|
|
|
|
|
bm |= (imm & 0x07) << 4;
|
|
|
|
|
bm |= (imm >> 3);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (2 + n_opr_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bm, 1);
|
|
|
|
|
emit_opr (f, buffer, n_opr_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bm_regd_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int Di = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Di))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dn = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dn))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t bm = Dn << 4;
|
|
|
|
|
bm |= 0x81;
|
|
|
|
|
|
|
|
|
|
uint8_t xb = Di | 0xb8;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (3);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bm, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, xb, 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bf_reg_opr_imm (const struct instruction *insn, short ie)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int Dd = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long width;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&width, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (width < 0 || width > 31)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid width value for %s"), insn->name);
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!lex_match (':'))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long offset;
|
|
|
|
|
if (!lex_constant (&offset))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (offset < 0 || offset > 31)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid offset value for %s"), insn->name);
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t i1 = width << 5;
|
|
|
|
|
i1 |= offset;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
uint8_t bb = ie ? 0x80 : 0x00;
|
|
|
|
|
bb |= 0x60;
|
|
|
|
|
bb |= (size - 1) << 2;
|
|
|
|
|
bb |= width >> 3;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (4 + n_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, 0x08 | Dd, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bb, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, i1, 1);
|
|
|
|
|
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bf_opr_reg_imm (const struct instruction *insn, short ie)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Ds = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Ds))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long width;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&width, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (width < 0 || width > 31)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid width value for %s"), insn->name);
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!lex_match (':'))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long offset;
|
|
|
|
|
if (!lex_constant (&offset))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (offset < 0 || offset > 31)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid offset value for %s"), insn->name);
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t i1 = width << 5;
|
|
|
|
|
i1 |= offset;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
uint8_t bb = ie ? 0x80 : 0x00;
|
|
|
|
|
bb |= 0x70;
|
|
|
|
|
bb |= (size - 1) << 2;
|
|
|
|
|
bb |= width >> 3;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (4 + n_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, 0x08 | Ds, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bb, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, i1, 1);
|
|
|
|
|
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bf_reg_reg_imm (const struct instruction *insn, short ie)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int Dd = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Ds = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Ds))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long width;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&width, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (width < 0 || width > 31)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid width value for %s"), insn->name);
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!lex_match (':'))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long offset;
|
|
|
|
|
if (!lex_constant (&offset))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (offset < 0 || offset > 31)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("Invalid offset value for %s"), insn->name);
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t bb = ie ? 0x80 : 0x00;
|
|
|
|
|
bb |= 0x20;
|
|
|
|
|
bb |= Ds << 2;
|
|
|
|
|
bb |= width >> 3;
|
|
|
|
|
|
|
|
|
|
uint8_t i1 = width << 5;
|
|
|
|
|
i1 |= offset;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (4);
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, 0x08 | Dd, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bb, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, i1, 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bf_reg_reg_reg (const struct instruction *insn ATTRIBUTE_UNUSED, short ie)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int Dd = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Ds = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Ds))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dp = 0;
|
|
|
|
|
if (!lex_reg_name ((0x01u << REG_D2) |
|
|
|
|
|
(0x01u << REG_D3) |
|
|
|
|
|
(0x01u << REG_D4) |
|
|
|
|
|
(0x01u << REG_D5),
|
|
|
|
|
&Dp))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t bb = ie ? 0x80 : 0x00;
|
|
|
|
|
bb |= Ds << 2;
|
|
|
|
|
bb |= Dp;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (3);
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, 0x08 | Dd, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bb , 1);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bf_opr_reg_reg (const struct instruction *insn, short ie)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int Ds = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Ds))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int Dp = 0;
|
|
|
|
|
if (!lex_reg_name ((0x01u << REG_D2) |
|
|
|
|
|
(0x01u << REG_D3) |
|
|
|
|
|
(0x01u << REG_D4) |
|
|
|
|
|
(0x01u << REG_D5),
|
|
|
|
|
&Dp))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
uint8_t bb = ie ? 0x80 : 0x00;
|
|
|
|
|
bb |= 0x50;
|
|
|
|
|
bb |= Dp;
|
|
|
|
|
bb |= (size - 1) << 2;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (3 + n_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, 0x08 | Ds, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bb , 1);
|
|
|
|
|
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bf_reg_opr_reg (const struct instruction *insn, short ie)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
int Dd = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dd))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dp = 0;
|
|
|
|
|
if (!lex_reg_name ((0x01u << REG_D2) |
|
|
|
|
|
(0x01u << REG_D3) |
|
|
|
|
|
(0x01u << REG_D4) |
|
|
|
|
|
(0x01u << REG_D5),
|
|
|
|
|
&Dp))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
uint8_t bb = ie ? 0x80 : 0x00;
|
|
|
|
|
bb |= 0x40;
|
|
|
|
|
bb |= Dp;
|
|
|
|
|
bb |= (size - 1) << 2;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (3 + n_bytes);
|
|
|
|
|
number_to_chars_bigendian (f++, PAGE2_PREBYTE, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, 0x08 | Dd, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bb , 1);
|
|
|
|
|
|
|
|
|
|
emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfe_reg_reg_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_reg_reg_reg (insn, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfi_reg_reg_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_reg_reg_reg (insn, 1);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfe_reg_reg_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_reg_reg_imm (insn, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfi_reg_reg_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_reg_reg_imm (insn, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfe_reg_opr_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_reg_opr_reg (insn, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfi_reg_opr_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_reg_opr_reg (insn, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfe_opr_reg_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_opr_reg_reg (insn, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfi_opr_reg_reg (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_opr_reg_reg (insn, 1);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfe_reg_opr_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_reg_opr_imm (insn, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfi_reg_opr_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_reg_opr_imm (insn, 1);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfe_opr_reg_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_opr_reg_imm (insn, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
bfi_opr_reg_imm (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
return bf_opr_reg_imm (insn, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
tb_reg_rel (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
int reg;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn | REG_BIT_XY, ®))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
bool long_displacement;
|
|
|
|
|
expressionS exp;
|
|
|
|
|
if (! lex_15_bit_offset (&long_displacement, &exp))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t lb = 0x00;
|
|
|
|
|
if (reg == REG_X || reg == REG_Y)
|
|
|
|
|
{
|
|
|
|
|
lb |= 0x08;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
lb |= reg;
|
|
|
|
|
}
|
|
|
|
|
if (reg == REG_Y)
|
|
|
|
|
lb |= 0x01;
|
|
|
|
|
|
|
|
|
|
if (0 == strncmp (insn->name + 2, "ne", 2))
|
|
|
|
|
lb |= 0x00 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "eq", 2))
|
|
|
|
|
lb |= 0x01 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "pl", 2))
|
|
|
|
|
lb |= 0x02 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "mi", 2))
|
|
|
|
|
lb |= 0x03 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "gt", 2))
|
|
|
|
|
lb |= 0x04 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "le", 2))
|
|
|
|
|
lb |= 0x05 << 4;
|
|
|
|
|
|
|
|
|
|
switch (insn->name[0])
|
|
|
|
|
{
|
|
|
|
|
case 'd':
|
|
|
|
|
lb |= 0x80;
|
|
|
|
|
break;
|
|
|
|
|
case 't':
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
gas_assert (0);
|
|
|
|
|
break;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (long_displacement ? 4 : 3);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, lb, 1);
|
|
|
|
|
|
|
|
|
|
emit_15_bit_offset (f, 4, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
tb_opr_rel (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
bool long_displacement;
|
|
|
|
|
expressionS exp2;
|
|
|
|
|
if (! lex_15_bit_offset (&long_displacement, &exp2))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t lb = 0x0C;
|
|
|
|
|
|
|
|
|
|
if (0 == strncmp (insn->name + 2, "ne", 2))
|
|
|
|
|
lb |= 0x00 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "eq", 2))
|
|
|
|
|
lb |= 0x01 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "pl", 2))
|
|
|
|
|
lb |= 0x02 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "mi", 2))
|
|
|
|
|
lb |= 0x03 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "gt", 2))
|
|
|
|
|
lb |= 0x04 << 4;
|
|
|
|
|
else if (0 == strncmp (insn->name + 2, "le", 2))
|
|
|
|
|
lb |= 0x05 << 4;
|
|
|
|
|
|
|
|
|
|
switch (insn->name[0])
|
|
|
|
|
{
|
|
|
|
|
case 'd':
|
|
|
|
|
lb |= 0x80;
|
|
|
|
|
break;
|
|
|
|
|
case 't':
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
gas_assert (0);
|
|
|
|
|
break;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
|
|
|
|
|
lb |= size -1;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (n_bytes + (long_displacement ? 4 : 3));
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, lb, 1);
|
|
|
|
|
f = emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
|
|
|
|
emit_15_bit_offset (f, n_bytes + 4, &exp2);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
test_br_reg_reg_rel (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
int Di = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Di))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int Dn = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dn))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool long_displacement;
|
|
|
|
|
expressionS exp;
|
|
|
|
|
if (! lex_15_bit_offset (&long_displacement, &exp))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t bm = 0x81;
|
|
|
|
|
uint8_t xb = 0xb8;
|
|
|
|
|
|
|
|
|
|
bm |= Dn << 4;
|
|
|
|
|
xb |= Di;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (long_displacement ? 5 : 4);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bm, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, xb, 1);
|
|
|
|
|
|
|
|
|
|
emit_15_bit_offset (f, 5, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
test_br_opr_reg_rel (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int Dn = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Dn))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t bm = 0x81;
|
|
|
|
|
bm |= Dn << 4;
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
bm |= (size -1) << 2;
|
|
|
|
|
|
|
|
|
|
bool long_displacement;
|
|
|
|
|
|
|
|
|
|
expressionS exp2;
|
|
|
|
|
if (! lex_15_bit_offset (&long_displacement, &exp2))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int n = n_bytes + (long_displacement ? 4 : 3);
|
|
|
|
|
char *f = s12z_new_insn (n);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bm, 1);
|
|
|
|
|
f = emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
|
|
|
|
emit_15_bit_offset (f, n, &exp2);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
test_br_opr_imm_rel (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
uint8_t buffer[4];
|
|
|
|
|
int n_bytes;
|
|
|
|
|
expressionS exp;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_opr (buffer, &n_bytes, &exp, false))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long imm;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (imm < 0 || imm > 31)
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
bool long_displacement;
|
|
|
|
|
expressionS exp2;
|
|
|
|
|
if (! lex_15_bit_offset (&long_displacement, &exp2))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
int size = size_from_suffix (insn, 0);
|
|
|
|
|
|
|
|
|
|
uint8_t bm = 0x80;
|
|
|
|
|
bm |= (imm & 0x07) << 4;
|
|
|
|
|
bm |= (imm >> 3) & 0x03;
|
|
|
|
|
if (size == 4)
|
|
|
|
|
bm |= 0x08;
|
|
|
|
|
else if (size == 2)
|
|
|
|
|
bm |= 0x02;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (n_bytes + (long_displacement ? 4 : 3));
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bm, 1);
|
|
|
|
|
f = emit_opr (f, buffer, n_bytes, &exp);
|
|
|
|
|
|
|
|
|
|
emit_15_bit_offset (f, n_bytes + 4, &exp2);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
static bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
test_br_reg_imm_rel (const struct instruction *insn)
|
|
|
|
|
{
|
|
|
|
|
char *ilp = input_line_pointer;
|
|
|
|
|
|
|
|
|
|
int Di = 0;
|
|
|
|
|
if (!lex_reg_name (REG_BIT_Dn, &Di))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
long imm;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (!lex_imm (&imm, NULL))
|
2018-05-18 22:26:18 +08:00
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (imm < 0 || imm > 31)
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!lex_match (','))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
bool long_displacement;
|
|
|
|
|
expressionS exp;
|
|
|
|
|
if (! lex_15_bit_offset (&long_displacement, &exp))
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
uint8_t bm = Di;
|
|
|
|
|
bm |= imm << 3;
|
|
|
|
|
|
|
|
|
|
char *f = s12z_new_insn (long_displacement ? 4 : 3);
|
|
|
|
|
number_to_chars_bigendian (f++, insn->opc, 1);
|
|
|
|
|
number_to_chars_bigendian (f++, bm, 1);
|
|
|
|
|
|
|
|
|
|
emit_15_bit_offset (f, 4, &exp);
|
|
|
|
|
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
fail_line_pointer = input_line_pointer;
|
|
|
|
|
input_line_pointer = ilp;
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static const struct instruction opcodes[] = {
|
|
|
|
|
{"bgnd", 1, 0x00, no_operands, 0},
|
|
|
|
|
{"nop", 1, 0x01, no_operands, 0},
|
|
|
|
|
|
|
|
|
|
{"brclr", 1, 0x02, test_br_reg_reg_rel, 0},
|
|
|
|
|
{"brset", 1, 0x03, test_br_reg_reg_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"brclr", 1, 0x02, test_br_reg_imm_rel, 0},
|
|
|
|
|
{"brset", 1, 0x03, test_br_reg_imm_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"brclr.b", 1, 0x02, test_br_opr_reg_rel, 0},
|
|
|
|
|
{"brclr.w", 1, 0x02, test_br_opr_reg_rel, 0},
|
|
|
|
|
{"brclr.l", 1, 0x02, test_br_opr_reg_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"brset.b", 1, 0x03, test_br_opr_reg_rel, 0},
|
|
|
|
|
{"brset.w", 1, 0x03, test_br_opr_reg_rel, 0},
|
|
|
|
|
{"brset.l", 1, 0x03, test_br_opr_reg_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"brclr.b", 1, 0x02, test_br_opr_imm_rel, 0},
|
|
|
|
|
{"brclr.w", 1, 0x02, test_br_opr_imm_rel, 0},
|
|
|
|
|
{"brclr.l", 1, 0x02, test_br_opr_imm_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"brset.b", 1, 0x03, test_br_opr_imm_rel, 0},
|
|
|
|
|
{"brset.w", 1, 0x03, test_br_opr_imm_rel, 0},
|
|
|
|
|
{"brset.l", 1, 0x03, test_br_opr_imm_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"psh", 1, 0x04, psh_pull, 0},
|
|
|
|
|
{"pul", 1, 0x04, psh_pull, 0},
|
|
|
|
|
|
|
|
|
|
{"rts", 1, 0x05, no_operands, 0},
|
|
|
|
|
{"lea", 1, 0x06, reg67sxy_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"dbne", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"dbeq", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"dbpl", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"dbmi", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"dbgt", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"dble", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"dbne.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbeq.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbpl.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbmi.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbgt.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dble.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"dbne.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbeq.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbpl.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbmi.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbgt.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dble.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"dbne.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbeq.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbpl.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbmi.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbgt.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dble.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"dbne.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbeq.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbpl.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbmi.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dbgt.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"dble.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"tbne", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"tbeq", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"tbpl", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"tbmi", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"tbgt", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
{"tble", 1, 0x0b, tb_reg_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"tbne.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbeq.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbpl.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbmi.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbgt.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tble.b", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"tbne.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbeq.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbpl.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbmi.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbgt.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tble.w", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"tbne.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbeq.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbpl.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbmi.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbgt.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tble.p", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"tbne.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbeq.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbpl.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbmi.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tbgt.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
{"tble.l", 1, 0x0b, tb_opr_rel, 0},
|
|
|
|
|
|
|
|
|
|
{"mov.b", 1, 0x0c, imm_opr, 0},
|
|
|
|
|
{"mov.w", 1, 0x0d, imm_opr, 0},
|
|
|
|
|
{"mov.p", 1, 0x0e, imm_opr, 0},
|
|
|
|
|
{"mov.l", 1, 0x0f, imm_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"rol", 1, 0x10, rol, 0},
|
|
|
|
|
{"rol.b", 1, 0x10, rol, 0},
|
|
|
|
|
{"rol.w", 1, 0x10, rol, 0},
|
|
|
|
|
{"rol.p", 1, 0x10, rol, 0},
|
|
|
|
|
{"rol.l", 1, 0x10, rol, 0},
|
|
|
|
|
|
|
|
|
|
{"ror", 1, 0x10, ror, 0},
|
|
|
|
|
{"ror.b", 1, 0x10, ror, 0},
|
|
|
|
|
{"ror.w", 1, 0x10, ror, 0},
|
|
|
|
|
{"ror.p", 1, 0x10, ror, 0},
|
|
|
|
|
{"ror.l", 1, 0x10, ror, 0},
|
|
|
|
|
|
|
|
|
|
{"lsl", 1, 0x10, shift_reg, 0},
|
|
|
|
|
{"lsr", 1, 0x10, shift_reg, 0},
|
|
|
|
|
{"asl", 1, 0x10, shift_reg, 0},
|
|
|
|
|
{"asr", 1, 0x10, shift_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"lsl.b", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"lsl.w", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"lsl.p", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"lsl.l", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"asl.b", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"asl.w", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"asl.p", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"asl.l", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
|
|
|
|
|
{"lsr.b", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"lsr.w", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"lsr.p", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"lsr.l", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"asr.b", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"asr.w", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"asr.p", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
{"asr.l", 1, 0x10, shift_two_operand, 0},
|
|
|
|
|
|
|
|
|
|
{"lsl.b", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"lsl.w", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"lsl.p", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"lsl.l", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"asl.b", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"asl.w", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"asl.p", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"asl.l", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"lsr.b", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"lsr.w", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"lsr.p", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"lsr.l", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"asr.b", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"asr.w", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"asr.p", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
{"asr.l", 1, 0x10, shift_opr_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"mov.b", 1, 0x1c, opr_opr, 0},
|
|
|
|
|
{"mov.w", 1, 0x1d, opr_opr, 0},
|
|
|
|
|
{"mov.p", 1, 0x1e, opr_opr, 0},
|
|
|
|
|
{"mov.l", 1, 0x1f, opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"bra", 1, 0x20, rel, 0},
|
|
|
|
|
{"bsr", 1, 0x21, rel, 0},
|
|
|
|
|
{"bhi", 1, 0x22, rel, 0},
|
|
|
|
|
{"bls", 1, 0x23, rel, 0},
|
|
|
|
|
{"bcc", 1, 0x24, rel, 0},
|
2018-11-21 01:50:30 +08:00
|
|
|
|
{"bhs", 1, 0x24, rel, 0}, /* Alias for bcc */
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{"bcs", 1, 0x25, rel, 0},
|
2018-11-21 01:50:30 +08:00
|
|
|
|
{"blo", 1, 0x25, rel, 0}, /* Alias for bcs */
|
2018-05-18 22:26:18 +08:00
|
|
|
|
{"bne", 1, 0x26, rel, 0},
|
|
|
|
|
{"beq", 1, 0x27, rel, 0},
|
|
|
|
|
{"bvc", 1, 0x28, rel, 0},
|
|
|
|
|
{"bvs", 1, 0x29, rel, 0},
|
|
|
|
|
{"bpl", 1, 0x2a, rel, 0},
|
|
|
|
|
{"bmi", 1, 0x2b, rel, 0},
|
|
|
|
|
{"bge", 1, 0x2c, rel, 0},
|
|
|
|
|
{"blt", 1, 0x2d, rel, 0},
|
|
|
|
|
{"bgt", 1, 0x2e, rel, 0},
|
|
|
|
|
{"ble", 1, 0x2f, rel, 0},
|
|
|
|
|
|
|
|
|
|
{"inc", 1, 0x30, reg_inh, 0},
|
|
|
|
|
{"clr", 1, 0x38, reg_inh, 0},
|
|
|
|
|
{"dec", 1, 0x40, reg_inh, 0},
|
|
|
|
|
|
|
|
|
|
{"muls", 1, 0x48, mul_reg_reg_reg, 0},
|
|
|
|
|
{"mulu", 1, 0x48, mul_reg_reg_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"muls.b", 1, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
{"muls.w", 1, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
{"muls.l", 1, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mulu.b", 1, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
{"mulu.w", 1, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
{"mulu.l", 1, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"muls.b", 1, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
{"muls.w", 1, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
{"muls.l", 1, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"mulu.b", 1, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
{"mulu.w", 1, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
{"mulu.l", 1, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"muls.bb", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.bw", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.bp", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.bl", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"muls.wb", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.ww", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.wp", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.wl", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"muls.pb", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.pw", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.pp", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.pl", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"muls.lb", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.lw", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.lp", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"muls.ll", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mulu.bb", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.bw", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.bp", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.bl", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mulu.wb", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.ww", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.wp", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.wl", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mulu.pb", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.pw", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.pp", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.pl", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mulu.lb", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.lw", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.lp", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mulu.ll", 1, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"add", 1, 0x50, regd_imm, 0},
|
|
|
|
|
{"and", 1, 0x58, regd_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"add", 1, 0x60, regd_opr, 0},
|
|
|
|
|
{"and", 1, 0x68, regd_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"sub", 1, 0x70, regd_imm, 0},
|
|
|
|
|
{"or", 1, 0x78, regd_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"sub", 1, 0x80, regd_opr, 0},
|
|
|
|
|
{"or", 1, 0x88, regd_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"ld", 1, 0x90, regdxy_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"clr", 1, 0x9a, clr_xy, 0},
|
|
|
|
|
{"tfr", 1, 0x9e, tfr, 0},
|
|
|
|
|
{"zex", 1, 0x9e, tfr, 0},
|
|
|
|
|
|
2019-02-02 00:42:54 +08:00
|
|
|
|
{"ld", 1, 0xa0, regdxy_opr_src, 0xb0},
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
{"jmp", 1, 0xaa, opr, 0xba},
|
|
|
|
|
{"jsr", 1, 0xab, opr, 0xbb},
|
|
|
|
|
|
|
|
|
|
{"exg", 1, 0xae, tfr, 0},
|
|
|
|
|
{"sex", 1, 0xae, tfr, 0},
|
|
|
|
|
|
2019-02-02 00:42:54 +08:00
|
|
|
|
{"st", 1, 0xc0, regdxy_opr_dest, 0xd0},
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
{"andcc", 1, 0xce, imm8, 0},
|
|
|
|
|
{"orcc", 1, 0xde, imm8, 0},
|
|
|
|
|
|
|
|
|
|
{"inc.b", 1, 0x9c, opr, 0},
|
|
|
|
|
{"inc.w", 1, 0x9d, opr, 0},
|
|
|
|
|
{"inc.l", 1, 0x9f, opr, 0},
|
|
|
|
|
|
|
|
|
|
{"dec.b", 1, 0xac, opr, 0},
|
|
|
|
|
{"dec.w", 1, 0xad, opr, 0},
|
|
|
|
|
{"dec.l", 1, 0xaf, opr, 0},
|
|
|
|
|
|
|
|
|
|
{"clr.b", 1, 0xbc, opr, 0},
|
|
|
|
|
{"clr.w", 1, 0xbd, opr, 0},
|
|
|
|
|
{"clr.p", 1, 0xbe, opr, 0},
|
|
|
|
|
{"clr.l", 1, 0xbf, opr, 0},
|
|
|
|
|
|
|
|
|
|
{"com.b", 1, 0xcc, opr, 0},
|
|
|
|
|
{"com.w", 1, 0xcd, opr, 0},
|
|
|
|
|
{"com.l", 1, 0xcf, opr, 0},
|
|
|
|
|
|
|
|
|
|
{"neg.b", 1, 0xdc, opr, 0},
|
|
|
|
|
{"neg.w", 1, 0xdd, opr, 0},
|
|
|
|
|
{"neg.l", 1, 0xdf, opr, 0},
|
|
|
|
|
|
|
|
|
|
{"bclr", 1, 0xec, bm_regd_imm, 0},
|
|
|
|
|
{"bset", 1, 0xed, bm_regd_imm, 0},
|
|
|
|
|
{"btgl", 1, 0xee, bm_regd_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"bclr", 1, 0xec, bm_regd_reg, 0},
|
|
|
|
|
{"bset", 1, 0xed, bm_regd_reg, 0},
|
|
|
|
|
{"btgl", 1, 0xee, bm_regd_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"bclr.b", 1, 0xec, bm_opr_imm, 0},
|
|
|
|
|
{"bclr.w", 1, 0xec, bm_opr_imm, 0},
|
|
|
|
|
{"bclr.l", 1, 0xec, bm_opr_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"bset.b", 1, 0xed, bm_opr_imm, 0},
|
|
|
|
|
{"bset.w", 1, 0xed, bm_opr_imm, 0},
|
|
|
|
|
{"bset.l", 1, 0xed, bm_opr_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"btgl.b", 1, 0xee, bm_opr_imm, 0},
|
|
|
|
|
{"btgl.w", 1, 0xee, bm_opr_imm, 0},
|
|
|
|
|
{"btgl.l", 1, 0xee, bm_opr_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"bclr.b", 1, 0xec, bm_opr_reg, 0},
|
|
|
|
|
{"bclr.w", 1, 0xec, bm_opr_reg, 0},
|
|
|
|
|
{"bclr.l", 1, 0xec, bm_opr_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"bset.b", 1, 0xed, bm_opr_reg, 0},
|
|
|
|
|
{"bset.w", 1, 0xed, bm_opr_reg, 0},
|
|
|
|
|
{"bset.l", 1, 0xed, bm_opr_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"btgl.b", 1, 0xee, bm_opr_reg, 0},
|
|
|
|
|
{"btgl.w", 1, 0xee, bm_opr_reg, 0},
|
|
|
|
|
{"btgl.l", 1, 0xee, bm_opr_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"cmp", 1, 0xe0, regdxy_imm, 0},
|
2019-02-02 00:42:54 +08:00
|
|
|
|
{"cmp", 1, 0xf0, regdxy_opr_src, 0},
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
{"cmp", 1, 0xfc, regx_regy, 0},
|
|
|
|
|
{"sub", 1, 0xfd, regd6_regx_regy, 0},
|
|
|
|
|
{"sub", 1, 0xfe, regd6_regy_regx, 0},
|
|
|
|
|
|
|
|
|
|
{"swi", 1, 0xff, no_operands, 0},
|
|
|
|
|
|
|
|
|
|
/* Page 2 */
|
|
|
|
|
|
|
|
|
|
/* The -10 below is a kludge. The opcode is in fact 0x00 */
|
2019-02-02 00:42:54 +08:00
|
|
|
|
{"ld", 2, -10, regs_opr_src, 0},
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
/* The -9 below is a kludge. The opcode is in fact 0x01 */
|
2019-02-02 00:42:54 +08:00
|
|
|
|
{"st", 2, -9, regs_opr_dest, 0},
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
/* The -8 below is a kludge. The opcode is in fact 0x02 */
|
2019-02-02 00:42:54 +08:00
|
|
|
|
{"cmp", 2, -8, regs_opr_src, 0},
|
2018-05-18 22:26:18 +08:00
|
|
|
|
|
|
|
|
|
/* The -7 below is a kludge. The opcode is in fact 0x03 */
|
|
|
|
|
{"ld", 2, -7, regs_imm, 0},
|
|
|
|
|
|
|
|
|
|
/* The -6 below is a kludge. The opcode is in fact 0x04 */
|
|
|
|
|
{"cmp", 2, -6, regs_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"bfext", 2, 0x08, bfe_reg_reg_reg, 0},
|
|
|
|
|
{"bfext", 2, 0x08, bfe_reg_reg_imm, 0},
|
|
|
|
|
{"bfext.b", 2, 0x08, bfe_reg_opr_reg, 0},
|
|
|
|
|
{"bfext.w", 2, 0x08, bfe_reg_opr_reg, 0},
|
|
|
|
|
{"bfext.p", 2, 0x08, bfe_reg_opr_reg, 0},
|
|
|
|
|
{"bfext.l", 2, 0x08, bfe_reg_opr_reg, 0},
|
|
|
|
|
{"bfext.b", 2, 0x08, bfe_opr_reg_reg, 0},
|
|
|
|
|
{"bfext.w", 2, 0x08, bfe_opr_reg_reg, 0},
|
|
|
|
|
{"bfext.p", 2, 0x08, bfe_opr_reg_reg, 0},
|
|
|
|
|
{"bfext.l", 2, 0x08, bfe_opr_reg_reg, 0},
|
|
|
|
|
{"bfext.b", 2, 0x08, bfe_reg_opr_imm, 0},
|
|
|
|
|
{"bfext.w", 2, 0x08, bfe_reg_opr_imm, 0},
|
|
|
|
|
{"bfext.p", 2, 0x08, bfe_reg_opr_imm, 0},
|
|
|
|
|
{"bfext.l", 2, 0x08, bfe_reg_opr_imm, 0},
|
|
|
|
|
{"bfext.b", 2, 0x08, bfe_opr_reg_imm, 0},
|
|
|
|
|
{"bfext.w", 2, 0x08, bfe_opr_reg_imm, 0},
|
|
|
|
|
{"bfext.p", 2, 0x08, bfe_opr_reg_imm, 0},
|
|
|
|
|
{"bfext.l", 2, 0x08, bfe_opr_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{"bfins", 2, 0x08, bfi_reg_reg_reg, 0},
|
|
|
|
|
{"bfins", 2, 0x08, bfi_reg_reg_imm, 0},
|
|
|
|
|
{"bfins.b", 2, 0x08, bfi_reg_opr_reg, 0},
|
|
|
|
|
{"bfins.w", 2, 0x08, bfi_reg_opr_reg, 0},
|
|
|
|
|
{"bfins.p", 2, 0x08, bfi_reg_opr_reg, 0},
|
|
|
|
|
{"bfins.l", 2, 0x08, bfi_reg_opr_reg, 0},
|
|
|
|
|
{"bfins.b", 2, 0x08, bfi_opr_reg_reg, 0},
|
|
|
|
|
{"bfins.w", 2, 0x08, bfi_opr_reg_reg, 0},
|
|
|
|
|
{"bfins.p", 2, 0x08, bfi_opr_reg_reg, 0},
|
|
|
|
|
{"bfins.l", 2, 0x08, bfi_opr_reg_reg, 0},
|
|
|
|
|
{"bfins.b", 2, 0x08, bfi_reg_opr_imm, 0},
|
|
|
|
|
{"bfins.w", 2, 0x08, bfi_reg_opr_imm, 0},
|
|
|
|
|
{"bfins.p", 2, 0x08, bfi_reg_opr_imm, 0},
|
|
|
|
|
{"bfins.l", 2, 0x08, bfi_reg_opr_imm, 0},
|
|
|
|
|
{"bfins.b", 2, 0x08, bfi_opr_reg_imm, 0},
|
|
|
|
|
{"bfins.w", 2, 0x08, bfi_opr_reg_imm, 0},
|
|
|
|
|
{"bfins.p", 2, 0x08, bfi_opr_reg_imm, 0},
|
|
|
|
|
{"bfins.l", 2, 0x08, bfi_opr_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{"minu", 2, 0x10, regd_opr, 0},
|
|
|
|
|
{"maxu", 2, 0x18, regd_opr, 0},
|
|
|
|
|
{"mins", 2, 0x20, regd_opr, 0},
|
|
|
|
|
{"maxs", 2, 0x28, regd_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"clb", 2, 0x91, tfr, 0},
|
|
|
|
|
|
|
|
|
|
{"trap", 2, 0x00, trap_imm, 0},
|
|
|
|
|
{"abs", 2, 0x40, reg_inh, 0},
|
|
|
|
|
{"sat", 2, 0xa0, reg_inh, 0},
|
|
|
|
|
|
|
|
|
|
{"rti", 2, 0x90, no_operands, 0},
|
|
|
|
|
{"stop", 2, 0x05, no_operands, 0},
|
|
|
|
|
{"wai", 2, 0x06, no_operands, 0},
|
|
|
|
|
{"sys", 2, 0x07, no_operands, 0},
|
|
|
|
|
|
|
|
|
|
{"bit", 2, 0x58, regd_imm, 0},
|
|
|
|
|
{"bit", 2, 0x68, regd_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"adc", 2, 0x50, regd_imm, 0},
|
|
|
|
|
{"adc", 2, 0x60, regd_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"sbc", 2, 0x70, regd_imm, 0},
|
|
|
|
|
{"eor", 2, 0x78, regd_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"sbc", 2, 0x80, regd_opr, 0},
|
|
|
|
|
{"eor", 2, 0x88, regd_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divs", 2, 0x30, mul_reg_reg_reg, 0},
|
|
|
|
|
{"divu", 2, 0x30, mul_reg_reg_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"divs.b", 2, 0x30, mul_reg_reg_opr, 0},
|
|
|
|
|
{"divs.w", 2, 0x30, mul_reg_reg_opr, 0},
|
|
|
|
|
{"divs.l", 2, 0x30, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divu.b", 2, 0x30, mul_reg_reg_opr, 0},
|
|
|
|
|
{"divu.w", 2, 0x30, mul_reg_reg_opr, 0},
|
|
|
|
|
{"divu.l", 2, 0x30, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divs.b", 2, 0x30, mul_reg_reg_imm, 0},
|
|
|
|
|
{"divs.w", 2, 0x30, mul_reg_reg_imm, 0},
|
|
|
|
|
{"divs.l", 2, 0x30, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"divu.b", 2, 0x30, mul_reg_reg_imm, 0},
|
|
|
|
|
{"divu.w", 2, 0x30, mul_reg_reg_imm, 0},
|
|
|
|
|
{"divu.l", 2, 0x30, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"divs.bb", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.bw", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.bp", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.bl", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divs.wb", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.ww", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.wp", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.wl", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divs.pb", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.pw", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.pp", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.pl", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divs.lb", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.lw", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.lp", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divs.ll", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divu.bb", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.bw", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.bp", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.bl", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divu.wb", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.ww", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.wp", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.wl", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divu.pb", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.pw", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.pp", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.pl", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"divu.lb", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.lw", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.lp", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
{"divu.ll", 2, 0x30, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmuls", 2, 0xb0, mul_reg_reg_reg, 0},
|
|
|
|
|
{"qmulu", 2, 0xb0, mul_reg_reg_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"qmuls.b", 2, 0xb0, mul_reg_reg_opr, 0},
|
|
|
|
|
{"qmuls.w", 2, 0xb0, mul_reg_reg_opr, 0},
|
|
|
|
|
{"qmuls.l", 2, 0xb0, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmulu.b", 2, 0xb0, mul_reg_reg_opr, 0},
|
|
|
|
|
{"qmulu.w", 2, 0xb0, mul_reg_reg_opr, 0},
|
|
|
|
|
{"qmulu.l", 2, 0xb0, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmuls.b", 2, 0xb0, mul_reg_reg_imm, 0},
|
|
|
|
|
{"qmuls.w", 2, 0xb0, mul_reg_reg_imm, 0},
|
|
|
|
|
{"qmuls.l", 2, 0xb0, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"qmulu.b", 2, 0xb0, mul_reg_reg_imm, 0},
|
|
|
|
|
{"qmulu.w", 2, 0xb0, mul_reg_reg_imm, 0},
|
|
|
|
|
{"qmulu.l", 2, 0xb0, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"qmuls.bb", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.bw", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.bp", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.bl", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmuls.wb", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.ww", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.wp", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.wl", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmuls.pb", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.pw", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.pp", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.pl", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmuls.lb", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.lw", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.lp", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmuls.ll", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmulu.bb", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.bw", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.bp", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.bl", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmulu.wb", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.ww", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.wp", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.wl", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmulu.pb", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.pw", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.pp", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.pl", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"qmulu.lb", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.lw", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.lp", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
{"qmulu.ll", 2, 0xb0, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macs", 2, 0x48, mul_reg_reg_reg, 0},
|
|
|
|
|
{"macu", 2, 0x48, mul_reg_reg_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"macs.b", 2, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
{"macs.w", 2, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
{"macs.l", 2, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macu.b", 2, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
{"macu.w", 2, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
{"macu.l", 2, 0x48, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macs.b", 2, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
{"macs.w", 2, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
{"macs.l", 2, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"macu.b", 2, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
{"macu.w", 2, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
{"macu.l", 2, 0x48, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"macs.bb", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.bw", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.bp", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.bl", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macs.wb", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.ww", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.wp", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.wl", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macs.pb", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.pw", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.pp", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.pl", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macs.lb", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.lw", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.lp", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macs.ll", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macu.bb", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.bw", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.bp", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.bl", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macu.wb", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.ww", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.wp", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.wl", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macu.pb", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.pw", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.pp", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.pl", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"macu.lb", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.lw", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.lp", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
{"macu.ll", 2, 0x48, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mods", 2, 0x38, mul_reg_reg_reg, 0},
|
|
|
|
|
{"modu", 2, 0x38, mul_reg_reg_reg, 0},
|
|
|
|
|
|
|
|
|
|
{"mods.b", 2, 0x38, mul_reg_reg_opr, 0},
|
|
|
|
|
{"mods.w", 2, 0x38, mul_reg_reg_opr, 0},
|
|
|
|
|
{"mods.l", 2, 0x38, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"modu.b", 2, 0x38, mul_reg_reg_opr, 0},
|
|
|
|
|
{"modu.w", 2, 0x38, mul_reg_reg_opr, 0},
|
|
|
|
|
{"modu.l", 2, 0x38, mul_reg_reg_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mods.b", 2, 0x38, mul_reg_reg_imm, 0},
|
|
|
|
|
{"mods.w", 2, 0x38, mul_reg_reg_imm, 0},
|
|
|
|
|
{"mods.l", 2, 0x38, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"modu.b", 2, 0x38, mul_reg_reg_imm, 0},
|
|
|
|
|
{"modu.w", 2, 0x38, mul_reg_reg_imm, 0},
|
|
|
|
|
{"modu.l", 2, 0x38, mul_reg_reg_imm, 0},
|
|
|
|
|
|
|
|
|
|
{"mods.bb", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.bw", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.bp", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.bl", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mods.wb", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.ww", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.wp", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.wl", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mods.pb", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.pw", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.pp", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.pl", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"mods.lb", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.lw", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.lp", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"mods.ll", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"modu.bb", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.bw", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.bp", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.bl", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"modu.wb", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.ww", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.wp", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.wl", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"modu.pb", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.pw", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.pp", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.pl", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
|
|
|
|
|
{"modu.lb", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.lw", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.lp", 2, 0x38, mul_reg_opr_opr, 0},
|
|
|
|
|
{"modu.ll", 2, 0x38, mul_reg_opr_opr, 0}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Gas line assembler entry point. */
|
|
|
|
|
|
|
|
|
|
/* This is the main entry point for the machine-dependent assembler. str
|
|
|
|
|
points to a machine-dependent instruction. This function is supposed to
|
|
|
|
|
emit the frags/bytes it assembles to. */
|
|
|
|
|
void
|
|
|
|
|
md_assemble (char *str)
|
|
|
|
|
{
|
|
|
|
|
char *op_start;
|
|
|
|
|
char *op_end;
|
|
|
|
|
char name[20];
|
|
|
|
|
size_t nlen = 0;
|
|
|
|
|
|
|
|
|
|
fail_line_pointer = NULL;
|
|
|
|
|
|
|
|
|
|
/* Find the opcode end and get the opcode in 'name'. The opcode is forced
|
|
|
|
|
lower case (the opcode table only has lower case op-codes). */
|
|
|
|
|
for (op_start = op_end = str;
|
|
|
|
|
*op_end && !is_end_of_line[(int)*op_end] && *op_end != ' ';
|
|
|
|
|
op_end++)
|
|
|
|
|
{
|
|
|
|
|
name[nlen] = TOLOWER (op_start[nlen]);
|
|
|
|
|
nlen++;
|
|
|
|
|
gas_assert (nlen < sizeof (name) - 1);
|
|
|
|
|
}
|
|
|
|
|
name[nlen] = 0;
|
|
|
|
|
|
|
|
|
|
if (nlen == 0)
|
|
|
|
|
{
|
|
|
|
|
as_bad (_("No instruction or missing opcode."));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
input_line_pointer = skip_whites (op_end);
|
|
|
|
|
|
|
|
|
|
size_t i;
|
|
|
|
|
for (i = 0; i < sizeof (opcodes) / sizeof (opcodes[0]); ++i)
|
|
|
|
|
{
|
|
|
|
|
const struct instruction *opc = opcodes + i;
|
|
|
|
|
if (0 == strcmp (name, opc->name))
|
|
|
|
|
{
|
|
|
|
|
if (opc->parse_operands (opc))
|
|
|
|
|
return;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
as_bad (_("Invalid instruction: \"%s\""), str);
|
|
|
|
|
as_bad (_("First invalid token: \"%s\""), fail_line_pointer);
|
|
|
|
|
while (*input_line_pointer++)
|
|
|
|
|
;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* 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)
|
|
|
|
|
{
|
|
|
|
|
long ret = fixP->fx_size + fixP->fx_frag->fr_address;
|
|
|
|
|
if (fixP->fx_addsy && S_IS_DEFINED (fixP->fx_addsy))
|
|
|
|
|
ret += fixP->fx_where;
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* We need a port-specific relaxation function to cope with sym2 - sym1
|
|
|
|
|
relative expressions with both symbols in the same segment (but not
|
|
|
|
|
necessarily in the same frag as this insn), for example:
|
|
|
|
|
ldab sym2-(sym1-2),pc
|
|
|
|
|
sym1:
|
|
|
|
|
The offset can be 5, 9 or 16 bits long. */
|
|
|
|
|
|
|
|
|
|
long
|
|
|
|
|
s12z_relax_frag (segT seg ATTRIBUTE_UNUSED, fragS *fragP ATTRIBUTE_UNUSED,
|
|
|
|
|
long stretch ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
md_convert_frag (bfd *abfd ATTRIBUTE_UNUSED, asection *sec ATTRIBUTE_UNUSED,
|
|
|
|
|
fragS *fragP ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* On an ELF system, we can't relax a weak symbol. The weak symbol
|
|
|
|
|
can be overridden at final link time by a non weak symbol. We can
|
|
|
|
|
relax externally visible symbol because there is no shared library
|
|
|
|
|
and such symbol can't be overridden (unless they are weak). */
|
|
|
|
|
|
|
|
|
|
/* 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 ATTRIBUTE_UNUSED, asection *segment ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return false;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* If while processing a fixup, a reloc really needs to be created
|
|
|
|
|
then it is done here. */
|
|
|
|
|
arelent *
|
|
|
|
|
tc_gen_reloc (asection *section, fixS *fixp)
|
|
|
|
|
{
|
|
|
|
|
arelent *reloc = XNEW (arelent);
|
|
|
|
|
reloc->sym_ptr_ptr = XNEW (asymbol *);
|
|
|
|
|
*reloc->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
|
|
|
|
|
reloc->address = fixp->fx_frag->fr_address + fixp->fx_where;
|
|
|
|
|
reloc->howto = bfd_reloc_type_lookup (stdoutput, fixp->fx_r_type);
|
|
|
|
|
if (reloc->howto == (reloc_howto_type *) NULL)
|
|
|
|
|
{
|
|
|
|
|
as_bad_where (fixp->fx_file, fixp->fx_line,
|
|
|
|
|
_("Relocation %d is not supported by object file format."),
|
|
|
|
|
(int) fixp->fx_r_type);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (0 == (section->flags & SEC_CODE))
|
|
|
|
|
reloc->addend = fixp->fx_offset;
|
|
|
|
|
else
|
|
|
|
|
reloc->addend = fixp->fx_addnumber;
|
|
|
|
|
|
|
|
|
|
return reloc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* See whether we need to force a relocation into the output file. */
|
|
|
|
|
int
|
|
|
|
|
tc_s12z_force_relocation (fixS *fixP)
|
|
|
|
|
{
|
|
|
|
|
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. */
|
2019-04-13 00:39:01 +08:00
|
|
|
|
bfd_boolean
|
2018-05-18 22:26:18 +08:00
|
|
|
|
tc_s12z_fix_adjustable (fixS *fixP ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
2019-04-13 00:39:01 +08:00
|
|
|
|
return true;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
md_apply_fix (fixS *fixP, valueT *valP, segT seg ATTRIBUTE_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
long value = *valP;
|
|
|
|
|
|
|
|
|
|
if (fixP->fx_addsy == (symbolS *) NULL)
|
|
|
|
|
fixP->fx_done = 1;
|
|
|
|
|
|
|
|
|
|
/* We don't actually support subtracting a symbol. */
|
|
|
|
|
if (fixP->fx_subsy != (symbolS *) NULL)
|
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line, _("Expression too complex."));
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Patch the instruction with the resolved operand. Elf relocation
|
|
|
|
|
info will also be generated to take care of linker/loader fixups.
|
|
|
|
|
*/
|
|
|
|
|
char *where = fixP->fx_frag->fr_literal + fixP->fx_where;
|
|
|
|
|
|
|
|
|
|
switch (fixP->fx_r_type)
|
|
|
|
|
{
|
|
|
|
|
case BFD_RELOC_8:
|
|
|
|
|
((bfd_byte *) where)[0] = (bfd_byte) value;
|
|
|
|
|
break;
|
2018-10-23 21:33:18 +08:00
|
|
|
|
case BFD_RELOC_16:
|
|
|
|
|
bfd_putb16 ((bfd_vma) value, (unsigned char *) where);
|
|
|
|
|
break;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
case BFD_RELOC_24:
|
|
|
|
|
bfd_putb24 ((bfd_vma) value, (unsigned char *) where);
|
|
|
|
|
break;
|
2019-02-02 00:42:54 +08:00
|
|
|
|
case BFD_RELOC_S12Z_OPR:
|
|
|
|
|
{
|
|
|
|
|
switch (fixP->fx_size)
|
|
|
|
|
{
|
|
|
|
|
case 3:
|
|
|
|
|
bfd_putb24 ((bfd_vma) value, (unsigned char *) where);
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
bfd_putb16 ((bfd_vma) value, (unsigned char *) where);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
abort ();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
2018-05-18 22:26:18 +08:00
|
|
|
|
case BFD_RELOC_32:
|
|
|
|
|
bfd_putb32 ((bfd_vma) value, (unsigned char *) where);
|
|
|
|
|
break;
|
|
|
|
|
case BFD_RELOC_16_PCREL:
|
2019-02-02 00:42:54 +08:00
|
|
|
|
if (value < -0x4000 || value > 0x3FFF)
|
2018-05-18 22:26:18 +08:00
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
|
|
|
|
_("Value out of 16-bit range."));
|
|
|
|
|
|
|
|
|
|
bfd_putb16 ((bfd_vma) value | 0x8000, (unsigned char *) where);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
as_fatal (_("Line %d: unknown relocation type: 0x%x."),
|
|
|
|
|
fixP->fx_line, fixP->fx_r_type);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Set the ELF specific flags. */
|
|
|
|
|
void
|
|
|
|
|
s12z_elf_final_processing (void)
|
|
|
|
|
{
|
|
|
|
|
}
|