diff options
author | Nick Clifton <nickc@redhat.com> | 2004-07-07 17:28:53 +0000 |
---|---|---|
committer | Nick Clifton <nickc@redhat.com> | 2004-07-07 17:28:53 +0000 |
commit | 1fe1f39c06f2d04c2058b902feeafe46fd1ff48b (patch) | |
tree | 7b86a452584e85765f7fb2facedad588636e177b /gas/config/tc-crx.c | |
parent | 5c02dc5924528fb856e17cb5c7e8ea30a1f7111d (diff) | |
download | gdb-1fe1f39c06f2d04c2058b902feeafe46fd1ff48b.zip gdb-1fe1f39c06f2d04c2058b902feeafe46fd1ff48b.tar.gz gdb-1fe1f39c06f2d04c2058b902feeafe46fd1ff48b.tar.bz2 |
Add new port: crx-elf
Diffstat (limited to 'gas/config/tc-crx.c')
-rw-r--r-- | gas/config/tc-crx.c | 2506 |
1 files changed, 2506 insertions, 0 deletions
diff --git a/gas/config/tc-crx.c b/gas/config/tc-crx.c new file mode 100644 index 0000000..3b3c49e --- /dev/null +++ b/gas/config/tc-crx.c @@ -0,0 +1,2506 @@ +/* tc-crx.c -- Assembler code for the CRX CPU core. + Copyright 2004 Free Software Foundation, Inc. + + Contributed by Tomer Levi, NSC, Israel. + Originally written for GAS 2.12 by Tomer Levi, NSC, Israel. + Updates, BFDizing, GNUifying and ELF support by Tomer Levi. + + 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 2, 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, 59 Temple Place - Suite 330, Boston, + MA 02111-1307, USA. */ + +#include "as.h" +#include "safe-ctype.h" +#include "dwarf2dbg.h" +#include "opcode/crx.h" +#include "elf/crx.h" + +/* Include <limits.h> do define ULONG_MAX, LONG_MAX, LONG_MIN. */ +#include <limits.h> + +/* Word is considered here as a 16-bit unsigned short int. */ +#define WORD_SIZE 16 +#define WORD_SHIFT 16 + +/* Register is 4-bit size. */ +#define REG_SIZE 4 + +/* Maximum size of a single instruction (in words). */ +#define INSN_MAX_SIZE 3 + +/* Maximum bits which may be set in a `mask16' operand. */ +#define MAX_REGS_IN_MASK16 8 + +/* Escape to 16-bit immediate. */ +#define ESC_16 0xE +/* Escape to 32-bit immediate. */ +#define ESC_32 0xF + +/* Utility macros for string comparison. */ +#define streq(a, b) (strcmp (a, b) == 0) +#define strneq(a, b, c) (strncmp (a, b, c) == 0) + +/* A mask to set n_bits starting from offset offs. */ +#define SET_BITS_MASK(offs,n_bits) ((((1 << (n_bits)) - 1) << (offs))) +/* A mask to clear n_bits starting from offset offs. */ +#define CLEAR_BITS_MASK(offs,n_bits) (~(((1 << (n_bits)) - 1) << (offs))) + +/* Get the argument type for each operand of a given instruction. */ +#define GET_ACTUAL_TYPE \ + for (i = 0; i < insn->nargs; i++) \ + atyp_act[i] = getarg_type (instruction->operands[i].op_type) + +/* Get the size (in bits) for each operand of a given instruction. */ +#define GET_ACTUAL_SIZE \ + for (i = 0; i < insn->nargs; i++) \ + bits_act[i] = getbits (instruction->operands[i].op_type) + +/* Non-zero if OP is instruction with no operands. */ +#define NO_OPERANDS_INST(OP) \ + (streq (OP, "di") || streq (OP, "nop") \ + || streq (OP, "retx") || streq (OP, "ei") \ + || streq (OP, "wait") || streq (OP, "eiwait")) + +/* Print a number NUM, shifted by SHIFT bytes, into a location + pointed by index BYTE of array 'output_opcode'. */ +#define CRX_PRINT(BYTE, NUM, SHIFT) output_opcode[BYTE] |= (NUM << SHIFT) + +/* Opcode mnemonics hash table. */ +static struct hash_control *crx_inst_hash; +/* CRX registers hash table. */ +static struct hash_control *reg_hash; +/* CRX coprocessor registers hash table. */ +static struct hash_control *copreg_hash; +/* Current instruction we're assembling. */ +const inst *instruction; + +/* Initialize global variables. */ +long output_opcode[2]; +/* Nonzero means a relocatable symbol. */ +int relocatable; +/* Nonzero means a constant's bit-size was already set. */ +int size_was_set; +/* Nonzero means a negative constant. */ +int signflag; +/* Nonzero means a CST4 instruction. */ +int cst4flag; +/* A copy of the original instruction (used in error messages). */ +char ins_parse[MAX_INST_LEN]; +/* Nonzero means instruction is represented in post increment mode. */ +int post_inc_mode; +/* Holds the current processed argument number. */ +int processing_arg_number; + +/* Generic assembler global variables which must be defined by all targets. */ + +/* Characters which always start a comment. */ +const char comment_chars[] = "#"; + +/* Characters which start a comment at the beginning of a line. */ +const char line_comment_chars[] = "#"; + +/* This array holds machine specific line separator characters. */ +const char line_separator_chars[] = ";"; + +/* Chars that can be used to separate mant from exp in floating point nums. */ +const char EXP_CHARS[] = "eE"; + +/* Chars that mean this number is a floating point constant as in 0f12.456 */ +const char FLT_CHARS[] = "f'"; + +/* Target-specific multicharacter options, not const-declared at usage. */ +const char *md_shortopts = ""; +struct option md_longopts[] = { + {NULL, no_argument, NULL, 0} +}; +size_t md_longopts_size = sizeof (md_longopts); + +/* 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[] = +{ + /* In CRX machine, align is in bytes (not a ptwo boundary). */ + {"align", s_align_bytes, 0}, + {0, 0, 0} +}; + +const relax_typeS md_relax_table[] = +{ + /* bCC */ + {0xfa, -0x100, 2, 1}, /* 8 */ + {0xfffe, -0x10000, 4, 2}, /* 16 */ + {0xfffffffe, -0xfffffffe, 6, 0}, /* 32 */ + + /* bal */ + {0xfffe, -0x10000, 4, 4}, /* 16 */ + {0xfffffffe, -0xfffffffe, 6, 0}, /* 32 */ + + /* cmpbr */ + {0xfe, -0x100, 4, 6}, /* 8 */ + {0xfffffe, -0x1000000, 6, 0} /* 24 */ +}; + +static void reset_vars (char *, ins *); +static reg get_register (char *); +static copreg get_copregister (char *); +static void get_number_of_bits (ins *, int); +static argtype getarg_type (operand_type); +static int getbits (operand_type); +static int get_number_of_operands (void); +static void get_operandtype (char *, int, ins *); +static int gettrap (char *); +static void handle_pi_insn (char *); +static int get_cinv_parameters (char *); +static unsigned long getconstant (unsigned long, int); +static int getreg_image (reg); +static void parse_operands (ins *, char *); +static void parse_insn (ins *, char *); +static void print_operand (int, int, argument *); +static void print_constant (int, int, argument *); +static int exponent2scale (int); +static void mask_const (unsigned long *, int); +static void mask_reg (int, unsigned short *); +static int process_label_constant (char *, ins *, int); +static void set_indexmode_parameters (char *, ins *, int); +static void set_cons_rparams (char *, ins *, int); +static char * preprocess_reglist (char *, int *); +static int assemble_insn (char *, ins *); +static void print_insn (ins *); + +/* Return the bit size for a given operand. */ + +static int +getbits (operand_type op) +{ + if (op < MAX_OPRD) + return crx_optab[op].bit_size; + else + return 0; +} + +/* Return the argument type of a given operand. */ + +static argtype +getarg_type (operand_type op) +{ + if (op < MAX_OPRD) + return crx_optab[op].arg_type; + else + return nullargs; +} + +/* Get the core processor register 'reg_name'. */ + +static reg +get_register (char *reg_name) +{ + const reg_entry *reg; + + reg = (const reg_entry *) hash_find (reg_hash, reg_name); + + if (reg != NULL) + return reg->value.reg_val; + else + return nullregister; +} + +/* Get the coprocessor register 'copreg_name'. */ + +static copreg +get_copregister (char *copreg_name) +{ + const reg_entry *copreg; + + copreg = (const reg_entry *) hash_find (copreg_hash, copreg_name); + + if (copreg != NULL) + return copreg->value.copreg_val; + else + return nullcopregister; +} + +/* Mask a constant to the number of bits it is to be mapped to. */ + +static void +mask_const (unsigned long int *t, int size) +{ + *t &= (((LONGLONG)1 << size) - 1); +} + +/* Round up a section size to the appropriate boundary. */ + +valueT +md_section_align (segT seg, valueT val) +{ + /* Round .text section to a multiple of 2. */ + if (seg == text_section) + return (val + 1) & ~1; + return val; +} + +/* Parse an operand that is machine-specific (remove '*'). */ + +void +md_operand (expressionS * exp) +{ + char c = *input_line_pointer; + + switch (c) + { + case '*': + input_line_pointer++; + expression (exp); + break; + default: + break; + } +} + +/* Reset global variables before parsing a new instruction. */ + +static void +reset_vars (char *op, ins *crx_ins) +{ + unsigned int i; + + processing_arg_number = relocatable = size_was_set + = signflag = post_inc_mode = cst4flag = 0; + memset (&output_opcode, '\0', sizeof (output_opcode)); + + /* Memset the 'signflag' field in every argument. */ + for (i = 0; i < MAX_OPERANDS; i++) + crx_ins->arg[i].signflag = 0; + + /* Save a copy of the original OP (used in error messages). */ + strcpy (ins_parse, op); +} + +/* Generate a relocation entry for a fixup. */ + +arelent * +tc_gen_reloc (asection *section ATTRIBUTE_UNUSED, fixS * fixP) +{ + arelent * reloc; + + reloc = xmalloc (sizeof (arelent)); + reloc->sym_ptr_ptr = xmalloc (sizeof (asymbol *)); + *reloc->sym_ptr_ptr = symbol_get_bfdsym (fixP->fx_addsy); + reloc->address = fixP->fx_frag->fr_address + fixP->fx_where; + reloc->addend = fixP->fx_offset; + + if (fixP->fx_subsy != NULL) + /* We don't resolve difference expressions. */ + as_bad_where (fixP->fx_file, fixP->fx_line, + _("can't resolve `%s' {%s section} - `%s' {%s section}"), + fixP->fx_addsy ? S_GET_NAME (fixP->fx_addsy) : "0", + segment_name (fixP->fx_addsy + ? S_GET_SEGMENT (fixP->fx_addsy) + : absolute_section), + S_GET_NAME (fixP->fx_subsy), + segment_name (S_GET_SEGMENT (fixP->fx_addsy))); + + assert ((int) fixP->fx_r_type > 0); + 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, + _("internal error: reloc %d (`%s') not supported by object file format"), + fixP->fx_r_type, + bfd_get_reloc_code_name (fixP->fx_r_type)); + return NULL; + } + assert (!fixP->fx_pcrel == !reloc->howto->pc_relative); + + return reloc; +} + +/* Prepare machine-dependent frags for relaxation. */ + +int +md_estimate_size_before_relax (fragS *fragp, asection *seg) +{ + /* If symbol is undefined or located in a different section, + select the largest supported relocation. */ + relax_substateT subtype; + relax_substateT rlx_state[] = {0, 2, + 3, 4, + 5, 6}; + + for (subtype = 0; subtype < ARRAY_SIZE (rlx_state); subtype += 2) + { + if (fragp->fr_subtype == rlx_state[subtype] + && (!S_IS_DEFINED (fragp->fr_symbol) + || seg != S_GET_SEGMENT (fragp->fr_symbol))) + { + fragp->fr_subtype = rlx_state[subtype + 1]; + break; + } + } + + if (fragp->fr_subtype >= ARRAY_SIZE (md_relax_table)) + abort (); + + return md_relax_table[fragp->fr_subtype].rlx_length; +} + +void +md_convert_frag (bfd *abfd ATTRIBUTE_UNUSED, asection *sec, fragS *fragP) +{ + /* 'opcode' points to the start of the instruction, whether + we need to change the instruction's fixed encoding. */ + char *opcode = fragP->fr_literal + fragP->fr_fix; + bfd_reloc_code_real_type reloc; + + subseg_change (sec, 0); + + switch (fragP->fr_subtype) + { + case 0: + reloc = BFD_RELOC_CRX_REL8; + break; + case 1: + *opcode = 0x7e; + reloc = BFD_RELOC_CRX_REL16; + break; + case 2: + *opcode = 0x7f; + reloc = BFD_RELOC_CRX_REL32; + break; + case 3: + reloc = BFD_RELOC_CRX_REL16; + break; + case 4: + *++opcode = 0x31; + reloc = BFD_RELOC_CRX_REL32; + break; + case 5: + reloc = BFD_RELOC_CRX_REL8_CMP; + break; + case 6: + *++opcode = 0x31; + reloc = BFD_RELOC_CRX_REL24; + break; + default: + abort (); + break; + } + + fix_new (fragP, fragP->fr_fix, + bfd_get_reloc_size (bfd_reloc_type_lookup (stdoutput, reloc)), + fragP->fr_symbol, fragP->fr_offset, 1, reloc); + fragP->fr_var = 0; + fragP->fr_fix += md_relax_table[fragP->fr_subtype].rlx_length; +} + +/* Process machine-dependent command line options. Called once for + each option on the command line that the machine-independent part of + GAS does not understand. */ + +int +md_parse_option (int c ATTRIBUTE_UNUSED, char *arg ATTRIBUTE_UNUSED) +{ + return 0; +} + +/* Machine-dependent usage-output. */ + +void +md_show_usage (FILE *stream ATTRIBUTE_UNUSED) +{ + return; +} + +/* Turn a string in input_line_pointer into a floating point constant + of type TYPE, and store the appropriate bytes in *LITP. The number + of LITTLENUMS emitted is stored in *SIZEP. An error message is + returned, or NULL on OK. */ + +char * +md_atof (int type, char *litP, int *sizeP) +{ + int prec; + LITTLENUM_TYPE words[4]; + char *t; + int i; + + switch (type) + { + case 'f': + prec = 2; + break; + + case 'd': + prec = 4; + break; + + default: + *sizeP = 0; + return _("bad call to md_atof"); + } + + t = atof_ieee (input_line_pointer, type, words); + if (t) + input_line_pointer = t; + + *sizeP = prec * 2; + + if (! target_big_endian) + { + for (i = prec - 1; i >= 0; i--) + { + md_number_to_chars (litP, (valueT) words[i], 2); + litP += 2; + } + } + else + { + for (i = 0; i < prec; i++) + { + md_number_to_chars (litP, (valueT) words[i], 2); + litP += 2; + } + } + + return NULL; +} + +/* Apply a fixS (fixup of an instruction or data that we didn't have + enough info to complete immediately) to the data in a frag. + Since linkrelax is nonzero and TC_LINKRELAX_FIXUP is defined to disable + relaxation of debug sections, this function is called only when + fixuping relocations of debug sections. */ + +void +md_apply_fix3 (fixS *fixP, valueT *valP, segT seg) +{ + valueT val = * valP; + char *buf = fixP->fx_frag->fr_literal + fixP->fx_where; + fixP->fx_offset = 0; + + switch (fixP->fx_r_type) + { + case BFD_RELOC_CRX_NUM8: + bfd_put_8 (stdoutput, (unsigned char) val, buf); + break; + case BFD_RELOC_CRX_NUM16: + bfd_put_16 (stdoutput, val, buf); + break; + case BFD_RELOC_CRX_NUM32: + bfd_put_32 (stdoutput, val, buf); + break; + default: + /* We shouldn't ever get here because linkrelax is nonzero. */ + abort (); + break; + } + + fixP->fx_done = 0; + + if (fixP->fx_addsy == NULL + && fixP->fx_pcrel == 0) + fixP->fx_done = 1; + + if (fixP->fx_pcrel == 1 + && fixP->fx_addsy != NULL + && S_GET_SEGMENT (fixP->fx_addsy) == seg) + fixP->fx_done = 1; +} + +/* The location from which a PC relative jump should be calculated, + given a PC relative reloc. */ + +long +md_pcrel_from (fixS *fixp) +{ + return fixp->fx_frag->fr_address + fixp->fx_where; +} + +/* This function is called once, at assembler startup time. This should + set up all the tables, etc that the MD part of the assembler needs. */ + +void +md_begin (void) +{ + const char *hashret = NULL; + int i = 0; + + /* Set up a hash table for the instructions. */ + crx_inst_hash = hash_new (); + if (crx_inst_hash == NULL) + as_fatal (_("Virtual memory exhausted")); + + while (crx_instruction[i].mnemonic != NULL) + { + const char *mnemonic = crx_instruction[i].mnemonic; + + hashret = hash_insert (crx_inst_hash, mnemonic, + (PTR) &crx_instruction[i]); + + if (hashret != NULL && *hashret != '\0') + as_fatal (_("Can't hash `%s': %s\n"), crx_instruction[i].mnemonic, + *hashret == 0 ? _("(unknown reason)") : hashret); + + /* Insert unique names into hash table. The CRX instruction set + has many identical opcode names that have different opcodes based + on the operands. This hash table then provides a quick index to + the first opcode with a particular name in the opcode table. */ + do + { + ++i; + } + while (crx_instruction[i].mnemonic != NULL + && streq (crx_instruction[i].mnemonic, mnemonic)); + } + + /* Initialize reg_hash hash table. */ + reg_hash = hash_new (); + + { + const reg_entry *regtab; + + for (regtab = crx_regtab; + regtab < (crx_regtab + NUMREGS); regtab++) + { + hashret = hash_insert (reg_hash, regtab->name, (PTR) regtab); + if (hashret) + as_fatal (_("Internal Error: Can't hash %s: %s"), + regtab->name, + hashret); + } + } + + /* Initialize copreg_hash hash table. */ + copreg_hash = hash_new (); + + { + const reg_entry *copregtab; + + for (copregtab = crx_copregtab; copregtab < (crx_copregtab + NUMCOPREGS); + copregtab++) + { + hashret = hash_insert (copreg_hash, copregtab->name, (PTR) copregtab); + if (hashret) + as_fatal (_("Internal Error: Can't hash %s: %s"), + copregtab->name, + hashret); + } + } + /* Set linkrelax here to avoid fixups in most sections. */ + linkrelax = 1; +} + +/* Get the number of bits corresponding to a constant - + here we check for possible overflow cases. */ + +static void +get_number_of_bits (ins * crx_ins, int op_num) +{ + int cnt_bits = 0; + unsigned long int temp = crx_ins->arg[op_num].constant; + const cst4_entry *cst4_op; + + /* If the constant's size was already set - nothing to do. */ + if (size_was_set) + return; + + /* Already dealt with negative numbers in process_label_constants. */ + while (temp > 0) + { + temp >>= 1; + cnt_bits++; + } + + if (IS_INSN_TYPE (ARITH_INS) && !relocatable && !signflag) + { + if (cnt_bits == 16) + { + crx_ins->arg[op_num].size = 17; + return; + } + } + /* If a signed +ve is represented in 6 bits then we have to represent + it in 22 bits in case of the index mode of addressing. */ + if (IS_INSN_TYPE (LD_STOR_INS) + || IS_INSN_TYPE (LD_STOR_INS_INC) + || IS_INSN_TYPE (STOR_IMM_INS) + || IS_INSN_TYPE (CSTBIT_INS)) + { + if (!signflag && crx_ins->arg[op_num].type == arg_icr) + { + if (cnt_bits == 6) + { + crx_ins->arg[op_num].size = 7; + return; + } + if (cnt_bits == 22) + as_bad (_("Offset out of range in Instruction `%s'"), ins_parse); + } + } + /* If a signed +ve is represnted in 16 bits in case of load/stor disp16 + then change it to 17 bits. + If a signed +ve is represnted in 12 bits in post increment instruction + increase it to 13 bits. */ + if (IS_INSN_TYPE (LD_STOR_INS)) + { + if (!signflag && crx_ins->arg[op_num].type == arg_cr) + { + if (cnt_bits == 16) + { + crx_ins->arg[op_num].size = 17; + return; + } + if (cnt_bits == 32) + as_bad (_("Offset out of range in Instruction `%s'"), ins_parse); + } + } + + if (IS_INSN_TYPE (CSTBIT_INS) + || IS_INSN_TYPE (LD_STOR_INS_INC) + || IS_INSN_TYPE (STOR_IMM_INS)) + { + if (!signflag && crx_ins->arg[op_num].type == arg_cr) + { + if (cnt_bits == 12) + { + crx_ins->arg[op_num].size = 13; + if (IS_INSN_TYPE (LD_STOR_INS_INC)) + as_bad (_("Offset out of range in Instruction `%s'"), ins_parse); + return; + } + if (IS_INSN_TYPE (CSTBIT_INS) || IS_INSN_TYPE (STOR_IMM_INS)) + { + if (cnt_bits == 28) + as_bad (_("Offset out of range in Instruction `%s'"), ins_parse); + } + + } + } + + /* Handle negative cst4 mapping for arithmetic/cmp&br operations. */ + if (signflag && !relocatable + && ((IS_INSN_TYPE (ARITH_INS) || IS_INSN_TYPE (ARITH_BYTE_INS)) + || ((IS_INSN_TYPE (CMPBR_INS) && op_num == 0)))) + { + for (cst4_op = cst4_map; cst4_op < (cst4_map + cst4_maps); cst4_op++) + { + if (crx_ins->arg[op_num].constant == (unsigned int)(-cst4_op->value)) + { + crx_ins->arg[op_num].size = 4; + crx_ins->arg[op_num].constant = cst4_op->binary; + crx_ins->arg[op_num].signflag = 0; + return; + } + } + } + /* Because of the cst4 mapping -- -1 and -4 already handled above + as well as for relocatable cases. */ + if (signflag && IS_INSN_TYPE (ARITH_BYTE_INS)) + { + if (!relocatable) + { + if (crx_ins->arg[op_num].constant <= 0xffff) + crx_ins->arg[op_num].size = 16; + else + /* Setting to 18 so that there is no match. */ + crx_ins->arg[op_num].size = 18; + } + else + crx_ins->arg[op_num].size = 16; + return; + } + + if (signflag && IS_INSN_TYPE (ARITH_INS)) + { + /* For all immediates which can be expressed in less than 16 bits. */ + if (crx_ins->arg[op_num].constant <= 0xffff && !relocatable) + { + crx_ins->arg[op_num].size = 16; + return; + } + /* Either it is relocatable or not representable in 16 bits. */ + if (crx_ins->arg[op_num].constant < 0xffffffff || relocatable) + { + crx_ins->arg[op_num].size = 32; + return; + } + crx_ins->arg[op_num].size = 33; + return; + } + if (signflag && !relocatable) + return; + + if (!relocatable) + crx_ins->arg[op_num].size = cnt_bits; + + /* Checking for Error Conditions. */ + if (IS_INSN_TYPE (ARITH_INS) && !signflag) + { + if (cnt_bits > 32) + as_bad (_("Cannot represent Immediate in %d bits in Instruction `%s'"), + cnt_bits, ins_parse); + } + else if (IS_INSN_TYPE (ARITH_BYTE_INS) && !signflag) + { + if (cnt_bits > 16) + as_bad (_("Cannot represent Immediate in %d bits in Instruction `%s'"), + cnt_bits, ins_parse); + } +} + +/* Handle the constants -immediate/absolute values and + Labels (jump targets/Memory locations). */ + +static int +process_label_constant (char *str, ins * crx_ins, int number) +{ + char *save; + unsigned long int temp, cnt; + const cst4_entry *cst4_op; + int is_cst4=0; + int constant_val = 0; + int cmp_br_type_flag = 0, i; + int br_type_flag = 0; + save = input_line_pointer; + signflag = 0; + + if (str[0] == '-') + { + signflag = 1; + str++; + } + else if (str[0] == '+') + str++; + + /* Preprocessing for cmpbr instruction and getting the size flag. */ + if (strstr (str, ":s") != NULL && (IS_INSN_TYPE (CMPBR_INS) + || IS_INSN_TYPE (COP_BRANCH_INS))) + cmp_br_type_flag = 8; + + if (strstr (str, ":l") != NULL && (IS_INSN_TYPE (CMPBR_INS) + || IS_INSN_TYPE (COP_BRANCH_INS))) + cmp_br_type_flag = 24; + + /* Branch instruction preprocessing. */ + if (IS_INSN_TYPE (BRANCH_INS)) + { + if (strstr (str, ":s") != NULL) + br_type_flag = 8; + else if (strstr (str, ":m") != NULL) + br_type_flag = 16; + else if (strstr (str, ":l") != NULL) + br_type_flag = 32; + } + /* Making the label cleared for processing removing :lms etc from labels. */ + if (cmp_br_type_flag != 0 || br_type_flag != 0) + { + i = 0; + while (str[i] != ':') + { + i++; + } + str[i] = '\0'; + } + input_line_pointer = str; + + expression (&crx_ins->exp); + + switch (crx_ins->exp.X_op) + { + case O_constant: + crx_ins->arg[number].constant = crx_ins->exp.X_add_number; + constant_val = crx_ins->exp.X_add_number; + if ((IS_INSN_TYPE (CMPBR_INS) || IS_INSN_TYPE (COP_BRANCH_INS)) + && number == 2) + { + /* This variable causes a warning (is to be handles by string + type implementation). */ + LONGLONG temp64 = 0; + + char ptr[20]; + char temp_str[30]; + unsigned int jump_value = 0; + int BR_MASK = 0, BR_SIZE = 0; + temp_str[0] = '\0'; + if (signflag) + { + temp_str[0] = '-'; + temp_str[1] = '\0'; + } + strncat (temp_str, str, strlen (str)); + temp64 = strtol (temp_str, (char **) &ptr,0); + /* This is not accurate : + Actually overflow is allowed here (see comment below). + Originally the call was to 'strtoll', which isn't + identified by MSVC. */ + if ((temp64 == LONG_MAX) || (temp64 == LONG_MIN)) + as_bad (_("Overflow in displacement in Instruction `%s'"), + ins_parse); + + /* If br *+x + It will be returned as '0' padded with 'x' uptill 64 bits + If br *-x + It will be returned as sign extended form + + Then search for validity of representation + Check whether upper 38 bits are all zeroes or all ones + If not report error. */ + if (!(((temp64 & UPPER31_MASK) == UPPER31_MASK) + || ((temp64 & UPPER31_MASK) == 0x0))) + as_bad (_("Overflow in displacement in Instruction `%s'"), + ins_parse); + + if (temp64 % 2 != 0) + as_bad (_("Odd Offset in displacement in Instruction `%s'"), + ins_parse); + + /* Determine the branch size. */ + jump_value = (unsigned int)temp64 & 0xFFFFFFFF; + if (((jump_value & 0xFFFFFF00) == 0xFFFFFF00) + || ((jump_value & 0xFFFFFF00) == 0x0)) + { + BR_MASK = 0xFF; + BR_SIZE = 8; + } + else + if (((jump_value & 0xFF000000) == 0xFF000000) + || ((jump_value & 0xFF000000) == 0x0)) + { + BR_MASK = 0xFFFFFF; + BR_SIZE = 24; + } + jump_value = jump_value >> 1; + crx_ins->arg[number].constant = jump_value & BR_MASK; + crx_ins->arg[number].size = BR_SIZE; + size_was_set = 1; + crx_ins->arg[number].signflag = signflag; + input_line_pointer = save; + return crx_ins->exp.X_op; + } + + if (IS_INSN_TYPE (BRANCH_INS) + || IS_INSN_MNEMONIC ("bal") + || IS_INSN_TYPE (DCR_BRANCH_INS)) + { + LONGLONG temp64 = 0; + char ptr[20]; + char temp_str[30]; + unsigned int jump_value = 0; + int BR_MASK = 0, BR_SIZE = 0; + + temp_str[0] = '\0'; + if (signflag) + { + temp_str[0] = '-'; + temp_str[1] = '\0'; + } + strncat (temp_str, str, strlen (str)); + temp64 = strtol (temp_str, (char **) &ptr,0); + /* This is not accurate : + Actually overflow is allowed here (see comment below). + Originally the call was to 'strtoll', which isn't + identified by MSVC. */ + if ((temp64 == LONG_MAX) || (temp64 == LONG_MIN)) + as_bad (_("Overflow in displacement in Instruction `%s'"), + ins_parse); + + /* If br *+x + It will be returned as '0' padded with 'x' uptill 64 bits + If br *-x + It will be returned as sign extended form + + Then search for validity of representation + Check whether upper 31 bits are all zeroes or all ones + If not report error. */ + if (!(((temp64 & UPPER31_MASK) == UPPER31_MASK) + || ((temp64 & UPPER31_MASK) == 0x0))) + as_bad (_("Overflow in displacement in Instruction `%s'"), + ins_parse); + + if (temp64 % 2 != 0) + as_bad (_("Odd Offset in displacement in Instruction `%s'"), + ins_parse); + + /* Determine the branch size. */ + jump_value = (unsigned int)temp64 & 0xFFFFFFFF; + if (!IS_INSN_MNEMONIC ("bal") && !IS_INSN_TYPE (DCR_BRANCH_INS) + && (((jump_value & 0xFFFFFF00) == 0xFFFFFF00) + || ((jump_value & 0xFFFFFF00) == 0x0))) + { + BR_MASK = 0xFF; + BR_SIZE = 8; + } + else + if (((jump_value & 0xFFFF0000) == 0xFFFF0000) + || ((jump_value & 0xFFFF0000) == 0x0)) + { + BR_MASK = 0xFFFF; + BR_SIZE = 16; + } + else + { + BR_MASK = 0xFFFFFFFF; + BR_SIZE = 32; + } + jump_value = jump_value >> 1; + crx_ins->arg[number].constant = jump_value & BR_MASK; + crx_ins->arg[number].size = BR_SIZE; + size_was_set = 1; + crx_ins->arg[number].signflag = signflag; + input_line_pointer = save; + return crx_ins->exp.X_op; + } + /* Fix for movd $0xF12344, r0 -- signflag has to be set. */ + if (constant_val < 0 && signflag != 1 + && !IS_INSN_TYPE (LD_STOR_INS) && !IS_INSN_TYPE (LD_STOR_INS_INC) + && !IS_INSN_TYPE (CSTBIT_INS) && !IS_INSN_TYPE (STOR_IMM_INS) + && !IS_INSN_TYPE (BRANCH_INS) && !IS_INSN_MNEMONIC ("bal")) + { + crx_ins->arg[number].constant = + ~(crx_ins->arg[number].constant) + 1; + signflag = 1; + } + /* For load/store instruction when the value is in the offset part. */ + if (constant_val < 0 && signflag != 1 + && (IS_INSN_TYPE (LD_STOR_INS) || IS_INSN_TYPE (LD_STOR_INS_INC) + || IS_INSN_TYPE (CSTBIT_INS) || IS_INSN_TYPE (STOR_IMM_INS))) + { + if (crx_ins->arg[number].type == arg_cr + || crx_ins->arg[number].type == arg_icr) + { + crx_ins->arg[number].constant = + ~(crx_ins->arg[number].constant) + 1; + signflag = 1; + } + } + if (signflag) + { + /* Signflag in never set in case of load store instructions + Mapping in case of only the arithinsn case. */ + if ((crx_ins->arg[number].constant != 1 + && crx_ins->arg[number].constant != 4) + || (!IS_INSN_TYPE (ARITH_INS) + && !IS_INSN_TYPE (ARITH_BYTE_INS) + && !IS_INSN_TYPE (CMPBR_INS))) + { + /* Counting the number of bits required to represent + the constant. */ + cnt = 0; + temp = crx_ins->arg[number].constant - 1; + while (temp > 0) + { + temp >>= 1; + cnt++; + } + crx_ins->arg[number].size = cnt + 1; + crx_ins->arg[number].constant = + ~(crx_ins->arg[number].constant) + 1; + if (IS_INSN_TYPE (ARITH_INS) || IS_INSN_TYPE (ARITH_BYTE_INS)) + { + char ptr[30]; + LONGLONG temp64; + + /* Tomer - Originally the call was to 'strtoull', which isn't + identified by MSVC. Instead we check for overflow. */ + temp64 = strtoul (str, (char **) &ptr, 0); + if (cnt < 4) + crx_ins->arg[number].size = 5; + + if (IS_INSN_TYPE (ARITH_INS)) + { + if (crx_ins->arg[number].size > 32 + /* Tomer - check for overflow. */ + || (temp64 == ULONG_MAX)) + { + if (crx_ins->arg[number].size > 32) + as_bad (_("In Instruction `%s': Immediate size is \ + %lu bits cannot be accomodated"), + ins_parse, cnt + 1); + + /* Tomer - check for overflow. */ + if (temp64 == ULONG_MAX) + as_bad (_("Value given more than 32 bits in \ + Instruction `%s'"), ins_parse); + } + } + if (IS_INSN_TYPE (ARITH_BYTE_INS)) + { + if (crx_ins->arg[number].size > 16 + || !((temp64 & 0xFFFF0000) == 0xFFFF0000 + || (temp64 & 0xFFFF0000) == 0x0)) + { + if (crx_ins->arg[number].size > 16) + as_bad (_("In Instruction `%s': Immediate size is \ + %lu bits cannot be accomodated"), + ins_parse, cnt + 1); + + if (!((temp64 & 0xFFFF0000) == 0xFFFF0000 + || (temp64 & 0xFFFF0000) == 0x0)) + as_bad (_("Value given more than 16 bits in \ + Instruction `%s'"), ins_parse); + } + } + } + if (IS_INSN_TYPE (LD_STOR_INS) && crx_ins->arg[number].type == arg_cr + && !post_inc_mode) + { + /* Cases handled --- + dispub4/dispuw4/dispud4 and for load store dispubwd4 + is applicable only. */ + if (crx_ins->arg[number].size <= 4) + crx_ins->arg[number].size = 5; + } + /* Argument number is checked to distinguish between + immediate and displacement in cmpbranch and bcopcond. */ + if ((IS_INSN_TYPE (CMPBR_INS) || IS_INSN_TYPE (COP_BRANCH_INS)) + && number == 2) + { + if (crx_ins->arg[number].size != 32) + crx_ins->arg[number].constant = + crx_ins->arg[number].constant >> 1; + } + + mask_const (&crx_ins->arg[number].constant, + (int) crx_ins->arg[number].size); + } + } + else + { + /* Argument number is checked to distinguish between + immediate and displacement in cmpbranch and bcopcond. */ + if (((IS_INSN_TYPE (CMPBR_INS) || IS_INSN_TYPE (COP_BRANCH_INS)) + && number == 2) + || IS_INSN_TYPE (BRANCH_NEQ_INS)) + { + if (IS_INSN_TYPE (BRANCH_NEQ_INS)) + { + if (crx_ins->arg[number].constant == 0) + as_bad (_("Instruction `%s' has Zero offset"), ins_parse); + } + + if (crx_ins->arg[number].constant % 2 != 0) + as_bad (_("Instruction `%s' has odd offset"), ins_parse); + + if (IS_INSN_TYPE (BRANCH_NEQ_INS)) + { + if (crx_ins->arg[number].constant > 32 + || crx_ins->arg[number].constant < 2) + as_bad (_("Instruction `%s' has illegal offset (%ld)"), + ins_parse, crx_ins->arg[number].constant); + + crx_ins->arg[number].constant -= 2; + } + + crx_ins->arg[number].constant = + crx_ins->arg[number].constant >> 1; + get_number_of_bits (crx_ins, number); + } + + /* Compare branch argument number zero to be compared - + mapped to cst4. */ + if (IS_INSN_TYPE (CMPBR_INS) && number == 0) + { + for (cst4_op = cst4_map; cst4_op < (cst4_map + cst4_maps); cst4_op++) + { + if (crx_ins->arg[number].constant == (unsigned int)cst4_op->value) + { + crx_ins->arg[number].constant = cst4_op->binary; + is_cst4 = 1; + break; + } + } + if (!is_cst4) + as_bad (_("Instruction `%s' has invalid imm value as an \ + operand"), ins_parse); + } + } + break; + + case O_symbol: + case O_subtract: + crx_ins->arg[number].constant = 0; + relocatable = 1; + + switch (crx_ins->arg[number].type) + { + case arg_cr: + /* Have to consider various cases here --load/stor++[bwd] rbase, reg. */ + if (IS_INSN_TYPE (LD_STOR_INS_INC)) + crx_ins->rtype = BFD_RELOC_CRX_REGREL12; + else if (IS_INSN_TYPE (CSTBIT_INS) + || IS_INSN_TYPE (STOR_IMM_INS)) + /* 'stor[bwd] imm' and '[stc]bit[bwd]'. */ + crx_ins->rtype = BFD_RELOC_CRX_REGREL28; + else + /* General load store instruction. */ + crx_ins->rtype = BFD_RELOC_CRX_REGREL32; + break; + case arg_icr: + /* Index Mode 22 bits relocation. */ + crx_ins->rtype = BFD_RELOC_CRX_REGREL22; + break; + case arg_c: + /* Absolute types. */ + /* Case for jumps...dx types. */ + /* For bal. */ + if (IS_INSN_MNEMONIC ("bal") || IS_INSN_TYPE (DCR_BRANCH_INS)) + crx_ins->rtype = BFD_RELOC_CRX_REL16; + else if (IS_INSN_TYPE (BRANCH_INS)) + { + crx_ins->rtype = BFD_RELOC_CRX_REL8; + + /* Overriding the above by the br_type_flag set above. */ + switch (br_type_flag) + { + default: + break; + case 8: + crx_ins->rtype = BFD_RELOC_CRX_REL8; + break; + case 16: + crx_ins->rtype = BFD_RELOC_CRX_REL16; + break; + case 32: + crx_ins->rtype = BFD_RELOC_CRX_REL32; + break; + } + } + else if (IS_INSN_TYPE (LD_STOR_INS) || IS_INSN_TYPE (STOR_IMM_INS) + || IS_INSN_TYPE (CSTBIT_INS)) + crx_ins->rtype = BFD_RELOC_CRX_ABS32; + else if (IS_INSN_TYPE (BRANCH_NEQ_INS)) + crx_ins->rtype = BFD_RELOC_CRX_REL4; + else if (IS_INSN_TYPE (CMPBR_INS) || IS_INSN_TYPE (COP_BRANCH_INS)) + { + if (cmp_br_type_flag == 24) + crx_ins->rtype = BFD_RELOC_CRX_REL24; + else + crx_ins->rtype = BFD_RELOC_CRX_REL8_CMP; + } + break; + case arg_ic: + case arg_dc: + if (IS_INSN_TYPE (ARITH_INS)) + crx_ins->rtype = BFD_RELOC_CRX_IMM32; + else if (IS_INSN_TYPE (ARITH_BYTE_INS)) + crx_ins->rtype = BFD_RELOC_CRX_IMM16; + break; + default: + break; + } + crx_ins->arg[number].size = (bfd_reloc_type_lookup (stdoutput, crx_ins->rtype))->bitsize; + break; + + default: + break; + } + + input_line_pointer = save; + crx_ins->arg[number].signflag = signflag; + return crx_ins->exp.X_op; +} + +/* Get the values of the scale to be encoded - + used for the scaled index mode of addressing. */ + +static int +exponent2scale (int val) +{ + int exponent; + + /* If 'val' is 0, the following 'for' will be an endless loop. */ + if (val == 0) + return 0; + + for (exponent = 0; (val != 1); val >>= 1, exponent++) + ; + + return exponent; +} + +/* This is used to set the index mode parameters. Used to set the attributes of + an indexmode type of operand. op_num is the operand number. */ + +static void +set_indexmode_parameters (char *operand, ins * crx_ins, int op_num) +{ + char address_str[30]; + char scale_str[MAX_OPERANDS]; + int scale_cnt = 0; + char reg_name[MAX_REGNAME_LEN]; + char regindex_name[MAX_REGNAME_LEN]; + int i = 0; + int reg_counter = 0, addr_cnt = 0, temp_int_val = 0; + + switch (crx_ins->arg[op_num].type) + { + case arg_icr: + while (operand[i] != '(') + { + address_str[addr_cnt++] = operand[i]; + i++; + } + address_str[addr_cnt] = '\0'; + process_label_constant (address_str, crx_ins, op_num); + i++; + reg_counter = 0; + while (operand[i] != ',' && operand[i] != ' ') + { + reg_name[reg_counter++] = operand[i]; + i++; + } + reg_name[reg_counter] = '\0'; + if ((crx_ins->arg[op_num].r = get_register (reg_name)) == nullregister) + as_bad (_("Illegal register `%s' in Instruction `%s'"), + reg_name, ins_parse); + + i++; + while (operand[i] == ' ') + i++; + + reg_counter = 0; + while (operand[i] != ')' && operand[i] != ',') + { + regindex_name[reg_counter++] = operand[i]; + i++; + } + regindex_name[reg_counter] = '\0'; + reg_counter = 0; + if ((crx_ins->arg[op_num].i_r = get_register (regindex_name)) + == nullregister) + as_bad (_("Illegal register `%s' in Instruction `%s'"), + regindex_name, ins_parse); + + /* Setting the scale parameters. */ + while (operand[i] == ' ') + i++; + + if (operand[i] == ')') + crx_ins->arg[op_num].scale = 0; + else + { + if (operand[i] == ',') + i++; + + while (operand[i] != ' ' && operand[i] != ')') + { + scale_str[scale_cnt++] = operand[i]; + i++; + } + + scale_str[scale_cnt] = '\0'; + /* Preprocess the scale string. */ + if (strstr (scale_str, "0x") != NULL + || strstr (scale_str, "0X") != NULL) + { + sscanf (scale_str, "%x", &temp_int_val); + memset (&scale_str, '\0', sizeof (scale_str)); + sprintf (scale_str, "%d", temp_int_val); + } + /* Preprocess over. */ + temp_int_val = atoi (scale_str); + + if (temp_int_val != 1 && temp_int_val != 2 + && temp_int_val != 4 && temp_int_val != 8) + as_bad (_("Illegal Scale - `%s'"), scale_str); + + crx_ins->arg[op_num].scale = exponent2scale (temp_int_val); + } + break; + default: + break; + } +} + +/* Parsing the operands of types + - constants + - rbase -> (register) + - offset(rbase) + - offset(rbase)+ - post increment mode. */ + +static void +set_cons_rparams (char *operand, ins * crx_ins, int op_num) +{ + int i = 0, reg_count = 0; + char reg_name[MAX_REGNAME_LEN]; + int change_flag = 0; + + if (crx_ins->arg[op_num].type == arg_dc) + change_flag = 1; + + switch (crx_ins->arg[op_num].type) + { + case arg_sc: /* Case *+347. */ + case arg_dc: /* Case $18. */ + i++; + case arg_c:/* Case where its a simple constant. */ + process_label_constant (operand + i, crx_ins, op_num); + crx_ins->arg[op_num].type = arg_c; + break; + case arg_dcr: /* Case $9(r13). */ + operand++; + case arg_cr: /* Case 9(r13. */ + while (operand[i] != '(') + i++; + operand[i] = '\0'; + process_label_constant (operand, crx_ins, op_num); + operand[i] = '('; + i++; + reg_count = 0; + while (operand[i] != ')') + { + reg_name[reg_count] = operand[i]; + i++; + reg_count++; + } + reg_name[reg_count] = '\0'; + if ((crx_ins->arg[op_num].r = get_register (reg_name)) == nullregister) + as_bad (_("Illegal register `%s' in Instruction `%s'"), + reg_name, ins_parse); + + crx_ins->arg[op_num].type = arg_cr; + /* Post increment is represented in assembly as offset (register)+. */ + if (strstr (operand + i, "+") != NULL) + /* There is a plus after the ')'. */ + post_inc_mode = 1; + break; + default: + break; + } + if (change_flag == 1) + crx_ins->arg[op_num].type = arg_ic; +} + +/* This is used to get the operand attributes - + operand - current operand to be used + number - operand number + crx_ins - current assembled instruction. */ + +static void +get_operandtype (char *operand, int number, ins * crx_ins) +{ + int ret_val; + char temp_operand[30]; + + switch (operand[0]) + { + /* When it is a register. */ + case 'r': + case 'c': + case 'i': + case 'u': + case 's': + case 'p': + case 'l': + case 'h': + /* Check whether this is a general processor register. */ + ret_val = get_register (operand); + if (ret_val != nullregister) + { + crx_ins->arg[number].type = arg_r; + crx_ins->arg[number].r = ret_val; + crx_ins->arg[number].size = REG_SIZE; + } + else + { + /* Check whether this is a core [special] coprocessor register. */ + ret_val = get_copregister (operand); + if (ret_val != nullcopregister) + { + crx_ins->arg[number].type = arg_copr; + if (ret_val >= cs0) + crx_ins->arg[number].type = arg_copsr; + crx_ins->arg[number].cr = ret_val; + crx_ins->arg[number].size = REG_SIZE; + } + else + { + if (strchr (operand, '(') != NULL) + { + if (strchr (operand, ',') != NULL + && (strchr (operand, ',') > strchr (operand, '('))) + { + crx_ins->arg[number].type = arg_icr; + crx_ins->arg[number].constant = 0; + set_indexmode_parameters (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + return; + } + else + crx_ins->arg[number].type = arg_cr; + } + else + crx_ins->arg[number].type = arg_c; + crx_ins->arg[number].constant = 0; + set_cons_rparams (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + } + } + break; + case '$': + if (strchr (operand, '(') != NULL) + crx_ins->arg[number].type = arg_dcr; + else + crx_ins->arg[number].type = arg_dc; + crx_ins->arg[number].constant = 0; + set_cons_rparams (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + break; + + case '(': + /* Augmenting a zero in front of an operand -- won't work for tbit/sbit. */ + strcpy (temp_operand, "0"); + strcat (temp_operand, operand); + if (strchr (temp_operand, ',') != NULL + && (strchr (temp_operand, ',') > strchr (temp_operand, '('))) + { + crx_ins->arg[number].type = arg_icr; + crx_ins->arg[number].constant = 0; + set_indexmode_parameters (temp_operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + return; + } + else + { + crx_ins->arg[number].type = arg_cr; + crx_ins->arg[number].constant = 0; + set_cons_rparams (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + if ((! strneq (instruction->mnemonic, "load", 4)) + && (! strneq (instruction->mnemonic, "stor", 4))) + { + crx_ins->arg[number].type = arg_rbase; + crx_ins->arg[number].size = REG_SIZE; + } + return; + } + break; + case '*': + crx_ins->arg[number].type = arg_sc; + crx_ins->arg[number].constant = 0; + set_cons_rparams (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + break; + case '+': + case '-': + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + if (strchr (operand, '(') != NULL) + { + if (strchr (operand, ',') != NULL + && (strchr (operand, ',') > strchr (operand, '('))) + { + crx_ins->arg[number].type = arg_icr; + crx_ins->arg[number].constant = 0; + set_indexmode_parameters (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + return; + } + else + crx_ins->arg[number].type = arg_cr; + } + else + crx_ins->arg[number].type = arg_c; + crx_ins->arg[number].constant = 0; + set_cons_rparams (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + break; + default: + if (strchr (operand, '(') != NULL) + { + if (strchr (operand, ',') != NULL + && (strchr (operand, ',') > strchr (operand, '('))) + { + crx_ins->arg[number].type = arg_icr; + crx_ins->arg[number].constant = 0; + set_indexmode_parameters (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + return; + } + else + crx_ins->arg[number].type = arg_cr; + } + else + crx_ins->arg[number].type = arg_c; + crx_ins->arg[number].constant = 0; + set_cons_rparams (operand, crx_ins, number); + get_number_of_bits (crx_ins, number); + break; + } +} + +/* Operands are parsed over here, separated into various operands. Each operand + is then analyzed to fillup the fields in the crx_ins data structure. */ + +static void +parse_operands (ins * crx_ins, char *operands) +{ + char *operandS; /* Operands string. */ + char *operandH, *operandT; /* Single operand head/tail pointers. */ + int allocated = 0; /* Indicates a new operands string was allocated. */ + char *operand[MAX_OPERANDS]; /* Separating the operands. */ + int op_num = 0; /* Current operand number we are parsing. */ + int bracket_flag = 0; /* Indicates a bracket '(' was found. */ + int sq_bracket_flag = 0; /* Indicates a square bracket '[' was found. */ + + /* Preprocess the list of registers, if necessary. */ + operandS = operandH = operandT = (INST_HAS_REG_LIST) ? + preprocess_reglist (operands, &allocated) : operands; + + while (*operandT != '\0') + { + if (*operandT == ',' && bracket_flag != 1 && sq_bracket_flag != 1) + { + *operandT++ = '\0'; + operand[op_num++] = strdup (operandH); + operandH = operandT; + continue; + } + + if (*operandT == ' ') + as_bad (_("Illegal operands (whitespace): `%s'"), ins_parse); + + if (*operandT == '(') + bracket_flag = 1; + else if (*operandT == '[') + sq_bracket_flag = 1; + + if (*operandT == ')') + { + if (bracket_flag) + bracket_flag = 0; + else + as_fatal (_("Missing matching brackets : `%s'"), ins_parse); + } + else if (*operandT == ']') + { + if (sq_bracket_flag) + sq_bracket_flag = 0; + else + as_fatal (_("Missing matching brackets : `%s'"), ins_parse); + } + + if (bracket_flag == 1 && *operandT == ')') + bracket_flag = 0; + else if (sq_bracket_flag == 1 && *operandT == ']') + sq_bracket_flag = 0; + + operandT++; + } + + /* Adding the last operand. */ + operand[op_num++] = strdup (operandH); + crx_ins->nargs = op_num; + + /* Verifying correct syntax of operands (all brackets should be closed). */ + if (bracket_flag || sq_bracket_flag) + as_fatal (_("Missing matching brackets : `%s'"), ins_parse); + + /* Now to recongnize the operand types. */ + for (op_num = 0; op_num < crx_ins->nargs; op_num++) + { + get_operandtype (operand[op_num], op_num, crx_ins); + free (operand[op_num]); + } + + if (allocated) + free (operandS); +} + +/* Get the trap index in dispatch table, given its name. + This routine is used by assembling the 'excp' instruction. */ + +static int +gettrap (char *s) +{ + const trap_entry *trap; + + for (trap = crx_traps; trap < (crx_traps + NUMTRAPS); trap++) + if (streq (trap->name, s)) + return trap->entry; + + as_bad (_("Unknown exception: `%s'"), s); + return 0; +} + +/* Post-Increment instructions are a sub-group within load/stor instruction + groups. Therefore, when parsing a Post-Increment insn, we have to advance + the instruction pointer to the start of that sub-group. */ + +static void +handle_pi_insn (char *operands) +{ + /* Assuming Post-Increment insn has the following format : + 'MNEMONIC DISP(REG)+, REG' (e.g. 'loadw 12(r5)+, r6'). */ + if (strstr (operands, ")+") != NULL) + while (! IS_INSN_TYPE (LD_STOR_INS_INC)) + instruction++; +} + +/* Top level module where instruction parsing starts. + crx_ins - data structure holds some information. + operands - holds the operands part of the whole instruction. */ + +static void +parse_insn (ins *insn, char *operands) +{ + /* Handle 'excp'/'cinv' */ + if (IS_INSN_MNEMONIC ("excp") || IS_INSN_MNEMONIC ("cinv")) + { + insn->nargs = 1; + insn->arg[0].type = arg_ic; + insn->arg[0].size = 4; + insn->arg[0].constant = IS_INSN_MNEMONIC ("excp") ? + gettrap (operands) : get_cinv_parameters (operands); + return; + } + + /* Handle load/stor post-increment instructions. */ + if (IS_INSN_TYPE (LD_STOR_INS) || IS_INSN_TYPE (STOR_IMM_INS)) + handle_pi_insn (operands); + + if (operands != NULL) + parse_operands (insn, operands); +} + +/* Cinv instruction requires special handling. */ + +static int +get_cinv_parameters (char * operand) +{ + char *p = operand; + int d_used = 0, i_used = 0, u_used = 0; + + while (*++p != ']') + { + if (*p == ',' || *p == ' ') + continue; + + if (*p == 'd') + d_used = 1; + else if (*p == 'i') + i_used = 1; + else if (*p == 'u') + u_used = 1; + else + as_bad (_("Illegal `cinv' parameter: `%c'"), *p); + } + + return ((d_used ? 4 : 0) + + (i_used ? 2 : 0) + + (u_used ? 1 : 0)); +} + +/* Retrieve the opcode image of a given register. + If the register is illegal for the current instruction, + issue an error. */ + +static int +getreg_image (reg r) +{ + const reg_entry *reg; + char *reg_name; + int special_register_flag = 0; + int movpr_flag = 0; /* Nonzero means current mnemonic is 'mtpr'/'mfpr' */ + + if (IS_INSN_MNEMONIC ("mtpr") || IS_INSN_MNEMONIC ("mfpr")) + movpr_flag = 1; + + if (((IS_INSN_MNEMONIC ("mtpr")) && (processing_arg_number == 1)) + || ((IS_INSN_MNEMONIC ("mfpr")) && (processing_arg_number == 0)) ) + special_register_flag = 1; + + /* Check whether the register is in registers table. */ + if (r < MAX_REG) + reg = &crx_regtab[r]; + /* Check whether the register is in coprocessor registers table. */ + else if (r < MAX_COPREG) + reg = &crx_copregtab[r-MAX_REG]; + /* Register not found. */ + else + { + as_bad (_("Unknown register: `%d'"), r); + return 0; + } + + reg_name = reg->name; + +/* Issue a error message when register is illegal. */ +#define IMAGE_ERR \ + as_bad (_("Illegal register (`%s') in Instruction: `%s'"), \ + reg_name, ins_parse); \ + break; + + switch (reg->type) + { + case CRX_U_REGTYPE: + case CRX_CFG_REGTYPE: + case CRX_MTPR_REGTYPE: + if (movpr_flag && special_register_flag) + return reg->image; + else + IMAGE_ERR; + + case CRX_R_REGTYPE: + case CRX_C_REGTYPE: + case CRX_CS_REGTYPE: + if (!(movpr_flag && special_register_flag)) + return reg->image; + else + IMAGE_ERR; + + default: + IMAGE_ERR; + } + + return 0; +} + +/* Routine used to get the binary-string equivalent of a integer constant + which currently require currbits to represent itself to be extended to + nbits. */ + +static unsigned long int +getconstant (unsigned long int x, int nbits) +{ + int cnt = 0; + unsigned long int temp = x; + + while (temp > 0) + { + temp >>= 1; + cnt++; + } + + /* Escape sequence to next 16bit immediate. */ + if (cnt > nbits) + as_bad (_("Value `%ld' truncated to fit `%d' bits in instruction `%s'"), + x, cnt, ins_parse); + else + { + if (signflag) + x |= SET_BITS_MASK (cnt, nbits - cnt); + else + x &= CLEAR_BITS_MASK (cnt, nbits - cnt); + } + + /* The following expression avoids overflow if + 'nbits' is the number of bits in 'bfd_vma'. */ + return (x & ((((1 << (nbits - 1)) - 1) << 1) | 1)); +} + +/* Print a constant value to 'output_opcode': + ARG holds the operand's type and value. + SHIFT represents the location of the operand to be print into. + NBITS determines the size (in bits) of the constant. */ + +static void +print_constant (int nbits, int shift, argument *arg) +{ + unsigned long mask = 0; + + long constant = getconstant (arg->constant, nbits); + + switch (nbits) + { + case 32: + case 28: + case 24: + case 22: + /* mask the upper part of the constant, that is, the bits + going to the lowest byte of output_opcode[0]. + The upper part of output_opcode[1] is always filled, + therefore it is always masked with 0xFFFF. */ + mask = (1 << (nbits - 16)) - 1; + /* Divide the constant between two consecutive words : + 0 1 2 3 + +---------+---------+---------+---------+ + | | X X X X | X X X X | | + +---------+---------+---------+---------+ + output_opcode[0] output_opcode[1] */ + + CRX_PRINT (0, (constant >> WORD_SHIFT) & mask, 0); + CRX_PRINT (1, (constant & 0xFFFF), WORD_SHIFT); + break; + + case 16: + case 12: + /* Special case - in arg_cr, the SHIFT represents the location + of the REGISTER, not the constant, which is itself not shifted. */ + if (arg->type == arg_cr) + { + CRX_PRINT (0, constant, 0); + break; + } + + /* When instruction size is 3, a 16-bit constant is always + filling the upper part of output_opcode[1]. */ + if (instruction->size > 2) + CRX_PRINT (1, constant, WORD_SHIFT); + else + CRX_PRINT (0, constant, shift); + break; + + default: + CRX_PRINT (0, constant, shift); + break; + } +} + +/* Print an operand to 'output_opcode', which later on will be + printed to the object file: + ARG holds the operand's type, size and value. + SHIFT represents the printing location of operand. + NBITS determines the size (in bits) of a constant operand. */ + +static void +print_operand (int nbits, int shift, argument *arg) +{ + switch (arg->type) + { + case arg_r: + CRX_PRINT (0, getreg_image (arg->r), shift); + break; + + case arg_copr: + if (arg->cr < c0 || arg->cr > c15) + as_bad (_("Illegal Co-processor register in Instruction `%s' "), + ins_parse); + CRX_PRINT (0, getreg_image (arg->cr), shift); + break; + + case arg_copsr: + if (arg->cr < cs0 || arg->cr > cs15) + as_bad (_("Illegal Co-processor special register in Instruction `%s' "), + ins_parse); + CRX_PRINT (0, getreg_image (arg->cr), shift); + break; + + case arg_ic: + print_constant (nbits, shift, arg); + break; + + case arg_icr: + /* 16 12 8 6 0 + +--------------------------------+ + | reg | r_base | scl| disp | + +--------------------------------+ */ + CRX_PRINT (0, getreg_image (arg->r), 12); + CRX_PRINT (0, getreg_image (arg->i_r), 8); + CRX_PRINT (0, arg->scale, 6); + print_constant (nbits, shift, arg); + break; + + case arg_rbase: + CRX_PRINT (0, getreg_image (arg->r), shift); + break; + + case arg_cr: + /* case base_cst4. */ + if ((instruction->flags & CST4MAP) && cst4flag) + output_opcode[0] |= (getconstant (arg->constant, nbits) + << (shift + REG_SIZE)); + else + /* rbase_dispu<NN> and other such cases. */ + print_constant (nbits, shift, arg); + /* Add the register argument to the output_opcode. */ + CRX_PRINT (0, getreg_image (arg->r), shift); + break; + + case arg_c: + print_constant (nbits, shift, arg); + break; + + default: + break; + } +} + +/* Retrieve the number of operands for the current assembled instruction. */ + +static int +get_number_of_operands (void) +{ + int i; + + for (i = 0; instruction->operands[i].op_type && i < MAX_OPERANDS; i++) + ; + return i; +} + +/* Assemble a single instruction : + Instruction has been parsed and all operand values set appropriately. + Algorithm for assembling - + For instruction to be assembled: + Step 1: Find instruction in the array crx_instruction with same mnemonic. + Step 2: Find instruction with same operand types. + Step 3: If (size_of_operands) match then done, else increment the + array_index and goto Step3. + Step 4: Cannot assemble + Returns 1 upon success, 0 upon failure. */ + +static int +assemble_insn (char *mnemonic, ins *insn) +{ + /* Argument type of each operand in the instruction we are looking for. */ + argtype atyp[MAX_OPERANDS]; + /* Argument type of each operand in the current instruction. */ + argtype atyp_act[MAX_OPERANDS]; + /* Size (in bits) of each operand in the instruction we are looking for. */ + int bits[MAX_OPERANDS]; + /* Size (in bits) of each operand in the current instruction. */ + int bits_act[MAX_OPERANDS]; + /* Location (in bits) of each operand in the current instruction. */ + int shift_act[MAX_OPERANDS]; + int match = 0; + int done_flag = 0; + int cst4maptype = 0; + int changed_already = 0; + unsigned int temp_value = 0; + int instrtype, i; + /* A pointer to the argument's constant value. */ + unsigned long int *cons; + /* Pointer to loop over all cst4_map entries. */ + const cst4_entry *cst4_op; + + /* Instruction has no operands -> copy only the constant opcode. */ + if (insn->nargs == 0) + { + output_opcode[0] = BIN (instruction->match, instruction->match_bits); + return 1; + } + + /* Find instruction with same number of operands. */ + while (get_number_of_operands () != insn->nargs + && IS_INSN_MNEMONIC (mnemonic)) + instruction++; + + if (!IS_INSN_MNEMONIC (mnemonic)) + return 0; + + /* Initialize argument type and size of each given operand. */ + for (i = 0; i < insn->nargs; i++) + { + atyp[i] = insn->arg[i].type; + bits[i] = insn->arg[i].size; + } + + /* Initialize argument type and size of each operand in current inst. */ + GET_ACTUAL_TYPE; + GET_ACTUAL_SIZE; + + while (match != 1 + /* Check we didn't get to end of table. */ + && instruction->mnemonic != NULL + /* Check that the actual mnemonic is still available. */ + && IS_INSN_MNEMONIC (mnemonic)) + { + /* Check for argement type compatibility. */ + for (i = 0; i < insn->nargs; i++) + { + if (atyp_act[i] == atyp[i]) + done_flag = 1; + else + { + done_flag = 0; + break; + } + } + if (done_flag) + { + /* Check for post inc mode of the current instruction. */ + if (post_inc_mode == 1 || IS_INSN_TYPE (LD_STOR_INS_INC)) + done_flag = (post_inc_mode == IS_INSN_TYPE (LD_STOR_INS_INC)); + } + + if (done_flag == 0) + { + /* Try again with next instruction. */ + instruction++; + GET_ACTUAL_TYPE; + GET_ACTUAL_SIZE; + continue; + } + else + { + /* Check for size compatibility. */ + for (i = 0; i < insn->nargs; i++) + { + if (bits[i] > bits_act[i]) + { + /* Actual size is too small - try again. */ + done_flag = 0; + instruction++; + GET_ACTUAL_TYPE; + GET_ACTUAL_SIZE; + break; + } + } + + } + + if (done_flag == 1) + { + /* Full match is found. */ + match = 1; + break; + } + } + + if (match == 0) + /* We haven't found a match - instruction can't be assembled. */ + return 0; + else + /* Full match - print the final image. */ + { + /* Handle positive constants. */ + if (!signflag) + { + if (IS_INSN_TYPE (LD_STOR_INS) && !relocatable) + { + /* Get the map type of the instruction. */ + instrtype = instruction->flags & REVERSE_MATCH ? 0 : 1; + cons = &insn->arg[instrtype].constant; + cst4maptype = instruction->flags & CST4MAP; + + switch (cst4maptype) + { + case DISPUB4: + /* 14 and 15 are reserved escape sequences of dispub4. */ + if (*cons == 14 || *cons == 15) + { + instruction++; + GET_ACTUAL_SIZE; + } + break; + + case DISPUW4: + /* Mapping has to be done. */ + if (*cons <= 15 && *cons % 2 != 0) + { + instruction++; + GET_ACTUAL_SIZE; + } + else if (*cons > 15 && *cons < 27 && *cons % 2 == 0) + { + instruction--; + GET_ACTUAL_SIZE; + } + if (*cons < 27 && *cons % 2 == 0) + *cons /= 2; + break; + + case DISPUD4: + /* Mapping has to be done. */ + if (*cons <= 15 && *cons % 4 != 0) + { + instruction++; + GET_ACTUAL_SIZE; + } + else if (*cons > 15 && *cons < 53 && *cons % 4 == 0) + { + instruction--; + GET_ACTUAL_SIZE; + } + if (*cons < 53 && *cons % 4 == 0) + *cons /= 4; + break; + default: + break; + } + } + if ((IS_INSN_TYPE (ARITH_BYTE_INS) || IS_INSN_TYPE (ARITH_INS)) + && !relocatable) + { + /* Check whether a cst4 mapping has to be done. */ + if ((instruction->operands[0].op_type == cst4 + || instruction->operands[0].op_type == i16) + && (instruction->operands[1].op_type == regr)) + { + /* 'const' equals reserved escape sequences -->> + represent as i16. */ + if (insn->arg[0].constant == ESC_16 + || insn->arg[0].constant == ESC_32) + { + instruction++; + GET_ACTUAL_SIZE; + } + else + { + /* Loop over cst4_map entries. */ + for (cst4_op = cst4_map; cst4_op < (cst4_map + cst4_maps); + cst4_op++) + { + /* 'const' equals a binary, which is already mapped + by a different value -->> represent as i16. */ + if (insn->arg[0].constant == (unsigned int)cst4_op->binary + && cst4_op->binary != cst4_op->value) + { + instruction++; + GET_ACTUAL_SIZE; + } + /* 'const' equals a value bigger than 16 -->> map to + its binary and represent as cst4. */ + else if (insn->arg[0].constant == (unsigned int)cst4_op->value + && insn->arg[0].constant >= 16) + { + instruction--; + insn->arg[0].constant = cst4_op->binary; + GET_ACTUAL_SIZE; + } + } + } + } + /* Special check for 'addub 0, r0' instruction - + The opcode '0000 0000 0000 0000' is not allowed. */ + if (IS_INSN_MNEMONIC ("addub")) + { + if ((instruction->operands[0].op_type == cst4) + && instruction->operands[1].op_type == regr) + { + if (insn->arg[0].constant == 0 && insn->arg[1].r == r0) + instruction++; + } + } + } + if (IS_INSN_TYPE (LD_STOR_INS) || IS_INSN_TYPE (STOR_IMM_INS) + || IS_INSN_TYPE (LD_STOR_INS_INC)) + { + instrtype = instruction->flags & REVERSE_MATCH ? 0 : 1; + if (instruction->operands[instrtype].op_type == rbase) + instruction++; + } + /* Error checking in case of post-increment instruction. */ + if (IS_INSN_TYPE (LD_STOR_INS_INC)) + { + if (!((strneq (instruction->mnemonic, "stor", 4)) + && (insn->arg[0].type != arg_r))) + if (insn->arg[0].r == insn->arg[1].r) + as_bad (_("Invalid instruction : `%s' Source and Destination register \ + same in Post INC mode"), ins_parse); + } + if (IS_INSN_TYPE (CSTBIT_INS) && !relocatable) + { + if (instruction->operands[1].op_type == rbase_dispu12) + { + if (insn->arg[1].constant == 0) + { + instruction--; + GET_ACTUAL_SIZE; + } + } + } + if ((IS_INSN_TYPE (LD_STOR_INS) || IS_INSN_TYPE (CSTBIT_INS) + || IS_INSN_TYPE (STOR_IMM_INS) + || IS_INSN_TYPE (LD_STOR_INS_INC)) & !relocatable) + { + instrtype = instruction->flags & REVERSE_MATCH ? 0 : 1; + changed_already = 0; + /* Convert 32 bits accesses to 16 bits accesses. */ + if (instruction->operands[instrtype].op_type == abs32) + { + if ((insn->arg[instrtype].constant & 0xFFFF0000) == 0xFFFF0000) + { + instruction--; + insn->arg[instrtype].constant = + insn->arg[instrtype].constant & 0xFFFF; + insn->arg[instrtype].size = 16; + changed_already = 1; + GET_ACTUAL_SIZE; + } + } + /* Convert 16 bits accesses to 32 bits accesses. */ + if (instruction->operands[instrtype].op_type == abs16 + && changed_already != 1) + { + instruction++; + insn->arg[instrtype].constant = + insn->arg[instrtype].constant & 0xFFFF; + insn->arg[instrtype].size = 32; + GET_ACTUAL_SIZE; + } + changed_already = 0; + } + if (IS_INSN_TYPE (BRANCH_INS) && !relocatable) + { + /* 0x7e and 0x7f are reserved escape sequences of dispe9. */ + if (insn->arg[0].constant == 0x7e || insn->arg[0].constant == 0x7f) + { + instruction++; + GET_ACTUAL_SIZE; + } + } + } + + for (i = 0; i < insn->nargs; i++) + { + if (instruction->operands[i].op_type == cst4 + || instruction->operands[i].op_type == rbase_cst4) + cst4flag = 1; + } + + /* First, copy the instruction's opcode. */ + output_opcode[0] = BIN (instruction->match, instruction->match_bits); + + /* Swap the argument values in case bcop instructions. */ + if (IS_INSN_TYPE (COP_BRANCH_INS)) + { + temp_value = insn->arg[0].constant; + insn->arg[0].constant = insn->arg[1].constant; + insn->arg[1].constant = temp_value; + } + + for (i = 0; i < insn->nargs; i++) + { + shift_act[i] = instruction->operands[i].shift; + signflag = insn->arg[i].signflag; + processing_arg_number = i; + print_operand (bits_act[i], shift_act[i], &insn->arg[i]); + } + } + + return 1; +} + +/* Set the appropriate bit for register 'r' in 'mask'. + This indicates that this register is loaded or stored by + the instruction. */ + +static void +mask_reg (int r, unsigned short int *mask) +{ + if ((reg)r > (reg)sp) + { + as_bad (_("Invalid Register in Register List")); + return; + } + + *mask |= (1 << r); +} + +/* Preprocess register list - create a 16-bit mask with one bit for each + of the 16 general purpose registers. If a bit is set, it indicates + that this register is loaded or stored by the instruction. */ + +static char * +preprocess_reglist (char *param, int *allocated) +{ + char reg_name[MAX_REGNAME_LEN]; /* Current parsed register name. */ + char *regP; /* Pointer to 'reg_name' string. */ + int reg_counter = 0; /* Count number of parsed registers. */ + unsigned short int mask = 0; /* Mask for 16 general purpose registers. */ + char *new_param; /* New created operands string. */ + char *paramP = param; /* Pointer to original opearands string. */ + char maskstring[10]; /* Array to print the mask as a string. */ + reg r; + copreg cr; + + /* If 'param' is already in form of a number, no need to preprocess. */ + if (strchr (paramP, '{') == NULL) + return param; + + /* Verifying correct syntax of operand. */ + if (strchr (paramP, '}') == NULL) + as_fatal (_("Missing matching brackets : `%s'"), ins_parse); + + while (*paramP++ != '{'); + + new_param = (char *)xcalloc (MAX_INST_LEN, sizeof (char)); + *allocated = 1; + strncpy (new_param, param, paramP - param - 1); + + while (*paramP != '}') + { + regP = paramP; + memset (®_name, '\0', sizeof (reg_name)); + + while (ISALNUM (*paramP)) + paramP++; + + strncpy (reg_name, regP, paramP - regP); + + if (IS_INSN_TYPE (COP_REG_INS)) + { + if ((cr = get_copregister (reg_name)) == nullcopregister) + as_bad (_("Illegal register `%s' in cop-register list"), reg_name); + mask_reg (getreg_image (cr - c0), &mask); + } + else + { + if ((r = get_register (reg_name)) == nullregister) + as_bad (_("Illegal register `%s' in register list"), reg_name); + mask_reg (getreg_image (r), &mask); + } + + if (++reg_counter > MAX_REGS_IN_MASK16) + as_bad (_("Maximum %d bits may be set in `mask16' operand"), + MAX_REGS_IN_MASK16); + + while (!ISALNUM (*paramP) && *paramP != '}') + paramP++; + } + + if (*++paramP != '\0') + as_warn (_("rest of line ignored; first ignored character is `%c'"), + *paramP); + + if (mask == 0) + as_bad (_("Illegal `mask16' operand, operation is undefined - `%s'"), + ins_parse); + + sprintf (maskstring, "$0x%x", mask); + strcat (new_param, maskstring); + return new_param; +} + +/* Print the instruction. + Handle also cases where the instruction is relaxable/relocatable. */ + +void +print_insn (ins *insn) +{ + unsigned int i, j, insn_size; + char *this_frag; + unsigned short words[4]; + + /* Arrange the insn encodings in a WORD size array. */ + for (i = 0, j = 0; i < 2; i++) + { + words[j++] = (output_opcode[i] >> 16) & 0xFFFF; + words[j++] = output_opcode[i] & 0xFFFF; + } + + /* Handle relaxtion. */ + if ((instruction->flags & RELAXABLE) && relocatable) + { + int relax_subtype; + + /* Write the maximal instruction size supported. */ + insn_size = INSN_MAX_SIZE; + + /* bCC */ + if (IS_INSN_TYPE (BRANCH_INS)) + relax_subtype = 0; + /* bal */ + else if (IS_INSN_TYPE (DCR_BRANCH_INS) || IS_INSN_MNEMONIC ("bal")) + relax_subtype = 3; + /* cmpbr */ + else if (IS_INSN_TYPE (CMPBR_INS)) + relax_subtype = 5; + else + abort (); + + this_frag = frag_var (rs_machine_dependent, insn_size * 2, + 4, relax_subtype, + insn->exp.X_add_symbol, + insn->exp.X_add_number, + 0); + } + else + { + insn_size = instruction->size; + this_frag = frag_more (insn_size * 2); + + /* Handle relocation. */ + if ((relocatable) && (insn->rtype != BFD_RELOC_NONE)) + { + reloc_howto_type *reloc_howto; + int size; + + reloc_howto = bfd_reloc_type_lookup (stdoutput, insn->rtype); + + if (!reloc_howto) + abort (); + + size = bfd_get_reloc_size (reloc_howto); + + if (size < 1 || size > 4) + abort (); + + fix_new_exp (frag_now, this_frag - frag_now->fr_literal, + size, &insn->exp, reloc_howto->pc_relative, + insn->rtype); + } + } + + /* Write the instruction encoding to frag. */ + for (i = 0; i < insn_size; i++) + { + md_number_to_chars (this_frag, (valueT) words[i], 2); + this_frag += 2; + } +} + +/* This is the guts of the machine-dependent assembler. OP points to a + machine dependent instruction. This function is supposed to emit + the frags/bytes it assembles to. */ + +void +md_assemble (char *op) +{ + ins crx_ins; + char *param; + char c; + + /* Reset global variables for a new instruction. */ + reset_vars (op, &crx_ins); + + /* Strip the mnemonic. */ + for (param = op; *param != 0 && !ISSPACE (*param); param++) + ; + c = *param; + *param++ = '\0'; + + /* Find the instruction. */ + instruction = (const inst *) hash_find (crx_inst_hash, op); + if (instruction == NULL) + { + as_bad (_("Unknown opcode: `%s'"), op); + return; + } + + /* Tie dwarf2 debug info to the address at the start of the insn. */ + dwarf2_emit_insn (0); + + if (NO_OPERANDS_INST (op)) + /* Handle instructions with no operands. */ + crx_ins.nargs = 0; + else + /* Parse the instruction's operands. */ + parse_insn (&crx_ins, param); + + /* Assemble the instruction. */ + if (assemble_insn (op, &crx_ins) == 0) + { + as_bad (_("Illegal operands in instruction : `%s'"), ins_parse); + return; + } + + /* Print the instruction. */ + print_insn (&crx_ins); +} |