aboutsummaryrefslogtreecommitdiff
path: root/gas/config/tc-arm.c
diff options
context:
space:
mode:
authorIan Lance Taylor <ian@airs.com>1997-06-10 15:20:05 +0000
committerIan Lance Taylor <ian@airs.com>1997-06-10 15:20:05 +0000
commitccc30467a614a220a2692bc12fdda64359d19e63 (patch)
treee88677333f90cc2f7e9e769ce535dde708cf15b3 /gas/config/tc-arm.c
parentba383e02dee555bd822e52ff537159ee728fd8fa (diff)
downloadgdb-ccc30467a614a220a2692bc12fdda64359d19e63.zip
gdb-ccc30467a614a220a2692bc12fdda64359d19e63.tar.gz
gdb-ccc30467a614a220a2692bc12fdda64359d19e63.tar.bz2
Tue Jun 10 11:18:09 1997 H.J. Lu <hjl@gnu.ai.mit.edu>
* config/tc-arm.c: Add prototypes for many static functions. (struct asm_opcode ): Add prototypes for parms field. (struct thumb_opcode ): Likewise. (fp_op2): Remove unused flags parameter. (output_inst): Make static. (arm_after_pass_hook): Remove unused ignore parameter. * config/tc-arm.h (arm_after_pass_hook): Declare. (arm_start_line_hook): Declare. (arm_frob_label): Declare.
Diffstat (limited to 'gas/config/tc-arm.c')
-rw-r--r--gas/config/tc-arm.c424
1 files changed, 282 insertions, 142 deletions
diff --git a/gas/config/tc-arm.c b/gas/config/tc-arm.c
index 667fbc7..37736b0 100644
--- a/gas/config/tc-arm.c
+++ b/gas/config/tc-arm.c
@@ -70,9 +70,14 @@
#define FPU_ALL 0xff000000 /* Note this is ~ARM_ANY */
#define FPU_MEMMULTI 0x7f000000 /* Not fpu_core */
+
#ifndef CPU_DEFAULT
+#if defined __thumb__
+#define CPU_DEFAULT (ARM_7 | ARM_THUMB)
+#else
#define CPU_DEFAULT ARM_ALL
#endif
+#endif
#ifndef FPU_DEFAULT
#define FPU_DEFAULT FPU_ALL
@@ -80,8 +85,10 @@
static unsigned long cpu_variant = CPU_DEFAULT | FPU_DEFAULT;
+#ifdef OBJ_COFF
/* Flags stored in private area of BFD COFF structure */
static boolean uses_apcs_26 = false;
+#endif
/* This array holds the chars that always start a comment. If the
pre-processor is disabled, these aren't very useful */
@@ -409,6 +416,41 @@ static void fix_new_arm PARAMS ((fragS *frag, int where,
int pc_rel, int reloc));
static int arm_reg_parse PARAMS ((char **ccp));
static int arm_psr_parse PARAMS ((char **ccp));
+static void symbol_locate PARAMS ((symbolS *, CONST char *, segT,
+ valueT, fragS *));
+static int add_to_lit_pool PARAMS ((void));
+static int validate_immediate PARAMS ((int));
+static int validate_offset_imm PARAMS ((int, int));
+static void opcode_select PARAMS ((int));
+static void end_of_line PARAMS ((char *));
+static int reg_required_here PARAMS ((char **, int));
+static int psr_required_here PARAMS ((char **, int));
+static int psrf_required_here PARAMS ((char **, int));
+static int co_proc_number PARAMS ((char **));
+static int cp_opc_expr PARAMS ((char **, int, int));
+static int cp_reg_required_here PARAMS ((char **, int));
+static int fp_reg_required_here PARAMS ((char **, int));
+static int cp_address_offset PARAMS ((char **));
+static int cp_address_required_here PARAMS ((char **));
+static int my_get_float_expression PARAMS ((char **));
+static int skip_past_comma PARAMS ((char **));
+static int walk_no_bignums PARAMS ((symbolS *));
+static int negate_data_op PARAMS ((unsigned long *,
+ unsigned long));
+static int data_op2 PARAMS ((char **));
+static int fp_op2 PARAMS ((char **));
+static long reg_list PARAMS ((char **));
+static void thumb_load_store PARAMS ((char *, int, int));
+static int decode_shift PARAMS ((char **, int));
+static int ldst_extend PARAMS ((char **, int));
+static void thumb_add_sub PARAMS ((char *, int));
+static void insert_reg PARAMS ((int));
+static void thumb_shift PARAMS ((char *, int));
+static void thumb_mov_compare PARAMS ((char *, int));
+static void set_constant_flonums PARAMS ((void));
+static valueT md_chars_to_number PARAMS ((char *, int));
+static void insert_reg_alias PARAMS ((char *, int));
+static void output_inst PARAMS ((char *));
/* ARM instructions take 4bytes in the object file, Thumb instructions
take 2: */
@@ -428,7 +470,8 @@ struct asm_opcode
CONST char *comp_suffix; /* Compulsory suffix that must follow conds */
CONST struct asm_flg *flags; /* Bits to toggle if flag 'n' set */
unsigned long variants; /* Which CPU variants this exists for */
- void (*parms)(); /* Function to call to parse args */
+ /* Function to call to parse args */
+ void (*parms) PARAMS ((char *, unsigned long));
};
static CONST struct asm_opcode insns[] =
@@ -573,10 +616,13 @@ static CONST struct asm_opcode insns[] =
#define OPCODE_BIC 14
#define OPCODE_MVN 15
+static void do_t_nop PARAMS ((char *operands));
static void do_t_arit PARAMS ((char *operands));
static void do_t_add PARAMS ((char *operands));
static void do_t_asr PARAMS ((char *operands));
-static void do_t_branch PARAMS ((char *operands));
+static void do_t_branch9 PARAMS ((char *operands));
+static void do_t_branch12 PARAMS ((char *operands));
+static void do_t_branch23 PARAMS ((char *operands));
static void do_t_bx PARAMS ((char *operands));
static void do_t_compare PARAMS ((char *operands));
static void do_t_ldmstm PARAMS ((char *operands));
@@ -679,7 +725,8 @@ struct thumb_opcode
CONST char *template; /* Basic string to match */
unsigned long value; /* Basic instruction code */
int size;
- void (*parms)(); /* Function to call to parse args */
+ /* Function to call to parse args */
+ void (*parms) PARAMS ((char *));
};
static CONST struct thumb_opcode tinsns[] =
@@ -688,26 +735,26 @@ static CONST struct thumb_opcode tinsns[] =
{"add", 0x0000, 2, do_t_add},
{"and", 0x4000, 2, do_t_arit},
{"asr", 0x0000, 2, do_t_asr},
- {"b", T_OPCODE_BRANCH, 2, do_t_branch},
- {"beq", 0xd0fe, 2, do_t_branch},
- {"bne", 0xd1fe, 2, do_t_branch},
- {"bcs", 0xd2fe, 2, do_t_branch},
- {"bhs", 0xd2fe, 2, do_t_branch},
- {"bcc", 0xd3fe, 2, do_t_branch},
- {"bul", 0xd3fe, 2, do_t_branch},
- {"blo", 0xd3fe, 2, do_t_branch},
- {"bmi", 0xd4fe, 2, do_t_branch},
- {"bpl", 0xd5fe, 2, do_t_branch},
- {"bvs", 0xd6fe, 2, do_t_branch},
- {"bvc", 0xd7fe, 2, do_t_branch},
- {"bhi", 0xd8fe, 2, do_t_branch},
- {"bls", 0xd9fe, 2, do_t_branch},
- {"bge", 0xdafe, 2, do_t_branch},
- {"blt", 0xdbfe, 2, do_t_branch},
- {"bgt", 0xdcfe, 2, do_t_branch},
- {"ble", 0xddfe, 2, do_t_branch},
+ {"b", T_OPCODE_BRANCH, 2, do_t_branch12},
+ {"beq", 0xd0fe, 2, do_t_branch9},
+ {"bne", 0xd1fe, 2, do_t_branch9},
+ {"bcs", 0xd2fe, 2, do_t_branch9},
+ {"bhs", 0xd2fe, 2, do_t_branch9},
+ {"bcc", 0xd3fe, 2, do_t_branch9},
+ {"bul", 0xd3fe, 2, do_t_branch9},
+ {"blo", 0xd3fe, 2, do_t_branch9},
+ {"bmi", 0xd4fe, 2, do_t_branch9},
+ {"bpl", 0xd5fe, 2, do_t_branch9},
+ {"bvs", 0xd6fe, 2, do_t_branch9},
+ {"bvc", 0xd7fe, 2, do_t_branch9},
+ {"bhi", 0xd8fe, 2, do_t_branch9},
+ {"bls", 0xd9fe, 2, do_t_branch9},
+ {"bge", 0xdafe, 2, do_t_branch9},
+ {"blt", 0xdbfe, 2, do_t_branch9},
+ {"bgt", 0xdcfe, 2, do_t_branch9},
+ {"ble", 0xddfe, 2, do_t_branch9},
{"bic", 0x4380, 2, do_t_arit},
- {"bl", 0xf7fffffe, 4, do_t_branch},
+ {"bl", 0xf7fffffe, 4, do_t_branch23},
{"bx", 0x4700, 2, do_t_bx},
{"cmn", T_OPCODE_CMN, 2, do_t_arit},
{"cmp", 0x0000, 2, do_t_compare},
@@ -740,7 +787,7 @@ static CONST struct thumb_opcode tinsns[] =
{"tst", T_OPCODE_TST, 2, do_t_arit},
/* Pseudo ops: */
{"adr", 0x0000, 2, do_t_adr},
- {"nop", 0x0000, 2, do_nop},
+ {"nop", 0x46C0, 2, do_t_nop}, /* mov r8,r8 */
};
struct reg_entry
@@ -856,7 +903,7 @@ literalT literals[MAX_LITERAL_POOL_SIZE];
int next_literal_pool_place = 0; /* Next free entry in the pool */
int lit_pool_num = 1; /* Next literal pool number */
symbolS *current_poolP = NULL;
-symbolS *symbol_make_empty ();
+symbolS *symbol_make_empty PARAMS ((void));
static int
add_to_lit_pool ()
@@ -896,7 +943,7 @@ add_to_lit_pool ()
return SUCCESS;
}
-/* Can't use symbol_new here, so have to create a symbol and them at
+/* Can't use symbol_new here, so have to create a symbol and then at
a later date assign it a value. Thats what these functions do */
static void
symbol_locate (symbolP, name, segment, valu, frag)
@@ -968,7 +1015,7 @@ symbol_make_empty ()
return symbolP;
}
-
+
/* Check that an immediate is valid, and if so, convert it to the right format
*/
@@ -1194,7 +1241,7 @@ s_code (unused)
break;
default:
- as_bad ("invalid operand to .code directive (%d)", temp);
+ as_bad ("invalid operand to .code directive (%d) (expecting 16 or 32)", temp);
}
}
@@ -2272,9 +2319,8 @@ data_op2 (str)
}
static int
-fp_op2 (str, flags)
+fp_op2 (str)
char **str;
- unsigned long flags;
{
while (**str == ' ')
(*str)++;
@@ -3694,7 +3740,7 @@ thumb_add_sub (str, subtract)
if (Rn != FAIL)
{
/* All register format. */
- if (Rd > 7 || Rs > 7 || Rd > 7)
+ if (Rd > 7 || Rs > 7 || Rn > 7)
{
if (Rs != Rd)
{
@@ -4060,6 +4106,7 @@ thumb_load_store (str, load_store, size)
}
else if (*str == '=')
{
+ /* TODO: We should allow the "ldr Rd,=expr" pseudo op in thumb mode */
abort ();
}
else
@@ -4145,7 +4192,7 @@ thumb_load_store (str, load_store, size)
inst.error = "Invalid offset";
return;
}
- inst.instruction |= offset << 6;
+ inst.instruction |= (offset >> size) << 6;
}
else
inst.reloc.type = BFD_RELOC_ARM_THUMB_OFFSET;
@@ -4169,6 +4216,15 @@ thumb_load_store (str, load_store, size)
end_of_line (str);
}
+static void
+do_t_nop (str)
+ char *str;
+{
+ /* Do nothing */
+ end_of_line (str);
+ return;
+}
+
/* Handle the Format 4 instructions that do not have equivalents in other
formats. That is, ADC, AND, EOR, SBC, ROR, TST, NEG, CMN, ORR, MUL,
BIC and MVN. */
@@ -4240,12 +4296,34 @@ do_t_asr (str)
}
static void
-do_t_branch (str)
+do_t_branch9 (str)
char *str;
{
if (my_get_expression (&inst.reloc.exp, &str))
return;
- inst.reloc.type = BFD_RELOC_ARM_PCREL_BRANCH;
+ inst.reloc.type = BFD_RELOC_THUMB_PCREL_BRANCH9;
+ inst.reloc.pc_rel = 1;
+ end_of_line (str);
+}
+
+static void
+do_t_branch12 (str)
+ char *str;
+{
+ if (my_get_expression (&inst.reloc.exp, &str))
+ return;
+ inst.reloc.type = BFD_RELOC_THUMB_PCREL_BRANCH12;
+ inst.reloc.pc_rel = 1;
+ end_of_line (str);
+}
+
+static void
+do_t_branch23 (str)
+ char *str;
+{
+ if (my_get_expression (&inst.reloc.exp, &str))
+ return;
+ inst.reloc.type = BFD_RELOC_THUMB_PCREL_BRANCH23;
inst.reloc.pc_rel = 1;
end_of_line (str);
}
@@ -4484,7 +4562,7 @@ do_t_adr (str)
char *str;
{
/* This is a pseudo-op of the form "adr rd, label" to be converted
- into a relative address of the form "add rd, pc, #label-.-8" */
+ into a relative address of the form "add rd, pc, #label-.-4" */
while (*str == ' ')
str++;
@@ -4498,7 +4576,7 @@ do_t_adr (str)
}
inst.reloc.type = BFD_RELOC_ARM_THUMB_ADD;
- inst.reloc.exp.X_add_number -= 8; /* PC relative adjust */
+ inst.reloc.exp.X_add_number -= 4; /* PC relative adjust */
inst.reloc.pc_rel = 1;
inst.instruction |= REG_PC; /* Rd is already placed into the instruction */
end_of_line (str);
@@ -4585,7 +4663,7 @@ md_begin ()
#ifdef OBJ_COFF
/* Set the flags in the private structure */
- coff_arm_bfd_set_private_flags (stdoutput, uses_apcs_26 ? F_APCS26 : 0);
+ bfd_set_private_flags (stdoutput, uses_apcs_26 ? F_APCS26 : 0);
#endif
{
@@ -4604,18 +4682,24 @@ md_begin ()
default:
case ARM_6 | ARM_3 | ARM_2: /* Actually no CPU type defined */
+ mach = bfd_mach_arm_4;
+ break;
+
case ARM_7: /* also ARM_6 */
mach = bfd_mach_arm_3;
break;
}
/* Catch special cases */
- if (cpu_variant & ARM_THUMB)
- mach = bfd_mach_arm_4T;
- else if (cpu_variant & ARM_LONGMUL)
- mach = bfd_mach_arm_3M;
- else if (cpu_variant & ARM_ARCH4)
- mach = bfd_mach_arm_4;
+ if (cpu_variant != (FPU_DEFAULT | CPU_DEFAULT))
+ {
+ if (cpu_variant & ARM_THUMB)
+ mach = bfd_mach_arm_4T;
+ else if (cpu_variant & ARM_ARCH4)
+ mach = bfd_mach_arm_4;
+ else if (cpu_variant & ARM_LONGMUL)
+ mach = bfd_mach_arm_3M;
+ }
bfd_set_arch_mach (stdoutput, TARGET_ARCH, mach);
}
@@ -4869,7 +4953,12 @@ md_apply_fix3 (fixP, val, seg)
assert (fixP->fx_r_type < BFD_RELOC_UNUSED);
/* Note whether this will delete the relocation. */
+#if 0 /* patch from REarnshaw to JDavis (disabled for the moment, since it doesn't work fully) */
+ if ((fixP->fx_addsy == 0 || fixP->fx_addsy->sy_value.X_op == O_constant)
+ && !fixP->fx_pcrel)
+#else
if (fixP->fx_addsy == 0 && !fixP->fx_pcrel)
+#endif
fixP->fx_done = 1;
/* If this symbol is in a different section then we need to leave it for
@@ -5010,65 +5099,65 @@ md_apply_fix3 (fixP, val, seg)
break;
case BFD_RELOC_ARM_PCREL_BRANCH:
- if (arm_data->thumb_mode)
- {
- unsigned long newval2;
- newval = md_chars_to_number (buf, THUMB_SIZE);
- if (fixP->fx_size == 4)
- {
- unsigned long diff;
-
- newval2 = md_chars_to_number (buf, THUMB_SIZE);
- diff = ((newval & 0x7ff) << 12) | ((newval2 & 0x7ff) << 1);
- if (diff & 0x400000)
- diff |= ~0x3fffff;
- value += diff;
- if ((value & 0x400000) && ((value & ~0x3fffff) != ~0x3fffff))
- as_bad_where (fixP->fx_file, fixP->fx_line,
- "Branch with link out of range");
-
- newval = (newval & 0xf800) | ((value & 0x7fffff) >> 12);
- newval2 = (newval2 & 0xf800) | ((value & 0xfff) >> 1);
- md_number_to_chars (buf, newval, THUMB_SIZE);
- md_number_to_chars (buf, newval2, THUMB_SIZE);
- }
- else
- {
- if (newval == T_OPCODE_BRANCH)
- {
- unsigned long diff = (newval & 0x7ff) << 1;
- if (diff & 0x800)
- diff |= ~0x7ff;
-
- value += diff;
- if ((value & 0x800) && ((value & ~0x7ff) != ~0x7ff))
- as_bad_where (fixP->fx_file, fixP->fx_line,
- "Branch out of range");
- newval = (newval & 0xf800) | ((value & 0xfff) >> 1);
- }
- else
- {
- unsigned long diff = (newval & 0xff) << 1;
- if (diff & 0x100)
- diff |= ~0xff;
-
- value += diff;
- if ((value & 0x100) && ((value & ~0xff) != ~0xff))
- as_bad_where (fixP->fx_file, fixP->fx_line,
- "Branch out of range");
- newval = (newval & 0xff00) | ((value & 0x1ff) >> 1);
- }
- md_number_to_chars (buf, newval, THUMB_SIZE);
- }
- }
- else
- {
- value = (value >> 2) & 0x00ffffff;
- newval = md_chars_to_number (buf, INSN_SIZE);
- value = (value + (newval & 0x00ffffff)) & 0x00ffffff;
- newval = value | (newval & 0xff000000);
- md_number_to_chars (buf, newval, INSN_SIZE);
- }
+ value = (value >> 2) & 0x00ffffff;
+ newval = md_chars_to_number (buf, INSN_SIZE);
+ value = (value + (newval & 0x00ffffff)) & 0x00ffffff;
+ newval = value | (newval & 0xff000000);
+ md_number_to_chars (buf, newval, INSN_SIZE);
+ break;
+
+ case BFD_RELOC_THUMB_PCREL_BRANCH9: /* conditional branch */
+ newval = md_chars_to_number (buf, THUMB_SIZE);
+ {
+ addressT diff = (newval & 0xff) << 1;
+ if (diff & 0x100)
+ diff |= ~0xff;
+
+ value += diff;
+ if ((value & 0x100) && ((value & ~0xff) != ~0xff))
+ as_bad_where (fixP->fx_file, fixP->fx_line,
+ "Branch out of range");
+ newval = (newval & 0xff00) | ((value & 0x1ff) >> 1);
+ }
+ md_number_to_chars (buf, newval, THUMB_SIZE);
+ break;
+
+ case BFD_RELOC_THUMB_PCREL_BRANCH12: /* unconditional branch */
+ newval = md_chars_to_number (buf, THUMB_SIZE);
+ {
+ addressT diff = (newval & 0x7ff) << 1;
+ if (diff & 0x800)
+ diff |= ~0x7ff;
+
+ value += diff;
+ if ((value & 0x800) && ((value & ~0x7ff) != ~0x7ff))
+ as_bad_where (fixP->fx_file, fixP->fx_line,
+ "Branch out of range");
+ newval = (newval & 0xf800) | ((value & 0xfff) >> 1);
+ }
+ md_number_to_chars (buf, newval, THUMB_SIZE);
+ break;
+
+ case BFD_RELOC_THUMB_PCREL_BRANCH23:
+ newval = md_chars_to_number (buf, THUMB_SIZE);
+ {
+ offsetT newval2;
+ addressT diff;
+
+ newval2 = md_chars_to_number (buf + 2, THUMB_SIZE);
+ diff = ((newval & 0x7ff) << 12) | ((newval2 & 0x7ff) << 1);
+ if (diff & 0x400000)
+ diff |= ~0x3fffff;
+ value += diff;
+ if ((value & 0x400000) && ((value & ~0x3fffff) != ~0x3fffff))
+ as_bad_where (fixP->fx_file, fixP->fx_line,
+ "Branch with link out of range");
+
+ newval = (newval & 0xf800) | ((value & 0x7fffff) >> 12);
+ newval2 = (newval2 & 0xf800) | ((value & 0xfff) >> 1);
+ md_number_to_chars (buf, newval, THUMB_SIZE);
+ md_number_to_chars (buf + 2, newval2, THUMB_SIZE);
+ }
break;
case BFD_RELOC_8:
@@ -5106,19 +5195,22 @@ md_apply_fix3 (fixP, val, seg)
switch (newval >> 12)
{
case 4: /* PC load */
- /* PC loads are somewhat odd, bit 2 of the PC is forced to zero
- for these loads, so we may need to round up the offset if the
- instruction is not word aligned since the final address must
- be. */
+ /* Thumb PC loads are somewhat odd, bit 1 of the PC is
+ forced to zero for these loads, so we will need to round
+ up the offset if the instruction address is not word
+ aligned (since the final address produced must be, and
+ we can only describe word-aligned immediate offsets). */
if ((fixP->fx_frag->fr_address + fixP->fx_where + value) & 3)
as_bad_where (fixP->fx_file, fixP->fx_line,
- "Invalid offset, target not word aligned");
+ "Invalid offset, target not word aligned (0x%08X)",
+ (unsigned int)(fixP->fx_frag->fr_address + fixP->fx_where + value));
if ((value + 2) & ~0x3fe)
as_bad_where (fixP->fx_file, fixP->fx_line,
"Invalid offset");
- /* Round up, since pc will be rounded down. */
+
+ /* Round up, since pc will be rounded down. */
newval |= (value + 2) >> 2;
break;
@@ -5296,6 +5388,9 @@ tc_gen_reloc (section, fixp)
case BFD_RELOC_ARM_PCREL_BRANCH:
case BFD_RELOC_RVA:
+ case BFD_RELOC_THUMB_PCREL_BRANCH9:
+ case BFD_RELOC_THUMB_PCREL_BRANCH12:
+ case BFD_RELOC_THUMB_PCREL_BRANCH23:
code = fixp->fx_r_type;
break;
@@ -5303,47 +5398,56 @@ tc_gen_reloc (section, fixp)
case BFD_RELOC_ARM_HWLITERAL:
/* If this is called then the a literal has been referenced across
a section boundry - possibly due to an implicit dump */
- as_bad ("Literal referenced across section boundry (Implicit dump?)");
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Literal referenced across section boundry (Implicit dump?)");
return NULL;
case BFD_RELOC_ARM_IMMEDIATE:
- as_bad ("Internal_relocation (type %d) not fixed up (IMMEDIATE)"
- , fixp->fx_r_type);
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Internal_relocation (type %d) not fixed up (IMMEDIATE)",
+ fixp->fx_r_type);
return NULL;
case BFD_RELOC_ARM_OFFSET_IMM:
- as_bad ("Internal_relocation (type %d) not fixed up (OFFSET_IMM)"
- , fixp->fx_r_type);
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Internal_relocation (type %d) not fixed up (OFFSET_IMM)",
+ fixp->fx_r_type);
return NULL;
case BFD_RELOC_ARM_OFFSET_IMM8:
- as_bad ("Internal_relocation (type %d) not fixed up (OFFSET_IMM8)"
- , fixp->fx_r_type);
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Internal_relocation (type %d) not fixed up (OFFSET_IMM8)",
+ fixp->fx_r_type);
return NULL;
case BFD_RELOC_ARM_SHIFT_IMM:
- as_bad ("Internal_relocation (type %d) not fixed up (SHIFT_IMM)"
- , fixp->fx_r_type);
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Internal_relocation (type %d) not fixed up (SHIFT_IMM)",
+ fixp->fx_r_type);
return NULL;
case BFD_RELOC_ARM_SWI:
- as_bad ("Internal_relocation (type %d) not fixed up (SWI)"
- , fixp->fx_r_type);
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Internal_relocation (type %d) not fixed up (SWI)",
+ fixp->fx_r_type);
return NULL;
case BFD_RELOC_ARM_MULTI:
- as_bad ("Internal_relocation (type %d) not fixed up (MULTI)"
- , fixp->fx_r_type);
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Internal_relocation (type %d) not fixed up (MULTI)",
+ fixp->fx_r_type);
return NULL;
case BFD_RELOC_ARM_CP_OFF_IMM:
- as_bad ("Internal_relocation (type %d) not fixed up (CP_OFF_IMM)"
- , fixp->fx_r_type);
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Internal_relocation (type %d) not fixed up (CP_OFF_IMM)",
+ fixp->fx_r_type);
return NULL;
case BFD_RELOC_ARM_THUMB_OFFSET:
- as_bad ("Internal_relocation (type %d) not fixed up (THUMB_OFFSET)"
- , fixp->fx_r_type);
+ as_bad_where (fixp->fx_file, fixp->fx_line,
+ "Internal_relocation (type %d) not fixed up (THUMB_OFFSET)",
+ fixp->fx_r_type);
return NULL;
default:
@@ -5389,7 +5493,7 @@ md_estimate_size_before_relax (fragP, segtype)
return (1);
}
-void
+static void
output_inst (str)
char *str;
{
@@ -5402,14 +5506,14 @@ output_inst (str)
}
to = frag_more (inst.size);
- if (thumb_mode && (inst.size > 2))
+ if (thumb_mode && (inst.size > THUMB_SIZE))
{
- md_number_to_chars (to, inst.instruction >> 16, 2);
- to += 2;
- inst.size = 2;
+ assert (inst.size == (2 * THUMB_SIZE));
+ md_number_to_chars (to, inst.instruction >> 16, THUMB_SIZE);
+ md_number_to_chars (to + 2, inst.instruction, THUMB_SIZE);
}
-
- md_number_to_chars (to, inst.instruction, inst.size);
+ else
+ md_number_to_chars (to, inst.instruction, inst.size);
if (inst.reloc.type != BFD_RELOC_NONE)
fix_new_arm (frag_now, to - frag_now->fr_literal,
@@ -5747,17 +5851,18 @@ md_parse_option (c, arg)
cpu_variant = ARM_ALL | FPU_ALL;
return 1;
}
- else if (! strcmp( str, "apcs-32" ))
+#ifdef OBJ_COFF
+ if (! strcmp (str, "apcs-32"))
{
uses_apcs_26 = false;
return 1;
}
- else if (! strcmp( str, "apcs-26" ))
+ else if (! strcmp (str, "apcs-26"))
{
uses_apcs_26 = true;
return 1;
}
-
+#endif
/* Strip off optional "arm" */
if (! strncmp (str, "arm", 3))
str += 3;
@@ -5830,7 +5935,7 @@ md_parse_option (c, arg)
{
case 'a': cpu_variant = (cpu_variant & ~ARM_ANY) | ARM_3; break;
case 0: cpu_variant = (cpu_variant & ~ARM_ANY) | ARM_2; break;
- default: as_bad( "Invalid architecture variant -m%s", arg ); break;
+ default: as_bad ("Invalid architecture variant -m%s", arg); break;
}
break;
@@ -5841,7 +5946,7 @@ md_parse_option (c, arg)
{
case 'm': cpu_variant |= ARM_LONGMUL; break;
case 0: break;
- default: as_bad( "Invalid architecture variant -m%s", arg ); break;
+ default: as_bad ("Invalid architecture variant -m%s", arg); break;
}
break;
@@ -5852,12 +5957,12 @@ md_parse_option (c, arg)
{
case 't': cpu_variant |= ARM_THUMB; break;
case 0: break;
- default: as_bad( "Invalid architecture variant -m%s", arg ); break;
+ default: as_bad ("Invalid architecture variant -m%s", arg); break;
}
break;
default:
- as_bad( "Invalid architecture variant -m%s", arg );
+ as_bad ("Invalid architecture variant -m%s", arg);
break;
}
break;
@@ -5888,8 +5993,11 @@ md_show_usage (fp)
-mall\t\t\tallow any instruction\n\
-mfpa10, -mfpa11\tselect floating point architecture\n\
-mfpe-old\t\tdon't allow floating-point multiple instructions\n\
--mno-fpu\t\tdon't allow any floating-point instructions.\n\
--mapcs-32, -mapcs-26\tspecify which ARM Procedure Calling Standard is in use\n");
+-mno-fpu\t\tdon't allow any floating-point instructions.\n");
+#ifdef OBJ_COFF
+ fprintf (fp,
+"-mapcs-32, -mapcs-26\tspecify which ARM Procedure Calling Standard is in use\n");
+#endif
#ifdef ARM_BI_ENDIAN
fprintf (fp,
"-EB\t\t\tassemble code for a big endian cpu\n\
@@ -5943,8 +6051,7 @@ fix_new_arm (frag, where, size, exp, pc_rel, reloc)
* for this kind of use. We need to dump the literal pool before
* references are made to a null symbol pointer. */
void
-arm_after_pass_hook (ignore)
- asection *ignore;
+arm_after_pass_hook ()
{
if (current_poolP != NULL)
{
@@ -5965,6 +6072,39 @@ arm_frob_label (sym)
symbolS *sym;
{
last_label_seen = sym;
+ ARM_SET_TYPE(sym,thumb_mode);
+}
+
+/* Adjust the symbol table. This marks Thumb symbols as distinct from
+ ARM ones. */
+
+void
+arm_adjust_symtab ()
+{
+ symbolS *sym;
+
+ for (sym = symbol_rootP; sym != NULL; sym = symbol_next (sym))
+ {
+ if (ARM_GET_TYPE(sym)) /* Thumb */
+ {
+ switch (S_GET_STORAGE_CLASS (sym))
+ {
+ case C_EXT:
+ S_SET_STORAGE_CLASS (sym, C_THUMBEXT);
+ break;
+ case C_STAT:
+ S_SET_STORAGE_CLASS (sym, C_THUMBSTAT);
+ break;
+ case C_LABEL:
+ S_SET_STORAGE_CLASS (sym, C_THUMBLABEL);
+ break;
+ default: /* do nothing */
+ break;
+ }
+ }
+ }
+
+ return;
}
int