blob: 3c3f5eb25ff86f16b7546667414bf753fa512e8e [file] [log] [blame]
/* Opcode table for the ARC.
Copyright 1994, 1995, 1997, 2001, 2002, 2003
Free Software Foundation, Inc.
Contributed by Doug Evans (dje@cygnus.com).
This file is part of GAS, the GNU Assembler, GDB, the GNU debugger, and
the GNU Binutils.
GAS/GDB 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 2, or (at your option)
any later version.
GAS/GDB 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 or GDB; see the file COPYING. If not, write to
the Free Software Foundation, 51 Franklin Street - Fifth Floor, Boston,
MA 02110-1301, USA. */
/* List of the various cpu types.
The tables currently use bit masks to say whether the instruction or
whatever is supported by a particular cpu. This lets us have one entry
apply to several cpus.
This duplicates bfd_mach_arc_xxx. For now I wish to isolate this from bfd
and bfd from this. Also note that these numbers are bit values as we want
to allow for things available on more than one ARC (but not necessarily all
ARCs). */
/* The `base' cpu must be 0 (table entries are omitted for the base cpu).
The cpu type is treated independently of endianness.
The complete `mach' number includes endianness.
These values are internal to opcodes/bfd/binutils/gas. */
#define ARC_MACH_ARC4 1
#define ARC_MACH_ARC5 2
#define ARC_MACH_ARC6 4
#define ARC_MACH_ARC7 8
/* Additional cpu values can be inserted here and ARC_MACH_BIG moved down. */
#define ARC_MACH_BIG 16
/* ARC processors which implement ARCompact ISA. */
#define ARCOMPACT (ARC_MACH_ARC5 | ARC_MACH_ARC6 | ARC_MACH_ARC7)
/* Mask of number of bits necessary to record cpu type. */
#define ARC_MACH_CPU_MASK (ARC_MACH_BIG - 1)
/* Mask of number of bits necessary to record cpu type + endianness. */
#define ARC_MACH_MASK ((ARC_MACH_BIG << 1) - 1)
/* Type to denote an ARC instruction (at least a 32 bit unsigned int). */
typedef unsigned int arc_insn;
struct arc_opcode {
char *syntax; /* syntax of insn */
unsigned long mask, value; /* recognize insn if (op&mask)==value */
int flags; /* various flag bits */
/* Values for `flags'. */
/* Return CPU number, given flag bits. */
#define ARC_OPCODE_CPU(bits) ((bits) & ARC_MACH_CPU_MASK)
/* Return MACH number, given flag bits. */
#define ARC_OPCODE_MACH(bits) ((bits) & ARC_MACH_MASK)
/* First opcode flag bit available after machine mask. */
#define ARC_OPCODE_FLAG_START (ARC_MACH_MASK + 1)
/* This insn is a conditional branch. */
#define ARC_OPCODE_COND_BRANCH (ARC_OPCODE_FLAG_START)
#define SYNTAX_LENGTH (ARC_OPCODE_COND_BRANCH << 1)
#define SYNTAX_3OP (SYNTAX_LENGTH )
#define SYNTAX_2OP (SYNTAX_3OP << 1)
#define SYNTAX_1OP (SYNTAX_2OP << 1)
#define SYNTAX_NOP (SYNTAX_1OP << 1)
#define OP1_DEST_IGNORED (SYNTAX_NOP << 1)
#define OP1_MUST_BE_IMM (OP1_DEST_IGNORED << 1)
#define OP1_IMM_IMPLIED (OP1_MUST_BE_IMM << 1)
#define SUFFIX_NONE (OP1_IMM_IMPLIED << 1)
#define SUFFIX_COND (SUFFIX_NONE << 1)
#define SUFFIX_FLAG (SUFFIX_COND << 1)
#define SYNTAX_VALID (SUFFIX_FLAG << 1)
#define AC_SYNTAX_3OP (0x01)
#define AC_SYNTAX_2OP (AC_SYNTAX_3OP << 1)
#define AC_SYNTAX_1OP (AC_SYNTAX_2OP << 1)
#define AC_SYNTAX_NOP (AC_SYNTAX_1OP << 1)
#define AC_SYNTAX_SIMD (AC_SYNTAX_NOP << 1)
#define AC_OP1_DEST_IGNORED (AC_SYNTAX_SIMD << 1)
#define AC_OP1_MUST_BE_IMM (AC_OP1_DEST_IGNORED << 1)
#define AC_OP1_IMM_IMPLIED (AC_OP1_MUST_BE_IMM << 1)
#define AC_SIMD_SYNTAX_VVV (AC_OP1_IMM_IMPLIED << 1)
#define AC_SIMD_SYNTAX_VV0 (AC_SIMD_SYNTAX_VVV << 1)
#define AC_SIMD_SYNTAX_VbI0 (AC_SIMD_SYNTAX_VV0 << 1)
#define AC_SIMD_SYNTAX_Vb00 (AC_SIMD_SYNTAX_VbI0 << 1)
#define AC_SIMD_SYNTAX_VbC0 (AC_SIMD_SYNTAX_Vb00 << 1)
#define AC_SIMD_SYNTAX_V00 (AC_SIMD_SYNTAX_VbC0 << 1)
#define AC_SIMD_SYNTAX_VC0 (AC_SIMD_SYNTAX_V00 << 1)
#define AC_SIMD_SYNTAX_VVC (AC_SIMD_SYNTAX_VC0 << 1)
#define AC_SIMD_SYNTAX_VV (AC_SIMD_SYNTAX_VVC << 1)
#define AC_SIMD_SYNTAX_VVI (AC_SIMD_SYNTAX_VV << 1)
#define AC_SIMD_SYNTAX_C (AC_SIMD_SYNTAX_VVI << 1)
#define AC_SIMD_SYNTAX_0 (AC_SIMD_SYNTAX_C << 1)
#define AC_SIMD_SYNTAX_CC (AC_SIMD_SYNTAX_0 << 1)
#define AC_SIMD_SYNTAX_C0 (AC_SIMD_SYNTAX_CC << 1)
#define AC_SIMD_SYNTAX_DC (AC_SIMD_SYNTAX_C0 << 1)
#define AC_SIMD_SYNTAX_D0 (AC_SIMD_SYNTAX_DC << 1)
#define AC_SIMD_SYNTAX_VD (AC_SIMD_SYNTAX_D0 << 1)
//#define AC_SUFFIX_NONE (AC_SIMD_SYNTAX_VD << 1)
#define AC_SUFFIX_NONE (0x1)
#define AC_SUFFIX_COND (AC_SUFFIX_NONE << 1)
#define AC_SUFFIX_FLAG (AC_SUFFIX_COND << 1)
#define AC_SIMD_FLAGS_NONE (AC_SUFFIX_FLAG << 1)
#define AC_SIMD_FLAG_SET (AC_SIMD_FLAGS_NONE << 1)
#define AC_SIMD_FLAG1_SET (AC_SIMD_FLAG_SET << 1)
#define AC_SIMD_FLAG2_SET (AC_SIMD_FLAG1_SET << 1)
#define AC_SIMD_ENCODE_U8 (AC_SIMD_FLAG2_SET << 1)
#define AC_SIMD_ENCODE_U6 (AC_SIMD_ENCODE_U8 << 1)
#define AC_SIMD_SCALE_1 (AC_SIMD_ENCODE_U6 << 1)
#define AC_SIMD_SCALE_2 (AC_SIMD_SCALE_1 << 1)
#define AC_SIMD_SCALE_3 (AC_SIMD_SCALE_2 << 1)
#define AC_SIMD_SCALE_4 (AC_SIMD_SCALE_3 << 1)
#define AC_SIMD_ENCODE_LIMM (AC_SIMD_SCALE_4 << 1)
#define I(x) (((unsigned) (x) & 31) << 27)
#define A(x) (((unsigned) (x) & ARC_MASK_REG) << ARC_SHIFT_REGA)
#define B(x) (((unsigned) (x) & ARC_MASK_REG) << ARC_SHIFT_REGB)
#define C(x) (((unsigned) (x) & ARC_MASK_REG) << ARC_SHIFT_REGC)
#define R(x,b,m) (((unsigned) (x) & (m)) << (b)) /* value X, mask M, at bit B */
/* These values are used to optimize assembly and disassembly. Each insn is
on a list of related insns (same first letter for assembly, same insn code
for disassembly). */
struct arc_opcode *next_asm; /* Next instruction to try during assembly. */
struct arc_opcode *next_dis; /* Next instruction to try during disassembly. */
/* Macros to create the hash values for the lists. */
#define ARC_HASH_OPCODE(string) \
((string)[0] >= 'a' && (string)[0] <= 'z' ? (string)[0] - 'a' : 26)
#define ARC_HASH_ICODE(insn) \
((unsigned int) (insn) >> 27)
/* Macros to access `next_asm', `next_dis' so users needn't care about the
underlying mechanism. */
#define ARC_OPCODE_NEXT_ASM(op) ((op)->next_asm)
#define ARC_OPCODE_NEXT_DIS(op) ((op)->next_dis)
};
struct arc_operand_value {
char *name; /* eg: "eq" */
short value; /* eg: 1 */
unsigned char type; /* index into `arc_operands' */
unsigned char flags; /* various flag bits */
/* Values for `flags'. */
/* Return CPU number, given flag bits. */
#define ARC_OPVAL_CPU(bits) ((bits) & ARC_MACH_CPU_MASK)
/* Return MACH number, given flag bits. */
#define ARC_OPVAL_MACH(bits) ((bits) & ARC_MACH_MASK)
};
struct arc_ext_operand_value {
struct arc_ext_operand_value *next;
struct arc_operand_value operand;
} *arc_ext_operands;
/* List of extension condition codes, core registers and auxiliary registers.
Calls to gas/config/tc-arc.c:arc_extoper built up this list. */
extern struct arc_ext_operand_value *arc_ext_operands;
struct arc_operand {
/* One of the insn format chars. */
unsigned char fmt;
/* The number of bits in the operand (may be unused for a modifier). */
unsigned char bits;
/* How far the operand is left shifted in the instruction, or
the modifier's flag bit (may be unused for a modifier. */
unsigned char shift;
/* Various flag bits. */
int flags;
/* Values for `flags'. */
/* This operand is a suffix to the opcode. */
#define ARC_OPERAND_SUFFIX 1
/* This operand is a relative branch displacement. The disassembler
prints these symbolically if possible. */
#define ARC_OPERAND_RELATIVE_BRANCH 2
/* This operand is an absolute branch address. The disassembler
prints these symbolically if possible. */
#define ARC_OPERAND_ABSOLUTE_BRANCH 4
/* This operand is an address. The disassembler
prints these symbolically if possible. */
#define ARC_OPERAND_ADDRESS 8
/* This operand is a long immediate value. */
#define ARC_OPERAND_LIMM 0x10
/* This operand takes signed values. */
#define ARC_OPERAND_SIGNED 0x20
/* This operand takes signed values, but also accepts a full positive
range of values. That is, if bits is 16, it takes any value from
-0x8000 to 0xffff. */
#define ARC_OPERAND_SIGNOPT 0x40
/* This operand should be regarded as a negative number for the
purposes of overflow checking (i.e., the normal most negative
number is disallowed and one more than the normal most positive
number is allowed). This flag will only be set for a signed
operand. */
#define ARC_OPERAND_NEGATIVE 0x80
/* This operand doesn't really exist. The program uses these operands
in special ways. */
#define ARC_OPERAND_FAKE 0x100
/* separate flags operand for j and jl instructions */
#define ARC_OPERAND_JUMPFLAGS 0x200
/* allow warnings and errors to be issued after call to insert_xxxxxx */
#define ARC_OPERAND_WARN 0x400
#define ARC_OPERAND_ERROR 0x800
/* this is a load operand */
#define ARC_OPERAND_LOAD 0x8000
/* this is a store operand */
#define ARC_OPERAND_STORE 0x10000
/* this is an unsigned operand */
#define ARC_OPERAND_UNSIGNED 0x20000
/* this operand's value must be 2-byte aligned */
#define ARC_OPERAND_2BYTE_ALIGNED 0x40000
/* this operand's value must be 4-byte aligned */
#define ARC_OPERAND_4BYTE_ALIGNED 0x80000
/* Modifier values. */
/* A dot is required before a suffix. Eg: .le */
#define ARC_MOD_DOT 0x1000
/* A normal register is allowed (not used, but here for completeness). */
#define ARC_MOD_REG 0x2000
/* An auxiliary register name is expected. */
#define ARC_MOD_AUXREG 0x4000
/* This should be a small data symbol, i.e. suffixed with an @sda */
#define ARC_MOD_SDASYM 0x100000
/* Sum of all ARC_MOD_XXX bits. */
#define ARC_MOD_BITS 0x107000
/* Non-zero if the operand type is really a modifier. */
#define ARC_MOD_P(X) ((X) & ARC_MOD_BITS)
/* enforce read/write only register restrictions */
#define ARC_REGISTER_READONLY 0x01
#define ARC_REGISTER_WRITEONLY 0x02
#define ARC_REGISTER_NOSHORT_CUT 0x04
/* Registers which are normally used in 16-bit ARCompact insns */
#define ARC_REGISTER_16 0x8
/*
FIXME: The following 5 definitions is a unclean way of passing
information to md_assemble. New opcode is a possibility but its
already very crowded.
*/
/*The u6 operand needs to be incremented by 1 for some pseudo mnemonics of
the BRcc instruction. */
#define ARC_INCR_U6 0x100000
#define ARC_SIMD_SCALE1 (ARC_INCR_U6 << 0x1)
#define ARC_SIMD_SCALE2 (ARC_SIMD_SCALE1 << 0x1)
#define ARC_SIMD_SCALE3 (ARC_SIMD_SCALE2 << 0x1)
#define ARC_SIMD_SCALE4 (ARC_SIMD_SCALE3 << 0x1)
/* Registers for the Aurora SIMD ISA*/
#define ARC_REGISTER_SIMD_VR 0x10
#define ARC_REGISTER_SIMD_I 0x20
#define ARC_REGISTER_SIMD_DR 0x40
/* Insertion function. This is used by the assembler. To insert an
operand value into an instruction, check this field.
If it is NULL, execute
i |= (p & ((1 << o->bits) - 1)) << o->shift;
(I is the instruction which we are filling in, O is a pointer to
this structure, and OP is the opcode value; this assumes twos
complement arithmetic).
If this field is not NULL, then simply call it with the
instruction and the operand value. It will return the new value
of the instruction. If the ERRMSG argument is not NULL, then if
the operand value is illegal, *ERRMSG will be set to a warning
string (the operand will be inserted in any case). If the
operand value is legal, *ERRMSG will be unchanged.
REG is non-NULL when inserting a register value. */
arc_insn (*insert) (arc_insn insn, const struct arc_operand *operand,
int mods, const struct arc_operand_value *reg,
long value, const char **errmsg);
/* Extraction function. This is used by the disassembler. To
extract this operand type from an instruction, check this field.
If it is NULL, compute
op = ((i) >> o->shift) & ((1 << o->bits) - 1);
if ((o->flags & ARC_OPERAND_SIGNED) != 0
&& (op & (1 << (o->bits - 1))) != 0)
op -= 1 << o->bits;
(I is the instruction, O is a pointer to this structure, and OP
is the result; this assumes twos complement arithmetic).
If this field is not NULL, then simply call it with the
instruction value. It will return the value of the operand. If
the INVALID argument is not NULL, *INVALID will be set to
non-zero if this operand type can not actually be extracted from
this operand (i.e., the instruction does not match). If the
operand is valid, *INVALID will not be changed.
INSN is a pointer to an array of two `arc_insn's. The first element is
the insn, the second is the limm if present.
Operands that have a printable form like registers and suffixes have
their struct arc_operand_value pointer stored in OPVAL. */
long (*extract) (arc_insn *insn,
const struct arc_operand *operand, int mods,
const struct arc_operand_value **opval, int *invalid);
};
enum
{
BR_exec_when_no_jump,
BR_exec_always,
BR_exec_when_jump
};
enum ARC_Debugger_OperandType
{
ARC_UNDEFINED,
ARC_LIMM,
ARC_SHIMM,
ARC_REGISTER,
ARCOMPACT_REGISTER /* Valid only for the
registers allowed in
16 bit mode */
};
enum Flow
{
noflow,
direct_jump,
direct_call,
indirect_jump,
indirect_call,
invalid_instr
};
enum { no_reg = 99 };
enum { allOperandsSize = 256 };
struct arcDisState
{
void *_this;
int instructionLen;
void (*err)(void*, const char*);
const char *(*coreRegName)(void*, int);
const char *(*auxRegName)(void*, int);
const char *(*condCodeName)(void*, int);
const char *(*instName)(void*, int, int, int*);
unsigned char* instruction;
unsigned index;
const char *comm[6]; /* instr name, cond, NOP, 3 operands */
union {
unsigned int registerNum;
unsigned int shortimm;
unsigned int longimm;
} source_operand;
enum ARC_Debugger_OperandType sourceType;
int opWidth;
int targets[4];
int addresses[4];
/* Set as a side-effect of calling the disassembler.
Used only by the debugger. */
enum Flow flow;
int register_for_indirect_jump;
int ea_reg1, ea_reg2, _offset;
int _cond, _opcode;
unsigned long words[2];
char *commentBuffer;
char instrBuffer[40];
char operandBuffer[allOperandsSize];
char _ea_present;
char _addrWriteBack; /* Address writeback */
char _mem_load;
char _load_len;
char nullifyMode;
unsigned char commNum;
unsigned char isBranch;
unsigned char tcnt;
unsigned char acnt;
};
/* Bits that say what version of cpu we have.
These should be passed to arc_init_opcode_tables.
At present, all there is is the cpu type. */
/* CPU number, given value passed to `arc_init_opcode_tables'. */
#define ARC_HAVE_CPU(bits) ((bits) & ARC_MACH_CPU_MASK)
/* MACH number, given value passed to `arc_init_opcode_tables'. */
#define ARC_HAVE_MACH(bits) ((bits) & ARC_MACH_MASK)
/* Special register values: */
#define ARC_REG_SHIMM_UPDATE 61
#define ARC_REG_SHIMM 63
#define ARC_REG_LIMM 62
/* Non-zero if REG is a constant marker. */
#define ARC_REG_CONSTANT_P(REG) ((REG) >= 61)
/* Positions and masks of various fields: */
#define ARC_SHIFT_REGA 21
#define ARC_SHIFT_REGB 15
#define ARC_SHIFT_REGC 9
#define ARC_SHIFT_REGA_AC 0
#define ARC_SHIFT_REGB_LOW_AC 24
#define ARC_SHIFT_REGB_HIGH_AC 12
#define ARC_SHIFT_REGC_AC 6
#define ARC_MASK_REG 63
/* Delay slot types. */
#define ARC_DELAY_NONE 0 /* no delay slot */
#define ARC_DELAY_NORMAL 1 /* delay slot in both cases */
#define ARC_DELAY_JUMP 2 /* delay slot only if branch taken */
/* Non-zero if X will fit in a signed 9 bit field. */
#define ARC_SHIMM_CONST_P(x) ((long) (x) >= -256 && (long) (x) <= 255)
extern const struct arc_operand arc_operands_a4[];
extern const struct arc_operand arc_operands_ac[];
extern const struct arc_operand *arc_operands;
extern int arc_operand_count;
extern const struct arc_operand_value arc_suffixes_a4[];
extern const struct arc_operand_value arc_suffixes_ac[];
extern const struct arc_operand_value *arc_suffixes;
extern int arc_suffixes_count;
extern const struct arc_operand_value arc_reg_names_a4[];
extern const struct arc_operand_value arc_reg_names_ac[];
extern const struct arc_operand_value *arc_reg_names;
extern int arc_reg_names_count;
extern unsigned char arc_operand_map_a4[];
extern unsigned char arc_operand_map_ac[];
extern unsigned char *arc_operand_map;
//extern int mach_a4;
//extern int compact_insn_16;
int mach_a4;
int compact_insn_16;
/* Utility fns in arc-opc.c. */
int arc_get_opcode_mach (int, int);
/* `arc_opcode_init_tables' must be called before `arc_xxx_supported'. */
void arc_opcode_init_tables (int);
void arc_opcode_init_insert (void);
void arc_opcode_init_extract (void);
const struct arc_opcode *arc_opcode_lookup_asm (const char *);
const struct arc_opcode *arc_opcode_lookup_dis (unsigned int);
int arc_opcode_limm_p (long *);
const struct arc_operand_value *arc_opcode_lookup_suffix (const struct arc_operand *type, int value);
int arc_opcode_supported (const struct arc_opcode *);
int arc_opval_supported (const struct arc_operand_value *);
int arc_insn_not_jl (arc_insn);
extern char *arc_aux_reg_name (int);
extern struct arc_operand_value *get_ext_suffix (char *);
extern int ac_branch_or_jump_insn (arc_insn);
extern int ac_lpcc_insn (arc_insn);
extern int ac_constant_operand (const struct arc_operand *);
extern int ac_register_operand (const struct arc_operand *);
extern int ac_symbol_operand (const struct arc_operand *);
extern int ARC700_register_simd_operand (char);
extern int arc_operand_type (int);
extern int ac_add_reg_sdasym_insn (arc_insn);
extern int ac_get_load_sdasym_insn_type (arc_insn, int);
extern int ac_get_store_sdasym_insn_type (arc_insn, int);
extern int arc_limm_fixup_adjust (arc_insn);