1998-03-01 22:58:51 +00:00
|
|
|
/* GAS interface for targets using CGEN: Cpu tools GENerator.
|
2001-11-01 09:24:29 +00:00
|
|
|
Copyright 1996, 1997, 1998, 1999, 2000, 2001
|
|
|
|
Free Software Foundation, Inc.
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
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
|
2001-11-01 09:24:29 +00:00
|
|
|
Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
|
1998-03-01 22:58:51 +00:00
|
|
|
|
1998-09-06 22:57:45 +00:00
|
|
|
#include <setjmp.h>
|
1998-03-01 22:58:51 +00:00
|
|
|
#include "ansidecl.h"
|
2000-05-12 23:15:20 +00:00
|
|
|
#include "libiberty.h"
|
1998-03-01 22:58:51 +00:00
|
|
|
#include "bfd.h"
|
1998-09-06 22:57:45 +00:00
|
|
|
#include "symcat.h"
|
2000-05-12 23:15:20 +00:00
|
|
|
#include "cgen-desc.h"
|
1998-03-01 22:58:51 +00:00
|
|
|
#include "as.h"
|
|
|
|
#include "subsegs.h"
|
2000-05-12 23:15:20 +00:00
|
|
|
#include "cgen.h"
|
2001-11-01 09:24:29 +00:00
|
|
|
#include "dwarf2dbg.h"
|
2000-05-12 23:15:20 +00:00
|
|
|
|
2002-01-27 12:00:11 +00:00
|
|
|
static void queue_fixup PARAMS ((int, int, expressionS *));
|
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
/* Opcode table descriptor, must be set by md_begin. */
|
|
|
|
|
|
|
|
CGEN_CPU_DESC gas_cgen_cpu_desc;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* Callback to insert a register into the symbol table.
|
|
|
|
A target may choose to let GAS parse the registers.
|
|
|
|
??? Not currently used. */
|
|
|
|
|
|
|
|
void
|
|
|
|
cgen_asm_record_register (name, number)
|
2001-11-01 09:24:29 +00:00
|
|
|
char *name;
|
2000-05-12 23:15:20 +00:00
|
|
|
int number;
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
|
|
|
/* Use symbol_create here instead of symbol_new so we don't try to
|
|
|
|
output registers into the object file's symbol table. */
|
|
|
|
symbol_table_insert (symbol_create (name, reg_section,
|
2001-11-01 09:24:29 +00:00
|
|
|
number, &zero_address_frag));
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* We need to keep a list of fixups. We can't simply generate them as
|
|
|
|
we go, because that would require us to first create the frag, and
|
|
|
|
that would screw up references to ``.''.
|
|
|
|
|
|
|
|
This is used by cpu's with simple operands. It keeps knowledge of what
|
|
|
|
an `expressionS' is and what a `fixup' is out of CGEN which for the time
|
|
|
|
being is preferable.
|
|
|
|
|
|
|
|
OPINDEX is the index in the operand table.
|
|
|
|
OPINFO is something the caller chooses to help in reloc determination. */
|
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
struct fixup {
|
2000-05-12 23:15:20 +00:00
|
|
|
int opindex;
|
|
|
|
int opinfo;
|
1998-03-01 22:58:51 +00:00
|
|
|
expressionS exp;
|
|
|
|
};
|
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
static struct fixup fixups[GAS_CGEN_MAX_FIXUPS];
|
2000-05-12 23:15:20 +00:00
|
|
|
static int num_fixups;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* Prepare to parse an instruction.
|
|
|
|
??? May wish to make this static and delete calls in md_assemble. */
|
|
|
|
|
|
|
|
void
|
2000-05-12 23:15:20 +00:00
|
|
|
gas_cgen_init_parse ()
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
|
|
|
num_fixups = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Queue a fixup. */
|
|
|
|
|
1998-09-06 22:57:45 +00:00
|
|
|
static void
|
2000-05-12 23:15:20 +00:00
|
|
|
queue_fixup (opindex, opinfo, expP)
|
1998-09-06 22:57:45 +00:00
|
|
|
int opindex;
|
2001-11-01 09:24:29 +00:00
|
|
|
int opinfo;
|
1998-09-06 22:57:45 +00:00
|
|
|
expressionS * expP;
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
|
|
|
/* We need to generate a fixup for this expression. */
|
2000-05-12 23:15:20 +00:00
|
|
|
if (num_fixups >= GAS_CGEN_MAX_FIXUPS)
|
|
|
|
as_fatal (_("too many fixups"));
|
2001-11-01 09:24:29 +00:00
|
|
|
fixups[num_fixups].exp = *expP;
|
1998-03-01 22:58:51 +00:00
|
|
|
fixups[num_fixups].opindex = opindex;
|
1998-09-06 22:57:45 +00:00
|
|
|
fixups[num_fixups].opinfo = opinfo;
|
|
|
|
++ num_fixups;
|
|
|
|
}
|
|
|
|
|
2002-01-27 12:00:11 +00:00
|
|
|
/* The following functions allow fixup chains to be stored, retrieved,
|
|
|
|
and swapped. They are a generalization of a pre-existing scheme
|
|
|
|
for storing, restoring and swapping fixup chains that was used by
|
|
|
|
the m32r port. The functionality is essentially the same, only
|
|
|
|
instead of only being able to store a single fixup chain, an entire
|
|
|
|
array of fixup chains can be stored. It is the user's responsibility
|
|
|
|
to keep track of how many fixup chains have been stored and which
|
|
|
|
elements of the array they are in.
|
|
|
|
|
2002-10-11 06:01:20 +00:00
|
|
|
The algorithms used are the same as in the old scheme. Other than the
|
|
|
|
"array-ness" of the whole thing, the functionality is identical to the
|
2002-01-27 12:00:11 +00:00
|
|
|
old scheme.
|
|
|
|
|
|
|
|
gas_cgen_initialize_saved_fixups_array():
|
|
|
|
Sets num_fixups_in_chain to 0 for each element. Call this from
|
|
|
|
md_begin() if you plan to use these functions and you want the
|
|
|
|
fixup count in each element to be set to 0 intially. This is
|
|
|
|
not necessary, but it's included just in case. It performs
|
|
|
|
the same function for each element in the array of fixup chains
|
|
|
|
that gas_init_parse() performs for the current fixups.
|
|
|
|
|
|
|
|
gas_cgen_save_fixups (element):
|
|
|
|
element - element number of the array you wish to store the fixups
|
|
|
|
to. No mechanism is built in for tracking what element
|
|
|
|
was last stored to.
|
|
|
|
|
|
|
|
gas_cgen_restore_fixups (element):
|
|
|
|
element - element number of the array you wish to restore the fixups
|
|
|
|
from.
|
|
|
|
|
|
|
|
gas_cgen_swap_fixups(int element):
|
|
|
|
element - swap the current fixups with those in this element number.
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct saved_fixups {
|
|
|
|
struct fixup fixup_chain[GAS_CGEN_MAX_FIXUPS];
|
|
|
|
int num_fixups_in_chain;
|
|
|
|
};
|
2000-05-12 23:15:20 +00:00
|
|
|
|
2002-01-27 12:00:11 +00:00
|
|
|
static struct saved_fixups stored_fixups[MAX_SAVED_FIXUP_CHAINS];
|
1998-09-06 22:57:45 +00:00
|
|
|
|
|
|
|
void
|
2002-01-27 12:00:11 +00:00
|
|
|
gas_cgen_initialize_saved_fixups_array ()
|
1998-09-06 22:57:45 +00:00
|
|
|
{
|
2002-01-27 12:00:11 +00:00
|
|
|
int i = 0;
|
1998-09-06 22:57:45 +00:00
|
|
|
|
2002-01-27 12:00:11 +00:00
|
|
|
while (i < MAX_SAVED_FIXUP_CHAINS)
|
|
|
|
stored_fixups[i++].num_fixups_in_chain = 0;
|
1998-09-06 22:57:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-01-27 12:00:11 +00:00
|
|
|
gas_cgen_save_fixups (i)
|
|
|
|
int i;
|
1998-09-06 22:57:45 +00:00
|
|
|
{
|
2002-01-27 12:00:11 +00:00
|
|
|
if (i < 0 || i >= MAX_SAVED_FIXUP_CHAINS)
|
|
|
|
{
|
|
|
|
as_fatal ("index into stored_fixups[] out of bounds");
|
|
|
|
return;
|
|
|
|
}
|
1998-09-06 22:57:45 +00:00
|
|
|
|
2002-01-27 12:00:11 +00:00
|
|
|
stored_fixups[i].num_fixups_in_chain = num_fixups;
|
|
|
|
memcpy (stored_fixups[i].fixup_chain, fixups,
|
|
|
|
sizeof (fixups[0]) * num_fixups);
|
|
|
|
num_fixups = 0;
|
1998-09-06 22:57:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-01-27 12:00:11 +00:00
|
|
|
gas_cgen_restore_fixups (i)
|
|
|
|
int i;
|
1998-09-06 22:57:45 +00:00
|
|
|
{
|
2002-01-27 12:00:11 +00:00
|
|
|
if (i < 0 || i >= MAX_SAVED_FIXUP_CHAINS)
|
1998-09-06 22:57:45 +00:00
|
|
|
{
|
2002-01-27 12:00:11 +00:00
|
|
|
as_fatal ("index into stored_fixups[] out of bounds");
|
|
|
|
return;
|
1998-09-06 22:57:45 +00:00
|
|
|
}
|
2002-01-27 12:00:11 +00:00
|
|
|
|
|
|
|
num_fixups = stored_fixups[i].num_fixups_in_chain;
|
2002-10-11 06:01:20 +00:00
|
|
|
memcpy (fixups, stored_fixups[i].fixup_chain,
|
2002-01-27 12:00:11 +00:00
|
|
|
(sizeof (stored_fixups[i].fixup_chain[0])) * num_fixups);
|
|
|
|
stored_fixups[i].num_fixups_in_chain = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gas_cgen_swap_fixups (i)
|
|
|
|
int i;
|
|
|
|
{
|
|
|
|
if (i < 0 || i >= MAX_SAVED_FIXUP_CHAINS)
|
1998-09-06 22:57:45 +00:00
|
|
|
{
|
2002-01-27 12:00:11 +00:00
|
|
|
as_fatal ("index into stored_fixups[] out of bounds");
|
|
|
|
return;
|
1998-09-06 22:57:45 +00:00
|
|
|
}
|
2002-01-27 12:00:11 +00:00
|
|
|
|
|
|
|
if (num_fixups == 0)
|
|
|
|
gas_cgen_restore_fixups (i);
|
|
|
|
|
|
|
|
else if (stored_fixups[i].num_fixups_in_chain == 0)
|
|
|
|
gas_cgen_save_fixups (i);
|
|
|
|
|
1998-09-06 22:57:45 +00:00
|
|
|
else
|
|
|
|
{
|
2002-01-27 12:00:11 +00:00
|
|
|
int tmp;
|
|
|
|
struct fixup tmp_fixup;
|
|
|
|
|
|
|
|
tmp = stored_fixups[i].num_fixups_in_chain;
|
|
|
|
stored_fixups[i].num_fixups_in_chain = num_fixups;
|
1998-09-06 22:57:45 +00:00
|
|
|
num_fixups = tmp;
|
2001-11-01 09:24:29 +00:00
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
for (tmp = GAS_CGEN_MAX_FIXUPS; tmp--;)
|
1998-09-06 22:57:45 +00:00
|
|
|
{
|
2002-01-27 12:00:11 +00:00
|
|
|
tmp_fixup = stored_fixups[i].fixup_chain [tmp];
|
|
|
|
stored_fixups[i].fixup_chain[tmp] = fixups [tmp];
|
|
|
|
fixups [tmp] = tmp_fixup;
|
1998-09-06 22:57:45 +00:00
|
|
|
}
|
|
|
|
}
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Default routine to record a fixup.
|
|
|
|
This is a cover function to fix_new.
|
|
|
|
It exists because we record INSN with the fixup.
|
|
|
|
|
|
|
|
FRAG and WHERE are their respective arguments to fix_new_exp.
|
|
|
|
LENGTH is in bits.
|
|
|
|
OPINFO is something the caller chooses to help in reloc determination.
|
|
|
|
|
|
|
|
At this point we do not use a bfd_reloc_code_real_type for
|
|
|
|
operands residing in the insn, but instead just use the
|
|
|
|
operand index. This lets us easily handle fixups for any
|
2002-01-27 12:00:11 +00:00
|
|
|
operand type. We pick a BFD reloc type in md_apply_fix3. */
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
fixS *
|
2000-05-12 23:15:20 +00:00
|
|
|
gas_cgen_record_fixup (frag, where, insn, length, operand, opinfo, symbol, offset)
|
1998-09-06 22:57:45 +00:00
|
|
|
fragS * frag;
|
|
|
|
int where;
|
|
|
|
const CGEN_INSN * insn;
|
|
|
|
int length;
|
|
|
|
const CGEN_OPERAND * operand;
|
|
|
|
int opinfo;
|
|
|
|
symbolS * symbol;
|
|
|
|
offsetT offset;
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
2001-11-01 09:24:29 +00:00
|
|
|
fixS *fixP;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* It may seem strange to use operand->attrs and not insn->attrs here,
|
|
|
|
but it is the operand that has a pc relative relocation. */
|
|
|
|
|
|
|
|
fixP = fix_new (frag, where, length / 8, symbol, offset,
|
2000-05-12 23:15:20 +00:00
|
|
|
CGEN_OPERAND_ATTR_VALUE (operand, CGEN_OPERAND_PCREL_ADDR),
|
|
|
|
(bfd_reloc_code_real_type)
|
|
|
|
((int) BFD_RELOC_UNUSED
|
|
|
|
+ (int) operand->type));
|
|
|
|
fixP->fx_cgen.insn = insn;
|
|
|
|
fixP->fx_cgen.opinfo = opinfo;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
return fixP;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Default routine to record a fixup given an expression.
|
|
|
|
This is a cover function to fix_new_exp.
|
|
|
|
It exists because we record INSN with the fixup.
|
|
|
|
|
|
|
|
FRAG and WHERE are their respective arguments to fix_new_exp.
|
|
|
|
LENGTH is in bits.
|
|
|
|
OPINFO is something the caller chooses to help in reloc determination.
|
|
|
|
|
|
|
|
At this point we do not use a bfd_reloc_code_real_type for
|
|
|
|
operands residing in the insn, but instead just use the
|
|
|
|
operand index. This lets us easily handle fixups for any
|
2002-01-27 12:00:11 +00:00
|
|
|
operand type. We pick a BFD reloc type in md_apply_fix3. */
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
fixS *
|
2000-05-12 23:15:20 +00:00
|
|
|
gas_cgen_record_fixup_exp (frag, where, insn, length, operand, opinfo, exp)
|
1998-09-06 22:57:45 +00:00
|
|
|
fragS * frag;
|
|
|
|
int where;
|
|
|
|
const CGEN_INSN * insn;
|
|
|
|
int length;
|
|
|
|
const CGEN_OPERAND * operand;
|
|
|
|
int opinfo;
|
|
|
|
expressionS * exp;
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
2001-11-01 09:24:29 +00:00
|
|
|
fixS *fixP;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* It may seem strange to use operand->attrs and not insn->attrs here,
|
|
|
|
but it is the operand that has a pc relative relocation. */
|
|
|
|
|
|
|
|
fixP = fix_new_exp (frag, where, length / 8, exp,
|
2000-05-12 23:15:20 +00:00
|
|
|
CGEN_OPERAND_ATTR_VALUE (operand, CGEN_OPERAND_PCREL_ADDR),
|
|
|
|
(bfd_reloc_code_real_type)
|
|
|
|
((int) BFD_RELOC_UNUSED
|
|
|
|
+ (int) operand->type));
|
|
|
|
fixP->fx_cgen.insn = insn;
|
|
|
|
fixP->fx_cgen.opinfo = opinfo;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
return fixP;
|
|
|
|
}
|
|
|
|
|
1998-09-06 22:57:45 +00:00
|
|
|
/* Used for communication between the next two procedures. */
|
|
|
|
static jmp_buf expr_jmp_buf;
|
2001-11-01 09:24:29 +00:00
|
|
|
static int expr_jmp_buf_p;
|
1998-09-06 22:57:45 +00:00
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* Callback for cgen interface. Parse the expression at *STRP.
|
|
|
|
The result is an error message or NULL for success (in which case
|
|
|
|
*STRP is advanced past the parsed text).
|
|
|
|
WANT is an indication of what the caller is looking for.
|
|
|
|
If WANT == CGEN_ASM_PARSE_INIT the caller is beginning to try to match
|
|
|
|
a table entry with the insn, reset the queued fixups counter.
|
|
|
|
An enum cgen_parse_operand_result is stored in RESULTP.
|
|
|
|
OPINDEX is the operand's table entry index.
|
|
|
|
OPINFO is something the caller chooses to help in reloc determination.
|
|
|
|
The resulting value is stored in VALUEP. */
|
|
|
|
|
|
|
|
const char *
|
2000-05-12 23:15:20 +00:00
|
|
|
gas_cgen_parse_operand (cd, want, strP, opindex, opinfo, resultP, valueP)
|
2001-11-01 09:24:29 +00:00
|
|
|
CGEN_CPU_DESC cd ATTRIBUTE_UNUSED;
|
2000-05-12 23:15:20 +00:00
|
|
|
enum cgen_parse_operand_type want;
|
2001-11-01 09:24:29 +00:00
|
|
|
const char **strP;
|
2000-05-12 23:15:20 +00:00
|
|
|
int opindex;
|
|
|
|
int opinfo;
|
2001-11-01 09:24:29 +00:00
|
|
|
enum cgen_parse_operand_result *resultP;
|
|
|
|
bfd_vma *valueP;
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
1998-09-06 22:57:45 +00:00
|
|
|
#ifdef __STDC__
|
2000-05-12 23:15:20 +00:00
|
|
|
/* These are volatile to survive the setjmp. */
|
|
|
|
char * volatile hold;
|
1998-09-06 22:57:45 +00:00
|
|
|
enum cgen_parse_operand_result * volatile resultP_1;
|
|
|
|
#else
|
2001-11-01 09:24:29 +00:00
|
|
|
static char *hold;
|
|
|
|
static enum cgen_parse_operand_result *resultP_1;
|
1998-09-06 22:57:45 +00:00
|
|
|
#endif
|
2002-01-27 12:00:11 +00:00
|
|
|
const char *errmsg;
|
2000-05-12 23:15:20 +00:00
|
|
|
expressionS exp;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
if (want == CGEN_PARSE_OPERAND_INIT)
|
|
|
|
{
|
2000-05-12 23:15:20 +00:00
|
|
|
gas_cgen_init_parse ();
|
1998-03-01 22:58:51 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
1998-09-06 22:57:45 +00:00
|
|
|
resultP_1 = resultP;
|
1998-03-01 22:58:51 +00:00
|
|
|
hold = input_line_pointer;
|
2001-11-01 09:24:29 +00:00
|
|
|
input_line_pointer = (char *) *strP;
|
1998-09-06 22:57:45 +00:00
|
|
|
|
|
|
|
/* We rely on md_operand to longjmp back to us.
|
2000-05-12 23:15:20 +00:00
|
|
|
This is done via gas_cgen_md_operand. */
|
1998-09-06 22:57:45 +00:00
|
|
|
if (setjmp (expr_jmp_buf) != 0)
|
|
|
|
{
|
2001-11-01 09:24:29 +00:00
|
|
|
expr_jmp_buf_p = 0;
|
1998-09-06 22:57:45 +00:00
|
|
|
input_line_pointer = (char *) hold;
|
2001-11-01 09:24:29 +00:00
|
|
|
*resultP_1 = CGEN_PARSE_OPERAND_RESULT_ERROR;
|
2002-01-27 12:00:11 +00:00
|
|
|
return _("illegal operand");
|
1998-09-06 22:57:45 +00:00
|
|
|
}
|
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
expr_jmp_buf_p = 1;
|
|
|
|
expression (&exp);
|
|
|
|
expr_jmp_buf_p = 0;
|
2002-01-27 12:00:11 +00:00
|
|
|
errmsg = NULL;
|
1998-09-06 22:57:45 +00:00
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
*strP = input_line_pointer;
|
1998-03-01 22:58:51 +00:00
|
|
|
input_line_pointer = hold;
|
|
|
|
|
|
|
|
/* FIXME: Need to check `want'. */
|
|
|
|
|
|
|
|
switch (exp.X_op)
|
|
|
|
{
|
2001-11-01 09:24:29 +00:00
|
|
|
case O_illegal:
|
2000-05-12 23:15:20 +00:00
|
|
|
errmsg = _("illegal operand");
|
2001-11-01 09:24:29 +00:00
|
|
|
*resultP = CGEN_PARSE_OPERAND_RESULT_ERROR;
|
1998-03-01 22:58:51 +00:00
|
|
|
break;
|
2001-11-01 09:24:29 +00:00
|
|
|
case O_absent:
|
2000-05-12 23:15:20 +00:00
|
|
|
errmsg = _("missing operand");
|
2001-11-01 09:24:29 +00:00
|
|
|
*resultP = CGEN_PARSE_OPERAND_RESULT_ERROR;
|
1998-03-01 22:58:51 +00:00
|
|
|
break;
|
2001-11-01 09:24:29 +00:00
|
|
|
case O_constant:
|
|
|
|
*valueP = exp.X_add_number;
|
|
|
|
*resultP = CGEN_PARSE_OPERAND_RESULT_NUMBER;
|
1998-03-01 22:58:51 +00:00
|
|
|
break;
|
2001-11-01 09:24:29 +00:00
|
|
|
case O_register:
|
|
|
|
*valueP = exp.X_add_number;
|
|
|
|
*resultP = CGEN_PARSE_OPERAND_RESULT_REGISTER;
|
1998-03-01 22:58:51 +00:00
|
|
|
break;
|
2001-11-01 09:24:29 +00:00
|
|
|
default:
|
|
|
|
queue_fixup (opindex, opinfo, &exp);
|
|
|
|
*valueP = 0;
|
|
|
|
*resultP = CGEN_PARSE_OPERAND_RESULT_QUEUED;
|
1998-03-01 22:58:51 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return errmsg;
|
|
|
|
}
|
|
|
|
|
1998-09-06 22:57:45 +00:00
|
|
|
/* md_operand handler to catch unrecognized expressions and halt the
|
|
|
|
parsing process so the next entry can be tried.
|
|
|
|
|
|
|
|
??? This could be done differently by adding code to `expression'. */
|
|
|
|
|
|
|
|
void
|
2000-05-12 23:15:20 +00:00
|
|
|
gas_cgen_md_operand (expressionP)
|
2001-11-01 09:24:29 +00:00
|
|
|
expressionS *expressionP ATTRIBUTE_UNUSED;
|
1998-09-06 22:57:45 +00:00
|
|
|
{
|
2001-11-01 09:24:29 +00:00
|
|
|
/* Don't longjmp if we're not called from within cgen_parse_operand(). */
|
|
|
|
if (expr_jmp_buf_p)
|
|
|
|
longjmp (expr_jmp_buf, 1);
|
1998-09-06 22:57:45 +00:00
|
|
|
}
|
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* Finish assembling instruction INSN.
|
|
|
|
BUF contains what we've built up so far.
|
1998-09-06 22:57:45 +00:00
|
|
|
LENGTH is the size of the insn in bits.
|
2000-05-12 23:15:20 +00:00
|
|
|
RELAX_P is non-zero if relaxable insns should be emitted as such.
|
|
|
|
Otherwise they're emitted in non-relaxable forms.
|
|
|
|
The "result" is stored in RESULT if non-NULL. */
|
1998-03-01 22:58:51 +00:00
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
void
|
|
|
|
gas_cgen_finish_insn (insn, buf, length, relax_p, result)
|
2001-11-01 09:24:29 +00:00
|
|
|
const CGEN_INSN *insn;
|
2000-05-12 23:15:20 +00:00
|
|
|
CGEN_INSN_BYTES_PTR buf;
|
|
|
|
unsigned int length;
|
|
|
|
int relax_p;
|
2001-11-01 09:24:29 +00:00
|
|
|
finished_insnS *result;
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
2000-05-12 23:15:20 +00:00
|
|
|
int i;
|
|
|
|
int relax_operand;
|
2001-11-01 09:24:29 +00:00
|
|
|
char *f;
|
1998-03-01 22:58:51 +00:00
|
|
|
unsigned int byte_len = length / 8;
|
|
|
|
|
|
|
|
/* ??? Target foo issues various warnings here, so one might want to provide
|
|
|
|
a hook here. However, our caller is defined in tc-foo.c so there
|
|
|
|
shouldn't be a need for a hook. */
|
2000-05-12 23:15:20 +00:00
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* Write out the instruction.
|
|
|
|
It is important to fetch enough space in one call to `frag_more'.
|
|
|
|
We use (f - frag_now->fr_literal) to compute where we are and we
|
|
|
|
don't want frag_now to change between calls.
|
|
|
|
|
|
|
|
Relaxable instructions: We need to ensure we allocate enough
|
|
|
|
space for the largest insn. */
|
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
if (CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_RELAX))
|
2001-11-01 09:24:29 +00:00
|
|
|
/* These currently shouldn't get here. */
|
|
|
|
abort ();
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* Is there a relaxable insn with the relaxable operand needing a fixup? */
|
|
|
|
|
|
|
|
relax_operand = -1;
|
2000-05-12 23:15:20 +00:00
|
|
|
if (relax_p && CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_RELAXABLE))
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
|
|
|
/* Scan the fixups for the operand affected by relaxing
|
|
|
|
(i.e. the branch address). */
|
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
for (i = 0; i < num_fixups; ++i)
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
2000-05-12 23:15:20 +00:00
|
|
|
if (CGEN_OPERAND_ATTR_VALUE (cgen_operand_lookup_by_num (gas_cgen_cpu_desc, fixups[i].opindex),
|
|
|
|
CGEN_OPERAND_RELAX))
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
|
|
|
relax_operand = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (relax_operand != -1)
|
|
|
|
{
|
2000-05-12 23:15:20 +00:00
|
|
|
int max_len;
|
2001-11-01 09:24:29 +00:00
|
|
|
fragS *old_frag;
|
|
|
|
expressionS *exp;
|
|
|
|
symbolS *sym;
|
|
|
|
offsetT off;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
#ifdef TC_CGEN_MAX_RELAX
|
|
|
|
max_len = TC_CGEN_MAX_RELAX (insn, byte_len);
|
|
|
|
#else
|
|
|
|
max_len = CGEN_MAX_INSN_SIZE;
|
|
|
|
#endif
|
|
|
|
/* Ensure variable part and fixed part are in same fragment. */
|
|
|
|
/* FIXME: Having to do this seems like a hack. */
|
|
|
|
frag_grow (max_len);
|
2000-05-12 23:15:20 +00:00
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* Allocate space for the fixed part. */
|
|
|
|
f = frag_more (byte_len);
|
2000-05-12 23:15:20 +00:00
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* Create a relaxable fragment for this instruction. */
|
|
|
|
old_frag = frag_now;
|
1998-09-06 22:57:45 +00:00
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
exp = &fixups[relax_operand].exp;
|
|
|
|
sym = exp->X_add_symbol;
|
|
|
|
off = exp->X_add_number;
|
|
|
|
if (exp->X_op != O_constant && exp->X_op != O_symbol)
|
|
|
|
{
|
|
|
|
/* Handle complex expressions. */
|
|
|
|
sym = make_expr_symbol (exp);
|
|
|
|
off = 0;
|
|
|
|
}
|
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
frag_var (rs_machine_dependent,
|
|
|
|
max_len - byte_len /* max chars */,
|
|
|
|
0 /* variable part already allocated */,
|
|
|
|
/* FIXME: When we machine generate the relax table,
|
|
|
|
machine generate a macro to compute subtype. */
|
|
|
|
1 /* subtype */,
|
2001-11-01 09:24:29 +00:00
|
|
|
sym,
|
|
|
|
off,
|
1998-03-01 22:58:51 +00:00
|
|
|
f);
|
2000-05-12 23:15:20 +00:00
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* Record the operand number with the fragment so md_convert_frag
|
2000-05-12 23:15:20 +00:00
|
|
|
can use gas_cgen_md_record_fixup to record the appropriate reloc. */
|
1998-09-06 22:57:45 +00:00
|
|
|
old_frag->fr_cgen.insn = insn;
|
|
|
|
old_frag->fr_cgen.opindex = fixups[relax_operand].opindex;
|
|
|
|
old_frag->fr_cgen.opinfo = fixups[relax_operand].opinfo;
|
2000-05-12 23:15:20 +00:00
|
|
|
if (result)
|
|
|
|
result->frag = old_frag;
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
|
|
|
else
|
2000-05-12 23:15:20 +00:00
|
|
|
{
|
|
|
|
f = frag_more (byte_len);
|
|
|
|
if (result)
|
|
|
|
result->frag = frag_now;
|
|
|
|
}
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* If we're recording insns as numbers (rather than a string of bytes),
|
|
|
|
target byte order handling is deferred until now. */
|
2000-05-12 23:15:20 +00:00
|
|
|
#if CGEN_INT_INSN_P
|
|
|
|
cgen_put_insn_value (gas_cgen_cpu_desc, f, length, *buf);
|
1998-03-01 22:58:51 +00:00
|
|
|
#else
|
|
|
|
memcpy (f, buf, byte_len);
|
|
|
|
#endif
|
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
/* Emit DWARF2 debugging information. */
|
|
|
|
dwarf2_emit_insn (byte_len);
|
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* Create any fixups. */
|
|
|
|
for (i = 0; i < num_fixups; ++i)
|
|
|
|
{
|
2000-05-12 23:15:20 +00:00
|
|
|
fixS *fixP;
|
|
|
|
const CGEN_OPERAND *operand =
|
|
|
|
cgen_operand_lookup_by_num (gas_cgen_cpu_desc, fixups[i].opindex);
|
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* Don't create fixups for these. That's done during relaxation.
|
|
|
|
We don't need to test for CGEN_INSN_RELAX as they can't get here
|
|
|
|
(see above). */
|
2000-05-12 23:15:20 +00:00
|
|
|
if (relax_p
|
|
|
|
&& CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_RELAXABLE)
|
|
|
|
&& CGEN_OPERAND_ATTR_VALUE (operand, CGEN_OPERAND_RELAX))
|
1998-03-01 22:58:51 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
#ifndef md_cgen_record_fixup_exp
|
2000-05-12 23:15:20 +00:00
|
|
|
#define md_cgen_record_fixup_exp gas_cgen_record_fixup_exp
|
1998-03-01 22:58:51 +00:00
|
|
|
#endif
|
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
fixP = md_cgen_record_fixup_exp (frag_now, f - frag_now->fr_literal,
|
|
|
|
insn, length, operand,
|
|
|
|
fixups[i].opinfo,
|
|
|
|
&fixups[i].exp);
|
|
|
|
if (result)
|
|
|
|
result->fixups[i] = fixP;
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
1998-09-06 22:57:45 +00:00
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
if (result)
|
|
|
|
{
|
|
|
|
result->num_fixups = num_fixups;
|
|
|
|
result->addr = f;
|
|
|
|
}
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Apply a fixup to the object code. This is called for all the
|
|
|
|
fixups we generated by the call to fix_new_exp, above. In the call
|
|
|
|
above we used a reloc code which was the largest legal reloc code
|
|
|
|
plus the operand index. Here we undo that to recover the operand
|
|
|
|
index. At this point all symbol values should be fully resolved,
|
|
|
|
and we attempt to completely resolve the reloc. If we can not do
|
|
|
|
that, we determine the correct reloc code and put it back in the fixup. */
|
|
|
|
|
|
|
|
/* FIXME: This function handles some of the fixups and bfd_install_relocation
|
|
|
|
handles the rest. bfd_install_relocation (or some other bfd function)
|
|
|
|
should handle them all. */
|
|
|
|
|
2002-01-27 12:00:11 +00:00
|
|
|
void
|
|
|
|
gas_cgen_md_apply_fix3 (fixP, valP, seg)
|
1998-09-06 22:57:45 +00:00
|
|
|
fixS * fixP;
|
2002-01-27 12:00:11 +00:00
|
|
|
valueT * valP;
|
2001-11-01 09:24:29 +00:00
|
|
|
segT seg ATTRIBUTE_UNUSED;
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
2001-11-01 09:24:29 +00:00
|
|
|
char *where = fixP->fx_frag->fr_literal + fixP->fx_where;
|
2002-01-27 12:00:11 +00:00
|
|
|
valueT value = * valP;
|
2001-11-01 09:24:29 +00:00
|
|
|
/* Canonical name, since used a lot. */
|
2000-05-12 23:15:20 +00:00
|
|
|
CGEN_CPU_DESC cd = gas_cgen_cpu_desc;
|
2001-11-01 09:24:29 +00:00
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
/* FIXME FIXME FIXME: The value we are passed in *valuep includes
|
|
|
|
the symbol values. Since we are using BFD_ASSEMBLER, if we are
|
|
|
|
doing this relocation the code in write.c is going to call
|
|
|
|
bfd_install_relocation, which is also going to use the symbol
|
|
|
|
value. That means that if the reloc is fully resolved we want to
|
|
|
|
use *valuep since bfd_install_relocation is not being used.
|
|
|
|
However, if the reloc is not fully resolved we do not want to use
|
|
|
|
*valuep, and must use fx_offset instead. However, if the reloc
|
|
|
|
is PC relative, we do want to use *valuep since it includes the
|
|
|
|
result of md_pcrel_from. This is confusing. */
|
|
|
|
|
|
|
|
if (fixP->fx_addsy == (symbolS *) NULL)
|
2002-01-27 12:00:11 +00:00
|
|
|
fixP->fx_done = 1;
|
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
else if (fixP->fx_pcrel)
|
2002-01-27 12:00:11 +00:00
|
|
|
;
|
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
value = fixP->fx_offset;
|
2002-01-27 12:00:11 +00:00
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
if (fixP->fx_subsy != (symbolS *) NULL)
|
|
|
|
{
|
|
|
|
if (S_GET_SEGMENT (fixP->fx_subsy) == absolute_section)
|
|
|
|
value -= S_GET_VALUE (fixP->fx_subsy);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* We don't actually support subtracting a symbol. */
|
2001-11-01 09:24:29 +00:00
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
2000-05-12 23:15:20 +00:00
|
|
|
_("expression too complex"));
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((int) fixP->fx_r_type >= (int) BFD_RELOC_UNUSED)
|
|
|
|
{
|
2000-05-12 23:15:20 +00:00
|
|
|
int opindex = (int) fixP->fx_r_type - (int) BFD_RELOC_UNUSED;
|
|
|
|
const CGEN_OPERAND *operand = cgen_operand_lookup_by_num (cd, opindex);
|
|
|
|
const char *errmsg;
|
1998-03-01 22:58:51 +00:00
|
|
|
bfd_reloc_code_real_type reloc_type;
|
2000-05-12 23:15:20 +00:00
|
|
|
CGEN_FIELDS *fields = alloca (CGEN_CPU_SIZEOF_FIELDS (cd));
|
|
|
|
const CGEN_INSN *insn = fixP->fx_cgen.insn;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* If the reloc has been fully resolved finish the operand here. */
|
|
|
|
/* FIXME: This duplicates the capabilities of code in BFD. */
|
|
|
|
if (fixP->fx_done
|
|
|
|
/* FIXME: If partial_inplace isn't set bfd_install_relocation won't
|
|
|
|
finish the job. Testing for pcrel is a temporary hack. */
|
|
|
|
|| fixP->fx_pcrel)
|
|
|
|
{
|
2000-05-12 23:15:20 +00:00
|
|
|
CGEN_CPU_SET_FIELDS_BITSIZE (cd) (fields, CGEN_INSN_BITSIZE (insn));
|
|
|
|
CGEN_CPU_SET_VMA_OPERAND (cd) (cd, opindex, fields, (bfd_vma) value);
|
|
|
|
|
|
|
|
#if CGEN_INT_INSN_P
|
|
|
|
{
|
|
|
|
CGEN_INSN_INT insn_value =
|
|
|
|
cgen_get_insn_value (cd, where, CGEN_INSN_BITSIZE (insn));
|
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
/* ??? 0 is passed for `pc'. */
|
2000-05-12 23:15:20 +00:00
|
|
|
errmsg = CGEN_CPU_INSERT_OPERAND (cd) (cd, opindex, fields,
|
|
|
|
&insn_value, (bfd_vma) 0);
|
|
|
|
cgen_put_insn_value (cd, where, CGEN_INSN_BITSIZE (insn),
|
|
|
|
insn_value);
|
|
|
|
}
|
|
|
|
#else
|
2001-11-01 09:24:29 +00:00
|
|
|
/* ??? 0 is passed for `pc'. */
|
|
|
|
errmsg = CGEN_CPU_INSERT_OPERAND (cd) (cd, opindex, fields, where,
|
|
|
|
(bfd_vma) 0);
|
2000-05-12 23:15:20 +00:00
|
|
|
#endif
|
1998-03-01 22:58:51 +00:00
|
|
|
if (errmsg)
|
2000-05-12 23:15:20 +00:00
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line, "%s", errmsg);
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (fixP->fx_done)
|
2002-01-27 12:00:11 +00:00
|
|
|
return;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* The operand isn't fully resolved. Determine a BFD reloc value
|
|
|
|
based on the operand information and leave it to
|
|
|
|
bfd_install_relocation. Note that this doesn't work when
|
|
|
|
partial_inplace == false. */
|
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
reloc_type = md_cgen_lookup_reloc (insn, operand, fixP);
|
2002-01-27 12:00:11 +00:00
|
|
|
|
1998-03-01 22:58:51 +00:00
|
|
|
if (reloc_type != BFD_RELOC_NONE)
|
2002-01-27 12:00:11 +00:00
|
|
|
fixP->fx_r_type = reloc_type;
|
1998-03-01 22:58:51 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
2000-05-12 23:15:20 +00:00
|
|
|
_("unresolved expression that must be resolved"));
|
1998-03-01 22:58:51 +00:00
|
|
|
fixP->fx_done = 1;
|
2002-01-27 12:00:11 +00:00
|
|
|
return;
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (fixP->fx_done)
|
|
|
|
{
|
|
|
|
/* We're finished with this fixup. Install it because
|
|
|
|
bfd_install_relocation won't be called to do it. */
|
|
|
|
switch (fixP->fx_r_type)
|
|
|
|
{
|
|
|
|
case BFD_RELOC_8:
|
|
|
|
md_number_to_chars (where, value, 1);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_16:
|
|
|
|
md_number_to_chars (where, value, 2);
|
|
|
|
break;
|
|
|
|
case BFD_RELOC_32:
|
|
|
|
md_number_to_chars (where, value, 4);
|
|
|
|
break;
|
2001-11-01 09:24:29 +00:00
|
|
|
case BFD_RELOC_64:
|
|
|
|
md_number_to_chars (where, value, 8);
|
|
|
|
break;
|
1998-03-01 22:58:51 +00:00
|
|
|
default:
|
2000-05-12 23:15:20 +00:00
|
|
|
as_bad_where (fixP->fx_file, fixP->fx_line,
|
|
|
|
_("internal error: can't install fix for reloc type %d (`%s')"),
|
|
|
|
fixP->fx_r_type, bfd_get_reloc_code_name (fixP->fx_r_type));
|
|
|
|
break;
|
1998-03-01 22:58:51 +00:00
|
|
|
}
|
|
|
|
}
|
2002-01-27 12:00:11 +00:00
|
|
|
/* else
|
|
|
|
bfd_install_relocation will be called to finish things up. */
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
/* Tuck `value' away for use by tc_gen_reloc.
|
|
|
|
See the comment describing fx_addnumber in write.h.
|
|
|
|
This field is misnamed (or misused :-). */
|
|
|
|
fixP->fx_addnumber = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Translate internal representation of relocation info to BFD target format.
|
|
|
|
|
|
|
|
FIXME: To what extent can we get all relevant targets to use this? */
|
|
|
|
|
|
|
|
arelent *
|
2000-05-12 23:15:20 +00:00
|
|
|
gas_cgen_tc_gen_reloc (section, fixP)
|
2001-11-01 09:24:29 +00:00
|
|
|
asection * section ATTRIBUTE_UNUSED;
|
1998-09-06 22:57:45 +00:00
|
|
|
fixS * fixP;
|
1998-03-01 22:58:51 +00:00
|
|
|
{
|
2001-11-01 09:24:29 +00:00
|
|
|
arelent *reloc;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
reloc = (arelent *) xmalloc (sizeof (arelent));
|
1998-03-01 22:58:51 +00:00
|
|
|
|
|
|
|
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,
|
2001-11-01 09:24:29 +00:00
|
|
|
_("relocation is not supported"));
|
1998-03-01 22:58:51 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert (!fixP->fx_pcrel == !reloc->howto->pc_relative);
|
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
reloc->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
|
|
|
|
*reloc->sym_ptr_ptr = symbol_get_bfdsym (fixP->fx_addsy);
|
|
|
|
|
2001-11-01 09:24:29 +00:00
|
|
|
/* Use fx_offset for these cases. */
|
|
|
|
if (fixP->fx_r_type == BFD_RELOC_VTABLE_ENTRY
|
2000-05-12 23:15:20 +00:00
|
|
|
|| fixP->fx_r_type == BFD_RELOC_VTABLE_INHERIT)
|
2001-11-01 09:24:29 +00:00
|
|
|
reloc->addend = fixP->fx_offset;
|
2000-05-12 23:15:20 +00:00
|
|
|
else
|
2001-11-01 09:24:29 +00:00
|
|
|
reloc->addend = fixP->fx_addnumber;
|
1998-03-01 22:58:51 +00:00
|
|
|
|
2000-05-12 23:15:20 +00:00
|
|
|
reloc->address = fixP->fx_frag->fr_address + fixP->fx_where;
|
1998-03-01 22:58:51 +00:00
|
|
|
return reloc;
|
|
|
|
}
|