275 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			275 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
| /* Opcode table for the ARC.
 | |
|    Copyright 1994, 1995, 1997 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, 59 Temple Place - Suite 330,
 | |
| Boston, MA 02111-1307, 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_BASE 0
 | |
| #define ARC_MACH_UNUSED1 1
 | |
| #define ARC_MACH_UNUSED2 2
 | |
| #define ARC_MACH_UNUSED4 4
 | |
| /* Additional cpu values can be inserted here and ARC_MACH_BIG moved down.  */
 | |
| #define ARC_MACH_BIG 8
 | |
| 
 | |
| /* Mask of number of bits necessary to record cpu type.  */
 | |
| #define ARC_MACH_CPU_MASK 7
 | |
| /* Mask of number of bits necessary to record cpu type + endianness.  */
 | |
| #define ARC_MACH_MASK 15
 | |
| 
 | |
| /* 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) << 0)
 | |
| /* This insn is a conditional branch.  */
 | |
| #define ARC_OPCODE_COND_BRANCH (ARC_OPCODE_FLAG_START)
 | |
| 
 | |
|   /* 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_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
 | |
| 
 | |
| /* 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
 | |
| 
 | |
| /* Sum of all ARC_MOD_XXX bits.  */
 | |
| #define ARC_MOD_BITS 0x7000
 | |
| 
 | |
| /* Non-zero if the operand type is really a modifier.  */
 | |
| #define ARC_MOD_P(X) ((X) & ARC_MOD_BITS)
 | |
| 
 | |
|   /* 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) PARAMS ((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) PARAMS ((arc_insn *insn,
 | |
| 			   const struct arc_operand *operand,
 | |
| 			   int mods, const struct arc_operand_value **opval,
 | |
| 			   int *invalid));
 | |
| };
 | |
| 
 | |
| /* 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_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[];
 | |
| extern const int arc_operand_count;
 | |
| extern /*const*/ struct arc_opcode arc_opcodes[];
 | |
| extern const int arc_opcodes_count;
 | |
| extern const struct arc_operand_value arc_suffixes[];
 | |
| extern const int arc_suffixes_count;
 | |
| extern const struct arc_operand_value arc_reg_names[];
 | |
| extern const int arc_reg_names_count;
 | |
| extern unsigned char arc_operand_map[];
 | |
| 
 | |
| /* Utility fns in arc-opc.c.  */
 | |
| int arc_get_opcode_mach PARAMS ((int, int));
 | |
| /* `arc_opcode_init_tables' must be called before `arc_xxx_supported'.  */
 | |
| void arc_opcode_init_tables PARAMS ((int));
 | |
| void arc_opcode_init_insert PARAMS ((void));
 | |
| void arc_opcode_init_extract PARAMS ((void));
 | |
| const struct arc_opcode *arc_opcode_lookup_asm PARAMS ((const char *));
 | |
| const struct arc_opcode *arc_opcode_lookup_dis PARAMS ((unsigned int));
 | |
| int arc_opcode_limm_p PARAMS ((long *));
 | |
| const struct arc_operand_value *arc_opcode_lookup_suffix PARAMS ((const struct arc_operand *type, int value));
 | |
| int arc_opcode_supported PARAMS ((const struct arc_opcode *));
 | |
| int arc_opval_supported PARAMS ((const struct arc_operand_value *));
 |