* config/tc-v850.h (TC_GENERIC_RELAX_TABLE): Define.
* config/tc-v850.c: Fix some indention problems. (md_relax_table): Define for D9->D99 branch displacement relaxing. (md_convert_frag): Do something useful instead of aborting. (md_estimate_size_before_relax): Likewise. (md_assemble): Note if the matching instruction has a relaxable operand. If it does, allocate frag with frag_var and don't do any fixups. So we can do 9bit displacement to 22bit displacement relaxing.
This commit is contained in:
parent
244558e354
commit
a334533c1b
3 changed files with 177 additions and 94 deletions
|
@ -1,3 +1,24 @@
|
||||||
|
start-sanitize-v850
|
||||||
|
Tue Oct 29 12:28:16 1996 Jeffrey A Law (law@cygnus.com)
|
||||||
|
|
||||||
|
* config/tc-v850.h (TC_GENERIC_RELAX_TABLE): Define.
|
||||||
|
* config/tc-v850.c: Fix some indention problems.
|
||||||
|
(md_relax_table): Define for D9->D99 branch displacement
|
||||||
|
relaxing.
|
||||||
|
(md_convert_frag): Do something useful instead of aborting.
|
||||||
|
(md_estimate_size_before_relax): Likewise.
|
||||||
|
(md_assemble): Note if the matching instruction has a relaxable
|
||||||
|
operand. If it does, allocate frag with frag_var and don't
|
||||||
|
do any fixups.
|
||||||
|
|
||||||
|
end-sanitize-v850
|
||||||
|
start-sanitize-d10v
|
||||||
|
Mon Oct 28 10:48:40 1996 Martin M. Hunt <hunt@pizza.cygnus.com>
|
||||||
|
|
||||||
|
* config/tc-d10v.h (md_cleanup): New function. This is needed to
|
||||||
|
write out any buffered instructions when a ".end" is found.
|
||||||
|
|
||||||
|
end-sanitize-d10v
|
||||||
Mon Oct 28 10:43:45 1996 Martin M. Hunt <hunt@pizza.cygnus.com>
|
Mon Oct 28 10:43:45 1996 Martin M. Hunt <hunt@pizza.cygnus.com>
|
||||||
|
|
||||||
* read.c (read_a_source_file): New hook md_cleanup().
|
* read.c (read_a_source_file): New hook md_cleanup().
|
||||||
|
|
|
@ -56,6 +56,11 @@ const char EXP_CHARS[] = "eE";
|
||||||
const char FLT_CHARS[] = "dD";
|
const char FLT_CHARS[] = "dD";
|
||||||
|
|
||||||
|
|
||||||
|
const relax_typeS md_relax_table[] = {
|
||||||
|
{0xff, -0x100, 2, 1},
|
||||||
|
{0x1fffff, -0x200000, 6, 0},
|
||||||
|
};
|
||||||
|
|
||||||
/* local functions */
|
/* local functions */
|
||||||
static unsigned long v850_insert_operand
|
static unsigned long v850_insert_operand
|
||||||
PARAMS ((unsigned long insn, const struct v850_operand *operand,
|
PARAMS ((unsigned long insn, const struct v850_operand *operand,
|
||||||
|
@ -410,14 +415,42 @@ md_atof (type, litp, sizep)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Very gross. */
|
||||||
void
|
void
|
||||||
md_convert_frag (abfd, sec, fragP)
|
md_convert_frag (abfd, sec, fragP)
|
||||||
bfd *abfd;
|
bfd *abfd;
|
||||||
asection *sec;
|
asection *sec;
|
||||||
fragS *fragP;
|
fragS *fragP;
|
||||||
{
|
{
|
||||||
/* printf ("call to md_convert_frag \n"); */
|
subseg_change (sec, 0);
|
||||||
abort ();
|
if (fragP->fr_subtype == 0)
|
||||||
|
{
|
||||||
|
fragP->fr_var = 0;
|
||||||
|
fragP->fr_fix = 2;
|
||||||
|
fix_new (fragP, 0, 2, fragP->fr_symbol,
|
||||||
|
fragP->fr_offset, 1, BFD_RELOC_UNUSED + (int)fragP->fr_opcode);
|
||||||
|
}
|
||||||
|
else if (fragP->fr_subtype == 1)
|
||||||
|
{
|
||||||
|
fragP->fr_var = 0;
|
||||||
|
fragP->fr_fix = 6;
|
||||||
|
/* Reverse the condition of the first branch. */
|
||||||
|
fragP->fr_literal[0] &= 0xf7;
|
||||||
|
/* Mask off all the displacement bits. */
|
||||||
|
fragP->fr_literal[0] &= 0x8f;
|
||||||
|
fragP->fr_literal[1] &= 0x07;
|
||||||
|
/* Now set the displacement bits so that we branch
|
||||||
|
around the unconditional branch. */
|
||||||
|
fragP->fr_literal[0] |= 0x30;
|
||||||
|
|
||||||
|
/* Now create the unconditional branch + fixup to the final
|
||||||
|
target. */
|
||||||
|
md_number_to_chars (&fragP->fr_literal[2], 0x00000780, 4);
|
||||||
|
fix_new (fragP, 2, 4, fragP->fr_symbol,
|
||||||
|
fragP->fr_offset, 1, BFD_RELOC_UNUSED + (int)fragP->fr_opcode + 1);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
abort ();
|
||||||
}
|
}
|
||||||
|
|
||||||
valueT
|
valueT
|
||||||
|
@ -537,7 +570,7 @@ md_assemble (str)
|
||||||
struct v850_opcode *opcode;
|
struct v850_opcode *opcode;
|
||||||
struct v850_opcode *next_opcode;
|
struct v850_opcode *next_opcode;
|
||||||
const unsigned char *opindex_ptr;
|
const unsigned char *opindex_ptr;
|
||||||
int next_opindex;
|
int next_opindex, relaxable;
|
||||||
unsigned long insn, insn_size;
|
unsigned long insn, insn_size;
|
||||||
char *f;
|
char *f;
|
||||||
int i;
|
int i;
|
||||||
|
@ -568,6 +601,7 @@ md_assemble (str)
|
||||||
{
|
{
|
||||||
const char *errmsg = NULL;
|
const char *errmsg = NULL;
|
||||||
|
|
||||||
|
relaxable = 0;
|
||||||
fc = 0;
|
fc = 0;
|
||||||
match = 0;
|
match = 0;
|
||||||
next_opindex = 0;
|
next_opindex = 0;
|
||||||
|
@ -593,6 +627,9 @@ md_assemble (str)
|
||||||
while (*str == ' ' || *str == ',' || *str == '[' || *str == ']')
|
while (*str == ' ' || *str == ',' || *str == '[' || *str == ']')
|
||||||
++str;
|
++str;
|
||||||
|
|
||||||
|
if (operand->flags & V850_OPERAND_RELAX)
|
||||||
|
relaxable = 1;
|
||||||
|
|
||||||
/* Gather the operand. */
|
/* Gather the operand. */
|
||||||
hold = input_line_pointer;
|
hold = input_line_pointer;
|
||||||
input_line_pointer = str;
|
input_line_pointer = str;
|
||||||
|
@ -647,107 +684,107 @@ md_assemble (str)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if ((operand->flags & V850_OPERAND_REG) != 0)
|
if ((operand->flags & V850_OPERAND_REG) != 0)
|
||||||
{
|
|
||||||
if (!register_name(&ex))
|
|
||||||
{
|
{
|
||||||
errmsg = "invalid register name";
|
if (!register_name(&ex))
|
||||||
goto error;
|
{
|
||||||
|
errmsg = "invalid register name";
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
else if ((operand->flags & V850_OPERAND_SRG) != 0)
|
||||||
else if ((operand->flags & V850_OPERAND_SRG) != 0)
|
|
||||||
{
|
|
||||||
if (!system_register_name(&ex))
|
|
||||||
{
|
{
|
||||||
errmsg = "invalid system register name";
|
if (!system_register_name(&ex))
|
||||||
goto error;
|
{
|
||||||
|
errmsg = "invalid system register name";
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
else if ((operand->flags & V850_OPERAND_EP) != 0)
|
||||||
else if ((operand->flags & V850_OPERAND_EP) != 0)
|
|
||||||
{
|
|
||||||
char *start = input_line_pointer;
|
|
||||||
char c = get_symbol_end ();
|
|
||||||
if (strcmp (start, "ep") != 0 && strcmp (start, "r30") != 0)
|
|
||||||
{
|
{
|
||||||
/* Put things back the way we found them. */
|
char *start = input_line_pointer;
|
||||||
|
char c = get_symbol_end ();
|
||||||
|
if (strcmp (start, "ep") != 0 && strcmp (start, "r30") != 0)
|
||||||
|
{
|
||||||
|
/* Put things back the way we found them. */
|
||||||
|
*input_line_pointer = c;
|
||||||
|
input_line_pointer = start;
|
||||||
|
errmsg = "expected EP register";
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
*input_line_pointer = c;
|
*input_line_pointer = c;
|
||||||
input_line_pointer = start;
|
str = input_line_pointer;
|
||||||
errmsg = "expected EP register";
|
input_line_pointer = hold;
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
*input_line_pointer = c;
|
|
||||||
str = input_line_pointer;
|
|
||||||
input_line_pointer = hold;
|
|
||||||
|
|
||||||
while (*str == ' ' || *str == ',' || *str == '[' || *str == ']')
|
while (*str == ' ' || *str == ',' || *str == '[' || *str == ']')
|
||||||
++str;
|
++str;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
else if ((operand->flags & V850_OPERAND_CC) != 0)
|
else if ((operand->flags & V850_OPERAND_CC) != 0)
|
||||||
{
|
|
||||||
if (!cc_name(&ex))
|
|
||||||
{
|
{
|
||||||
errmsg = "invalid condition code name";
|
if (!cc_name(&ex))
|
||||||
|
{
|
||||||
|
errmsg = "invalid condition code name";
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (register_name (&ex)
|
||||||
|
&& (operand->flags & V850_OPERAND_REG) == 0)
|
||||||
|
{
|
||||||
|
errmsg = "syntax error: register not expected";
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
}
|
else if (system_register_name (&ex)
|
||||||
else if (register_name (&ex)
|
&& (operand->flags & V850_OPERAND_SRG) == 0)
|
||||||
&& (operand->flags & V850_OPERAND_REG) == 0)
|
{
|
||||||
{
|
errmsg = "syntax error: system register not expected";
|
||||||
errmsg = "syntax error: register not expected";
|
goto error;
|
||||||
goto error;
|
}
|
||||||
}
|
else if (cc_name (&ex)
|
||||||
else if (system_register_name (&ex)
|
&& (operand->flags & V850_OPERAND_CC) == 0)
|
||||||
&& (operand->flags & V850_OPERAND_SRG) == 0)
|
{
|
||||||
{
|
errmsg = "syntax error: condition code not expected";
|
||||||
errmsg = "syntax error: system register not expected";
|
goto error;
|
||||||
goto error;
|
}
|
||||||
}
|
else
|
||||||
else if (cc_name (&ex)
|
{
|
||||||
&& (operand->flags & V850_OPERAND_CC) == 0)
|
expression(&ex);
|
||||||
{
|
}
|
||||||
errmsg = "syntax error: condition code not expected";
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
expression(&ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (ex.X_op)
|
switch (ex.X_op)
|
||||||
{
|
|
||||||
case O_illegal:
|
|
||||||
errmsg = "illegal operand";
|
|
||||||
goto error;
|
|
||||||
case O_absent:
|
|
||||||
errmsg = "missing operand";
|
|
||||||
goto error;
|
|
||||||
case O_register:
|
|
||||||
if ((operand->flags & (V850_OPERAND_REG | V850_OPERAND_SRG)) == 0)
|
|
||||||
{
|
{
|
||||||
errmsg = "invalid operand";
|
case O_illegal:
|
||||||
|
errmsg = "illegal operand";
|
||||||
goto error;
|
goto error;
|
||||||
}
|
case O_absent:
|
||||||
|
errmsg = "missing operand";
|
||||||
|
goto error;
|
||||||
|
case O_register:
|
||||||
|
if ((operand->flags & (V850_OPERAND_REG | V850_OPERAND_SRG)) == 0)
|
||||||
|
{
|
||||||
|
errmsg = "invalid operand";
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
insn = v850_insert_operand (insn, operand, ex.X_add_number,
|
insn = v850_insert_operand (insn, operand, ex.X_add_number,
|
||||||
(char *) NULL, 0);
|
(char *) NULL, 0);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case O_constant:
|
case O_constant:
|
||||||
insn = v850_insert_operand (insn, operand, ex.X_add_number,
|
insn = v850_insert_operand (insn, operand, ex.X_add_number,
|
||||||
(char *) NULL, 0);
|
(char *) NULL, 0);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
/* We need to generate a fixup for this expression. */
|
/* We need to generate a fixup for this expression. */
|
||||||
if (fc >= MAX_INSN_FIXUPS)
|
if (fc >= MAX_INSN_FIXUPS)
|
||||||
as_fatal ("too many fixups");
|
as_fatal ("too many fixups");
|
||||||
fixups[fc].exp = ex;
|
fixups[fc].exp = ex;
|
||||||
fixups[fc].opindex = *opindex_ptr;
|
fixups[fc].opindex = *opindex_ptr;
|
||||||
fixups[fc].reloc = BFD_RELOC_UNUSED;
|
fixups[fc].reloc = BFD_RELOC_UNUSED;
|
||||||
++fc;
|
++fc;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -787,12 +824,27 @@ md_assemble (str)
|
||||||
/* Write out the instruction.
|
/* Write out the instruction.
|
||||||
|
|
||||||
Four byte insns have an opcode with the two high bits on. */
|
Four byte insns have an opcode with the two high bits on. */
|
||||||
if ((insn & 0x0600) == 0x0600)
|
if (relaxable && fc > 0)
|
||||||
insn_size = 4;
|
{
|
||||||
|
f = frag_var (rs_machine_dependent, 6, 4, 0,
|
||||||
|
fixups[0].exp.X_add_symbol, 0, (char *)fixups[0].opindex);
|
||||||
|
insn_size = 2;
|
||||||
|
md_number_to_chars (f, insn, insn_size);
|
||||||
|
md_number_to_chars (f + 2, 0, 4);
|
||||||
|
fc = 0;
|
||||||
|
}
|
||||||
|
else if ((insn & 0x0600) == 0x0600)
|
||||||
|
{
|
||||||
|
insn_size = 4;
|
||||||
|
f = frag_more (insn_size);
|
||||||
|
md_number_to_chars (f, insn, insn_size);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
insn_size = 2;
|
{
|
||||||
f = frag_more (insn_size);
|
insn_size = 2;
|
||||||
md_number_to_chars (f, insn, insn_size);
|
f = frag_more (insn_size);
|
||||||
|
md_number_to_chars (f, insn, insn_size);
|
||||||
|
}
|
||||||
|
|
||||||
/* Create any fixups. At this point we do not use a
|
/* Create any fixups. At this point we do not use a
|
||||||
bfd_reloc_code_real_type, but instead just use the
|
bfd_reloc_code_real_type, but instead just use the
|
||||||
|
@ -867,14 +919,17 @@ tc_gen_reloc (seg, fixp)
|
||||||
return reloc;
|
return reloc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Assume everything will fit in two bytes, then expand as necessary. */
|
||||||
int
|
int
|
||||||
md_estimate_size_before_relax (fragp, seg)
|
md_estimate_size_before_relax (fragp, seg)
|
||||||
fragS *fragp;
|
fragS *fragp;
|
||||||
asection *seg;
|
asection *seg;
|
||||||
{
|
{
|
||||||
return 0;
|
fragp->fr_var = 4;
|
||||||
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
long
|
long
|
||||||
md_pcrel_from (fixp)
|
md_pcrel_from (fixp)
|
||||||
fixS *fixp;
|
fixS *fixp;
|
||||||
|
|
|
@ -52,3 +52,10 @@
|
||||||
#define tc_fix_adjustable(FIX) \
|
#define tc_fix_adjustable(FIX) \
|
||||||
(!(FIX)->fx_pcrel && (FIX)->fx_r_type != BFD_RELOC_V850_TDA_OFFSET)
|
(!(FIX)->fx_pcrel && (FIX)->fx_r_type != BFD_RELOC_V850_TDA_OFFSET)
|
||||||
|
|
||||||
|
/* We need to handle lo(), hi(), etc etc in .hword, .word, etc
|
||||||
|
directives, so we have to parse "cons" expressions ourselves. */
|
||||||
|
#define TC_PARSE_CONS_EXPRESSION(EXP, NBYTES) parse_cons_expression_v850 (EXP)
|
||||||
|
#define TC_CONS_FIX_NEW cons_fix_new_v850
|
||||||
|
extern const struct relax_type md_relax_table[];
|
||||||
|
#define TC_GENERIC_RELAX_TABLE md_relax_table
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue