aboutsummaryrefslogtreecommitdiff
path: root/gas/config/tc-mips.c
diff options
context:
space:
mode:
Diffstat (limited to 'gas/config/tc-mips.c')
-rw-r--r--gas/config/tc-mips.c123
1 files changed, 77 insertions, 46 deletions
diff --git a/gas/config/tc-mips.c b/gas/config/tc-mips.c
index c34cdf4..7363901 100644
--- a/gas/config/tc-mips.c
+++ b/gas/config/tc-mips.c
@@ -1726,6 +1726,7 @@ macro (ip)
offsetT maxnum;
bfd_reloc_code_real_type r;
char *p;
+ int hold_mips_optimize;
treg = (ip->insn_opcode >> 16) & 0x1f;
dreg = (ip->insn_opcode >> 11) & 0x1f;
@@ -2542,7 +2543,13 @@ macro (ip)
off1 = -8;
}
+ /* Set mips_optimize around the lui instruction to avoid
+ inserting an unnecessary nop after the lw. */
+ hold_mips_optimize = mips_optimize;
+ mips_optimize = 2;
macro_build_lui ((char *) NULL, &icnt, &expr1, AT);
+ mips_optimize = hold_mips_optimize;
+
macro_build ((char *) NULL, &icnt, &expr1,
mips_isa < 3 ? "addiu" : "daddiu",
"t,r,j", AT, AT, (int) BFD_RELOC_LO16);
@@ -2989,21 +2996,8 @@ macro (ip)
return;
case M_LI_SS:
- if (mips_pic == NO_PIC)
+ if (imm_expr.X_op == O_constant)
{
- assert (offset_expr.X_op == O_symbol
- && strcmp (segment_name (S_GET_SEGMENT
- (offset_expr.X_add_symbol)),
- ".lit4") == 0
- && offset_expr.X_add_number == 0);
- macro_build ((char *) NULL, &icnt, &offset_expr, "lwc1", "T,o(b)",
- treg, (int) BFD_RELOC_MIPS_LITERAL, GP);
- return;
- }
- else if (mips_pic == SVR4_PIC
- || mips_pic == EMBEDDED_PIC)
- {
- assert (imm_expr.X_op == O_constant);
load_register (&icnt, AT, &imm_expr);
macro_build ((char *) NULL, &icnt, (expressionS *) NULL,
"mtc1", "t,G", AT, treg);
@@ -3011,7 +3005,13 @@ macro (ip)
}
else
{
- abort ();
+ assert (offset_expr.X_op == O_symbol
+ && strcmp (segment_name (S_GET_SEGMENT
+ (offset_expr.X_add_symbol)),
+ ".lit4") == 0
+ && offset_expr.X_add_number == 0);
+ macro_build ((char *) NULL, &icnt, &offset_expr, "lwc1", "T,o(b)",
+ treg, (int) BFD_RELOC_MIPS_LITERAL, GP);
return;
}
@@ -3068,15 +3068,11 @@ macro (ip)
break;
case M_LI_DD:
- if (mips_pic == NO_PIC
- || mips_pic == EMBEDDED_PIC)
+ assert (offset_expr.X_op == O_symbol
+ && offset_expr.X_add_number == 0);
+ s = segment_name (S_GET_SEGMENT (offset_expr.X_add_symbol));
+ if (strcmp (s, ".lit8") == 0)
{
- /* Load a floating point number from the .lit8 section. */
- assert (offset_expr.X_op == O_symbol
- && strcmp (segment_name (S_GET_SEGMENT
- (offset_expr.X_add_symbol)),
- ".lit8") == 0
- && offset_expr.X_add_number == 0);
if (mips_isa >= 2)
{
macro_build ((char *) NULL, &icnt, &offset_expr, "ldc1",
@@ -3087,24 +3083,29 @@ macro (ip)
r = BFD_RELOC_MIPS_LITERAL;
goto dob;
}
- else if (mips_pic == SVR4_PIC)
+ else
{
- /* Load the double from the .rdata section. */
- macro_build ((char *) NULL, &icnt, &offset_expr,
- mips_isa < 3 ? "lw" : "ld",
- "t,o(b)", AT, (int) BFD_RELOC_MIPS_GOT16, GP);
+ assert (strcmp (s, RDATA_SECTION_NAME) == 0);
+ if (mips_pic == SVR4_PIC)
+ macro_build ((char *) NULL, &icnt, &offset_expr,
+ mips_isa < 3 ? "lw" : "ld",
+ "t,o(b)", AT, (int) BFD_RELOC_MIPS_GOT16, GP);
+ else
+ {
+ /* FIXME: This won't work for a 64 bit address. */
+ macro_build_lui ((char *) NULL, &icnt, &offset_expr, AT);
+ }
+
if (mips_isa >= 2)
{
macro_build ((char *) NULL, &icnt, &offset_expr, "ldc1",
- "T,o(b)", treg, (int) BFD_RELOC_LO16, GP);
+ "T,o(b)", treg, (int) BFD_RELOC_LO16, AT);
break;
}
breg = AT;
r = BFD_RELOC_LO16;
goto dob;
}
- else
- abort ();
case M_L_DOB:
/* Even on a big endian machine $fn comes before $fn+1. We have
@@ -3249,9 +3250,16 @@ macro (ip)
coproc ? treg + 1 : treg,
(int) BFD_RELOC_MIPS_GPREL, tempreg);
offset_expr.X_add_number += 4;
+
+ /* Set mips_optimize to 2 to avoid inserting an
+ undesired nop. */
+ hold_mips_optimize = mips_optimize;
+ mips_optimize = 2;
macro_build ((char *) NULL, &icnt, &offset_expr, s, fmt,
coproc ? treg : treg + 1,
(int) BFD_RELOC_MIPS_GPREL, tempreg);
+ mips_optimize = hold_mips_optimize;
+
p = frag_var (rs_machine_dependent, 12 + off, 0,
RELAX_ENCODE (8 + off, 12 + off, 0, 4 + off, 1,
used_at && mips_noat),
@@ -3331,9 +3339,16 @@ macro (ip)
coproc ? treg + 1 : treg,
(int) BFD_RELOC_LO16, AT);
expr1.X_add_number += 4;
+
+ /* Set mips_optimize to 2 to avoid inserting an undesired
+ nop. */
+ hold_mips_optimize = mips_optimize;
+ mips_optimize = 2;
macro_build ((char *) NULL, &icnt, &expr1, s, fmt,
coproc ? treg : treg + 1,
(int) BFD_RELOC_LO16, AT);
+ mips_optimize = hold_mips_optimize;
+
(void) frag_var (rs_machine_dependent, 0, 0,
RELAX_ENCODE (0, 0, -16 - off, -8, 1, 0),
offset_expr.X_add_symbol, (long) 0,
@@ -4449,15 +4464,15 @@ mips_ip (str, ip)
f -- immediate value
l -- .lit4
- When generating SVR4 PIC code, we do not use the
- .lit8 or .lit4 sections at all, in order to
- reserve the entire global offset table. When
- generating embedded PIC code, we use the .lit8
- section but not the .lit4 section (we can do .lit4
- inline easily; we need to put .lit8 somewhere in
- the data segment, and using .lit8 permits the
- linker to eventually combine identical .lit8
- entries). */
+ The .lit4 and .lit8 sections are only used if
+ permitted by the -G argument.
+
+ When generating embedded PIC code, we use the
+ .lit8 section but not the .lit4 section (we can do
+ .lit4 inline easily; we need to put .lit8
+ somewhere in the data segment, and using .lit8
+ permits the linker to eventually combine identical
+ .lit8 entries). */
f64 = *args == 'F' || *args == 'L';
@@ -4477,7 +4492,12 @@ mips_ip (str, ip)
assert (length == (f64 ? 8 : 4));
if (*args == 'f'
- || (mips_pic != NO_PIC && *args == 'l'))
+ || (*args == 'l'
+ && (mips_pic == EMBEDDED_PIC
+#ifdef GPOPT
+ || g_switch_value < 4
+#endif
+ )))
{
imm_expr.X_op = O_constant;
if (byte_order == LITTLE_ENDIAN)
@@ -4505,15 +4525,19 @@ mips_ip (str, ip)
{
default: /* unused default case avoids warnings. */
case 'L':
- newname = (mips_pic != SVR4_PIC
- ? ".lit8"
- : RDATA_SECTION_NAME);
+ newname = ".lit8";
+#ifdef GPOPT
+ if (g_switch_value < 8)
+ newname = RDATA_SECTION_NAME;
+#endif
break;
case 'F':
newname = RDATA_SECTION_NAME;
break;
case 'l':
- assert (mips_pic == NO_PIC);
+#ifdef GPOPT
+ assert (g_switch_value >= 4);
+#endif
newname = ".lit4";
break;
}
@@ -5019,7 +5043,9 @@ md_parse_option (c, arg)
case '4':
if (strcmp (p, "4000") == 0
|| strcmp (p, "4k") == 0
- || strcmp (p, "4K") == 0)
+ || strcmp (p, "4K") == 0
+ || strcmp (p, "4400") == 0
+ || strcmp (p, "4600") == 0)
mips_isa = 3;
break;
@@ -5029,6 +5055,11 @@ md_parse_option (c, arg)
|| strcmp (p, "6K") == 0)
mips_isa = 2;
break;
+
+ case 'o':
+ if (strcmp (p, "orion") == 0)
+ mips_isa = 3;
+ break;
}
if (mips_isa == -1)