]> git.ipfire.org Git - thirdparty/binutils-gdb.git/commitdiff
Correct GOLD PowerPC64 local-dynamic TLS linker optimization
authorAlan Modra <amodra@gmail.com>
Thu, 29 Jan 2015 10:03:26 +0000 (20:33 +1030)
committerAlan Modra <amodra@gmail.com>
Wed, 11 Feb 2015 12:36:44 +0000 (23:06 +1030)
Similar to b86ac8e3

* powerpc.cc (Target_powerpc::Relocate::relocate): Correct GOT_TLSLD
and GOT_TLSGD to LE optimization.

gold/ChangeLog
gold/powerpc.cc

index b7b2d9a78e66f8a0e35b29b87bf05054f84e6e44..2c4a8c3c007af6f3df7a215477ad7046ef101168 100644 (file)
@@ -1,6 +1,10 @@
 2015-02-11  Alan Modra  <amodra@gmail.com>
 
        Apply from master
+       2015-01-29  Alan Modra  <amodra@gmail.com>
+       * powerpc.cc (Target_powerpc::Relocate::relocate): Correct GOT_TLSLD
+       and GOT_TLSGD to LE optimization.
+
        2015-01-22  Alan Modra  <amodra@gmail.com>
        * powerpc.cc (Target_powerpc::Scan::local <got relocs>): Correct
        condition for need of ifunc plt entry.
index 707acda8b671ce7e01d682d3a9112f5417449871..264bbd82b50dfcf20a30c1c231bf53da8aa8b21b 100644 (file)
@@ -3092,8 +3092,6 @@ static const uint32_t addi_11_11  = 0x396b0000;
 static const uint32_t addi_12_12       = 0x398c0000;
 static const uint32_t addis_0_2                = 0x3c020000;
 static const uint32_t addis_0_13       = 0x3c0d0000;
-static const uint32_t addis_3_2                = 0x3c620000;
-static const uint32_t addis_3_13       = 0x3c6d0000;
 static const uint32_t addis_11_2       = 0x3d620000;
 static const uint32_t addis_11_11      = 0x3d6b0000;
 static const uint32_t addis_11_30      = 0x3d7e0000;
@@ -6921,9 +6919,12 @@ Target_powerpc<size, big_endian>::Relocate::relocate(
              || r_type == elfcpp::R_POWERPC_GOT_TLSGD16_LO)
            {
              Insn* iview = reinterpret_cast<Insn*>(view - 2 * big_endian);
-             Insn insn = addis_3_13;
+             Insn insn = elfcpp::Swap<32, big_endian>::readval(iview);
+             insn &= (1 << 26) - (1 << 21); // extract rt
              if (size == 32)
-               insn = addis_3_2;
+               insn |= addis_0_2;
+             else
+               insn |= addis_0_13;
              elfcpp::Swap<32, big_endian>::writeval(iview, insn);
              r_type = elfcpp::R_POWERPC_TPREL16_HA;
              value = psymval->value(object, rela.get_r_addend());
@@ -6956,9 +6957,12 @@ Target_powerpc<size, big_endian>::Relocate::relocate(
              || r_type == elfcpp::R_POWERPC_GOT_TLSLD16_LO)
            {
              Insn* iview = reinterpret_cast<Insn*>(view - 2 * big_endian);
-             Insn insn = addis_3_13;
+             Insn insn = elfcpp::Swap<32, big_endian>::readval(iview);
+             insn &= (1 << 26) - (1 << 21); // extract rt
              if (size == 32)
-               insn = addis_3_2;
+               insn |= addis_0_2;
+             else
+               insn |= addis_0_13;
              elfcpp::Swap<32, big_endian>::writeval(iview, insn);
              r_type = elfcpp::R_POWERPC_TPREL16_HA;
              value = dtp_offset;