* 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:
Jeff Law 1996-10-29 19:32:56 +00:00
parent 244558e354
commit a334533c1b
3 changed files with 177 additions and 94 deletions

View file

@ -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().

View file

@ -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,13 +415,41 @@ 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);
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 (); abort ();
} }
@ -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;
@ -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)
{
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; insn_size = 4;
f = frag_more (insn_size);
md_number_to_chars (f, insn, insn_size);
}
else else
{
insn_size = 2; insn_size = 2;
f = frag_more (insn_size); f = frag_more (insn_size);
md_number_to_chars (f, insn, 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;

View file

@ -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