Index: ELF/InputSection.cpp =================================================================== --- ELF/InputSection.cpp +++ ELF/InputSection.cpp @@ -209,6 +209,9 @@ case R_RELAX_GOT_PC_NOPIC: return Body.getVA(A); case R_GOT_OFF: + if (Config->EMachine == EM_MIPS) + return Body.getGotOffset() + + Out::Got->getMipsGotAdjustment(&Body) - MipsGPOffset; return Body.getGotOffset() + A; case R_MIPS_GOT_LOCAL_PAGE: // If relocation against MIPS local symbol requires GOT entry, this entry @@ -221,7 +224,7 @@ // have to put them into the "global" part of GOT and use dynamic // linker to determine their actual addresses. That is why we // create GOT entries for them in the "local" part of GOT. - return Out::Got->getMipsLocalEntryOffset(Body.getVA(A)); + return Out::Got->getMipsLocalOffset(Body, A); case R_PPC_OPD: { uint64_t SymVA = Body.getVA(A); // If we have an undefined weak symbol, we might get here with a symbol Index: ELF/OutputSections.h =================================================================== --- ELF/OutputSections.h +++ ELF/OutputSections.h @@ -11,6 +11,7 @@ #define LLD_ELF_OUTPUT_SECTIONS_H #include "Config.h" +#include "Relocations.h" #include "lld/Core/LLVM.h" #include "llvm/ADT/SmallPtrSet.h" @@ -112,12 +113,12 @@ GotSection(); void finalize() override; void writeTo(uint8_t *Buf) override; - void addEntry(SymbolBody &Sym); + void addEntry(SymbolBody &Sym, uintX_t Addend, RelExpr Expr); bool addDynTlsEntry(SymbolBody &Sym); bool addTlsIndex(); - bool empty() const { return MipsLocalEntries == 0 && Entries.empty(); } - uintX_t getMipsLocalEntryOffset(uintX_t EntryValue); + bool empty() const { return MipsPageEntries == 0 && Entries.empty(); } uintX_t getMipsLocalPageOffset(uintX_t Addr); + uintX_t getMipsLocalOffset(const SymbolBody &B, uintX_t Addend) const; uintX_t getGlobalDynAddr(const SymbolBody &B) const; uintX_t getGlobalDynOffset(const SymbolBody &B) const; uintX_t getNumEntries() const { return Entries.size(); } @@ -131,6 +132,10 @@ // Returns the number of entries in the local part of GOT including // the number of reserved entries. This method is MIPS-specific. unsigned getMipsLocalEntriesNum() const; + // Calculate MIPS GOT offset adjustment. Functon take in account number + // of entries in the global and local parts of the GOT and shift offset + // appropriately. + uintX_t getMipsGotAdjustment(const SymbolBody *B) const; uintX_t getTlsIndexVA() { return Base::getVA() + TlsIndexOff; } uint32_t getTlsIndexOff() { return TlsIndexOff; } @@ -142,10 +147,20 @@ private: std::vector Entries; uint32_t TlsIndexOff = -1; - uint32_t MipsLocalEntries = 0; + uint32_t MipsPageEntries = 0; // Output sections referenced by MIPS GOT relocations. llvm::SmallPtrSet *, 10> MipsOutSections; llvm::DenseMap MipsLocalGotPos; + // Entries from the 'local' part of MISP GOT with 'full' addresses. + // We have to track pairs of Symbol and Addends because each such pair + // requires unique GOT entry. + typedef std::pair MipsGotSymAddend; + llvm::DenseMap MipsLocalMap; + std::vector MipsLocal; + // Entries from the 'global` part of MIPS GOT. + std::vector MipsGlobal; + // Write MIPS-specific parts of the GOT. + void writeMipsGot(uint8_t *&Buf); }; template Index: ELF/OutputSections.cpp =================================================================== --- ELF/OutputSections.cpp +++ ELF/OutputSections.cpp @@ -92,7 +92,8 @@ this->Header.sh_addralign = sizeof(uintX_t); } -template void GotSection::addEntry(SymbolBody &Sym) { +template +void GotSection::addEntry(SymbolBody &Sym, uintX_t Addend, RelExpr Expr) { if (Config->EMachine == EM_MIPS) { // For "true" local symbols which can be referenced from the same module // only compiler creates two instructions for address loading: @@ -118,7 +119,7 @@ // // See "Global Offset Table" in Chapter 5: // ftp://www.linux-mips.org/pub/linux/mips/doc/ABI/mipsabi.pdf - if (Sym.isLocal()) { + if (Expr == R_MIPS_GOT_LOCAL_PAGE) { // At this point we do not know final symbol value so to reduce number // of allocated GOT entries do the following trick. Save all output // sections referenced by GOT relocations. Then later in the `finalize` @@ -126,19 +127,17 @@ // section and allocate appropriate number of GOT entries. auto *OutSec = cast>(&Sym)->Section->OutSec; MipsOutSections.insert(OutSec); - return; - } - if (!Sym.isPreemptible()) { - // In case of non-local symbols require an entry in the local part - // of MIPS GOT, we set GotIndex to 1 just to accent that this symbol - // has the GOT entry and escape creation more redundant GOT entries. - // FIXME (simon): We can try to store such symbols in the `Entries` - // container. But in that case we have to sort out that container - // and update GotIndex assigned to symbols. - Sym.GotIndex = 1; - ++MipsLocalEntries; - return; + } else if (Expr == R_MIPS_GOT_LOCAL) { + auto P = MipsLocalMap.insert( + std::make_pair(MipsGotSymAddend(&Sym, Addend), MipsLocalMap.size())); + if (P.second) + MipsLocal.push_back(std::make_pair(&Sym, Addend)); + } else { + Sym.GotIndex = MipsGlobal.size(); + Sym.IsInGlobalMipsGot = true; + MipsGlobal.push_back(&Sym); } + return; } Sym.GotIndex = Entries.size(); Entries.push_back(&Sym); @@ -170,22 +169,27 @@ GotSection::getMipsLocalPageOffset(uintX_t EntryValue) { // Initialize the entry by the %hi(EntryValue) expression // but without right-shifting. - return getMipsLocalEntryOffset((EntryValue + 0x8000) & ~0xffff); -} - -template -typename GotSection::uintX_t -GotSection::getMipsLocalEntryOffset(uintX_t EntryValue) { + EntryValue = (EntryValue + 0x8000) & ~0xffff; // Take into account MIPS GOT header. // See comment in the GotSection::writeTo. size_t NewIndex = MipsLocalGotPos.size() + 2; auto P = MipsLocalGotPos.insert(std::make_pair(EntryValue, NewIndex)); - assert(!P.second || MipsLocalGotPos.size() <= MipsLocalEntries); + assert(!P.second || MipsLocalGotPos.size() <= MipsPageEntries); return (uintX_t)P.first->second * sizeof(uintX_t) - MipsGPOffset; } template typename GotSection::uintX_t +GotSection::getMipsLocalOffset(const SymbolBody &B, + uintX_t Addend) const { + auto It = MipsLocalMap.find(std::make_pair(&B, Addend)); + if (It != MipsLocalMap.end()) + return (MipsPageEntries + It->second) * sizeof(uintX_t) - MipsGPOffset; + fatal("Cannot find GOT entry for the symbol"); +} + +template +typename GotSection::uintX_t GotSection::getGlobalDynAddr(const SymbolBody &B) const { return this->getVA() + B.GlobalDynIndex * sizeof(uintX_t); } @@ -198,12 +202,21 @@ template const SymbolBody *GotSection::getMipsFirstGlobalEntry() const { - return Entries.empty() ? nullptr : Entries.front(); + return MipsGlobal.empty() ? nullptr : MipsGlobal.front(); } template unsigned GotSection::getMipsLocalEntriesNum() const { - return MipsLocalEntries; + return MipsPageEntries + MipsLocal.size(); +} + +template +typename GotSection::uintX_t +GotSection::getMipsGotAdjustment(const SymbolBody *B) const { + uintX_t Off = getMipsLocalEntriesNum() * sizeof(uintX_t); + if (!B || !B->IsInGlobalMipsGot) + Off += MipsGlobal.size() * sizeof(uintX_t); + return Off; } template void GotSection::finalize() { @@ -211,55 +224,69 @@ if (Config->EMachine == EM_MIPS) { // Take into account MIPS GOT header. // See comment in the GotSection::writeTo. - MipsLocalEntries += 2; + MipsPageEntries += 2; for (const OutputSectionBase *OutSec : MipsOutSections) { // Calculate an upper bound of MIPS GOT entries required to store page // addresses of local symbols. We assume the worst case - each 64kb // page of the output section has at least one GOT relocation against it. // Add 0x8000 to the section's size because the page address stored // in the GOT entry is calculated as (value + 0x8000) & ~0xffff. - MipsLocalEntries += (OutSec->getSize() + 0x8000 + 0xfffe) / 0xffff; + MipsPageEntries += (OutSec->getSize() + 0x8000 + 0xfffe) / 0xffff; } - EntriesNum += MipsLocalEntries; + EntriesNum += MipsPageEntries + MipsLocal.size() + MipsGlobal.size(); } this->Header.sh_size = EntriesNum * sizeof(uintX_t); } -template void GotSection::writeTo(uint8_t *Buf) { - if (Config->EMachine == EM_MIPS) { - // Set the MSB of the second GOT slot. This is not required by any - // MIPS ABI documentation, though. - // - // There is a comment in glibc saying that "The MSB of got[1] of a - // gnu object is set to identify gnu objects," and in GNU gold it - // says "the second entry will be used by some runtime loaders". - // But how this field is being used is unclear. - // - // We are not really willing to mimic other linkers behaviors - // without understanding why they do that, but because all files - // generated by GNU tools have this special GOT value, and because - // we've been doing this for years, it is probably a safe bet to - // keep doing this for now. We really need to revisit this to see - // if we had to do this. - auto *P = reinterpret_cast(Buf); - P[1] = uintX_t(1) << (ELFT::Is64Bits ? 63 : 31); - for (std::pair &L : MipsLocalGotPos) { - uint8_t *Entry = Buf + L.second * sizeof(uintX_t); - write(Entry, L.first); - } - Buf += MipsLocalEntries * sizeof(uintX_t); +template void GotSection::writeMipsGot(uint8_t *&Buf) { + // Set the MSB of the second GOT slot. This is not required by any + // MIPS ABI documentation, though. + // + // There is a comment in glibc saying that "The MSB of got[1] of a + // gnu object is set to identify gnu objects," and in GNU gold it + // says "the second entry will be used by some runtime loaders". + // But how this field is being used is unclear. + // + // We are not really willing to mimic other linkers behaviors + // without understanding why they do that, but because all files + // generated by GNU tools have this special GOT value, and because + // we've been doing this for years, it is probably a safe bet to + // keep doing this for now. We really need to revisit this to see + // if we had to do this. + auto *P = reinterpret_cast(Buf); + P[1] = uintX_t(1) << (ELFT::Is64Bits ? 63 : 31); + // Write 'page address' entries to the local part of the GOT. + for (std::pair &L : MipsLocalGotPos) { + uint8_t *Entry = Buf + L.second * sizeof(uintX_t); + write(Entry, L.first); + } + Buf += MipsPageEntries * sizeof(uintX_t); + // Write 'full address' entries to the local part of the GOT. + for (const MipsGotSymAddend &P : MipsLocal) { + uint8_t *Entry = Buf; + Buf += sizeof(uintX_t); + const SymbolBody *B = P.first; + uintX_t VA = B->getVA() + P.second; + write(Entry, VA); + } + // Write entries to the globa; part of the GOT. + for (const SymbolBody *B : MipsGlobal) { + uint8_t *Entry = Buf; + Buf += sizeof(uintX_t); + uintX_t VA = B->getVA(); + write(Entry, VA); } +} + +template void GotSection::writeTo(uint8_t *Buf) { + if (Config->EMachine == EM_MIPS) + writeMipsGot(Buf); for (const SymbolBody *B : Entries) { uint8_t *Entry = Buf; Buf += sizeof(uintX_t); if (!B) continue; - // MIPS has special rules to fill up GOT entries. - // See "Global Offset Table" in Chapter 5 in the following document - // for detailed description: - // ftp://www.linux-mips.org/pub/linux/mips/doc/ABI/mipsabi.pdf - // As the first approach, we can just store addresses for all symbols. - if (Config->EMachine != EM_MIPS && B->isPreemptible()) + if (B->isPreemptible()) continue; // The dynamic linker will take care of it. uintX_t VA = B->getVA(); write(Entry, VA); @@ -1249,10 +1276,8 @@ const std::pair &R) { // Sort entries related to non-local preemptible symbols by GOT indexes. // All other entries go to the first part of GOT in arbitrary order. - bool LIsInLocalGot = !L.first->isInGot() || !L.first->isPreemptible(); - bool RIsInLocalGot = !R.first->isInGot() || !R.first->isPreemptible(); - if (LIsInLocalGot || RIsInLocalGot) - return !RIsInLocalGot; + if (!L.first->IsInGlobalMipsGot || !R.first->IsInGlobalMipsGot) + return !L.first->IsInGlobalMipsGot; return L.first->GotIndex < R.first->GotIndex; } Index: ELF/Relocations.cpp =================================================================== --- ELF/Relocations.cpp +++ ELF/Relocations.cpp @@ -155,7 +155,7 @@ {Target->adjustRelaxExpr(Type, nullptr, R_RELAX_TLS_GD_TO_IE), Type, Offset, Addend, &Body}); if (!Body.isInGot()) { - Out::Got->addEntry(Body); + Out::Got->addEntry(Body, Addend, Expr); Out::RelaDyn->addReloc({Target->TlsGotRel, Out::Got, Body.getGotOffset(), false, &Body, 0}); @@ -466,8 +466,6 @@ // For details see p. 4-19 at // ftp://www.linux-mips.org/pub/linux/mips/doc/ABI/mipsabi.pdf Addend += 4; - if (Expr == R_GOT_OFF) - Addend -= MipsGPOffset; if (Expr == R_GOTREL) { Addend -= MipsGPOffset; if (Body.isLocal()) @@ -575,7 +573,7 @@ // a dynamic relocation. // ftp://www.linux-mips.org/pub/linux/mips/doc/ABI/mipsabi.pdf p.4-19 if (Config->EMachine == EM_MIPS && !Body.isInGot()) - Out::Got->addEntry(Body); + Out::Got->addEntry(Body, Addend, Expr); continue; } @@ -607,7 +605,7 @@ if (refersToGotEntry(Expr)) { if (Body.isInGot()) continue; - Out::Got->addEntry(Body); + Out::Got->addEntry(Body, Addend, Expr); if (Config->EMachine == EM_MIPS) // MIPS ABI has special rules to process GOT entries Index: ELF/Symbols.h =================================================================== --- ELF/Symbols.h +++ ELF/Symbols.h @@ -96,6 +96,8 @@ bool isInPlt() const { return PltIndex != -1U; } bool hasThunk() const { return ThunkIndex != -1U; } + unsigned IsInGlobalMipsGot : 1; + template typename ELFT::uint getVA(typename ELFT::uint Addend = 0) const; Index: ELF/Symbols.cpp =================================================================== --- ELF/Symbols.cpp +++ ELF/Symbols.cpp @@ -92,12 +92,13 @@ SymbolBody::SymbolBody(Kind K, uint32_t NameOffset, uint8_t StOther, uint8_t Type) - : SymbolKind(K), NeedsCopyOrPltAddr(false), IsLocal(true), Type(Type), - StOther(StOther), NameOffset(NameOffset) {} + : IsInGlobalMipsGot(false), SymbolKind(K), NeedsCopyOrPltAddr(false), + IsLocal(true), Type(Type), StOther(StOther), NameOffset(NameOffset) {} SymbolBody::SymbolBody(Kind K, StringRef Name, uint8_t StOther, uint8_t Type) - : SymbolKind(K), NeedsCopyOrPltAddr(false), IsLocal(false), Type(Type), - StOther(StOther), Name({Name.data(), Name.size()}) {} + : IsInGlobalMipsGot(false), SymbolKind(K), NeedsCopyOrPltAddr(false), + IsLocal(false), Type(Type), StOther(StOther), + Name({Name.data(), Name.size()}) {} // Returns true if a symbol can be replaced at load-time by a symbol // with the same name defined in other ELF executable or DSO. @@ -149,8 +150,7 @@ } template typename ELFT::uint SymbolBody::getGotOffset() const { - return (Out::Got->getMipsLocalEntriesNum() + GotIndex) * - sizeof(typename ELFT::uint); + return GotIndex * sizeof(typename ELFT::uint); } template typename ELFT::uint SymbolBody::getGotPltVA() const { Index: test/ELF/mips-64-disp.s =================================================================== --- /dev/null +++ test/ELF/mips-64-disp.s @@ -0,0 +1,89 @@ +# Check R_MIPS_GOT_DISP relocations against various kind of symbols. + +# RUN: llvm-mc -filetype=obj -triple=mips64-unknown-linux \ +# RUN: %p/Inputs/mips-pic.s -o %t.so.o +# RUN: llvm-mc -filetype=obj -triple=mips64-unknown-linux %s -o %t.exe.o +# RUN: ld.lld %t.so.o -shared -o %t.so +# RUN: ld.lld %t.exe.o %t.so -o %t.exe +# RUN: llvm-objdump -d -t %t.exe | FileCheck %s +# RUN: llvm-readobj -r -mips-plt-got %t.exe | FileCheck -check-prefix=GOT %s + +# REQUIRES: mips + +# CHECK: __start: +# CHECK-NEXT: 20000: 24 42 80 40 addiu $2, $2, -32704 +# CHECK-NEXT: 20004: 24 42 80 20 addiu $2, $2, -32736 +# CHECK-NEXT: 20008: 24 42 80 28 addiu $2, $2, -32728 +# CHECK-NEXT: 2000c: 24 42 80 30 addiu $2, $2, -32720 +# CHECK-NEXT: 20010: 24 42 80 38 addiu $2, $2, -32712 + +# CHECK: 0000000000020014 .text 00000000 foo +# CHECK: 0000000000037ff0 .got 00000000 .hidden _gp +# CHECK: 0000000000020000 .text 00000000 __start +# CHECK: 0000000000000000 g F *UND* 00000000 foo1a + +# GOT: Relocations [ +# GOT-NEXT: ] +# GOT-NEXT: Primary GOT { +# GOT-NEXT: Canonical gp value: 0x37FF0 +# GOT-NEXT: Reserved entries [ +# GOT-NEXT: Entry { +# GOT-NEXT: Address: 0x30000 +# GOT-NEXT: Access: -32752 +# GOT-NEXT: Initial: 0x0 +# GOT-NEXT: Purpose: Lazy resolver +# GOT-NEXT: } +# GOT-NEXT: Entry { +# GOT-NEXT: Address: 0x30008 +# GOT-NEXT: Access: -32744 +# GOT-NEXT: Initial: 0x8000000000000000 +# GOT-NEXT: Purpose: Module pointer (GNU extension) +# GOT-NEXT: } +# GOT-NEXT: ] +# GOT-NEXT: Local entries [ +# GOT-NEXT: Entry { +# GOT-NEXT: Address: 0x30010 +# GOT-NEXT: Access: -32736 +# GOT-NEXT: Initial: 0x20014 +# GOT-NEXT: } +# GOT-NEXT: Entry { +# GOT-NEXT: Address: 0x30018 +# GOT-NEXT: Access: -32728 +# GOT-NEXT: Initial: 0x20004 +# GOT-NEXT: } +# GOT-NEXT: Entry { +# GOT-NEXT: Address: 0x30020 +# GOT-NEXT: Access: -32720 +# GOT-NEXT: Initial: 0x20008 +# GOT-NEXT: } +# GOT-NEXT: Entry { +# GOT-NEXT: Address: 0x30028 +# GOT-NEXT: Access: -32712 +# GOT-NEXT: Initial: 0x2000C +# GOT-NEXT: } +# GOT-NEXT: ] +# GOT-NEXT: Global entries [ +# GOT-NEXT: Entry { +# GOT-NEXT: Address: 0x30030 +# GOT-NEXT: Access: -32704 +# GOT-NEXT: Initial: 0x0 +# GOT-NEXT: Value: 0x0 +# GOT-NEXT: Type: Function +# GOT-NEXT: Section: Undefined +# GOT-NEXT: Name: foo1a +# GOT-NEXT: } +# GOT-NEXT: ] +# GOT-NEXT: Number of TLS and multi-GOT entries: 0 +# GOT-NEXT: } + + .text + .global __start +__start: + addiu $v0,$v0,%got_disp(foo1a) # R_MIPS_GOT_DISP + addiu $v0,$v0,%got_disp(foo) # R_MIPS_GOT_DISP + addiu $v0,$v0,%got_disp(.text+4) # R_MIPS_GOT_DISP + addiu $v0,$v0,%got_disp(.text+8) # R_MIPS_GOT_DISP + addiu $v0,$v0,%got_disp(.text+12) # R_MIPS_GOT_DISP + +foo: + nop Index: test/ELF/mips-64-got.s =================================================================== --- test/ELF/mips-64-got.s +++ test/ELF/mips-64-got.s @@ -13,14 +13,15 @@ # CHECK: __start: # CHECK-NEXT: 20000: df 82 80 20 ld $2, -32736($gp) -# CHECK-NEXT: 20004: 64 42 00 14 daddiu $2, $2, 20 -# CHECK-NEXT: 20008: 24 42 80 30 addiu $2, $2, -32720 +# CHECK-NEXT: 20004: 64 42 00 18 daddiu $2, $2, 24 +# CHECK-NEXT: 20008: 24 42 80 38 addiu $2, $2, -32712 # CHECK-NEXT: 2000c: 24 42 80 28 addiu $2, $2, -32728 +# CHECK-NEXT: 20010: 24 42 80 30 addiu $2, $2, -32720 -# CHECK: 0000000000020014 .text 00000000 foo +# CHECK: 0000000000020018 .text 00000000 foo # CHECK: 0000000000037ff0 .got 00000000 .hidden _gp # CHECK: 0000000000020000 .text 00000000 __start -# CHECK: 0000000000020010 .text 00000000 bar +# CHECK: 0000000000020014 .text 00000000 bar # GOT: Relocations [ # GOT-NEXT: ] @@ -49,13 +50,18 @@ # GOT-NEXT: Entry { # GOT-NEXT: Address: 0x30018 # GOT-NEXT: Access: -32728 -# GOT-NEXT: Initial: 0x20010 +# GOT-NEXT: Initial: 0x20014 # GOT-NEXT: } -# GOT-NEXT: ] -# GOT-NEXT: Global entries [ # GOT-NEXT: Entry { # GOT-NEXT: Address: 0x30020 # GOT-NEXT: Access: -32720 +# GOT-NEXT: Initial: 0x20018 +# GOT-NEXT: } +# GOT-NEXT: ] +# GOT-NEXT: Global entries [ +# GOT-NEXT: Entry { +# GOT-NEXT: Address: 0x30028 +# GOT-NEXT: Access: -32712 # GOT-NEXT: Initial: 0x0 # GOT-NEXT: Value: 0x0 # GOT-NEXT: Type: Function @@ -73,6 +79,7 @@ daddiu $v0,$v0,%got_ofst(foo) # R_MIPS_GOT_OFST addiu $v0,$v0,%got_disp(foo1a) # R_MIPS_GOT_DISP addiu $v0,$v0,%got_disp(bar) # R_MIPS_GOT_DISP + addiu $v0,$v0,%got_disp(foo) # R_MIPS_GOT_DISP bar: nop Index: test/ELF/mips-dynamic.s =================================================================== --- test/ELF/mips-dynamic.s +++ test/ELF/mips-dynamic.s @@ -71,8 +71,8 @@ # DSO: ] # DSO: DynamicSymbols [ # DSO: Name: @ -# DSO: Name: __start@ # DSO: Name: _foo@ +# DSO: Name: __start@ # DSO: ] # DSO: DynamicSection [ # DSO-NEXT: Tag Type Name/Value Index: test/ELF/mips-got16.s =================================================================== --- test/ELF/mips-got16.s +++ test/ELF/mips-got16.s @@ -17,7 +17,7 @@ # CHECK-NEXT: 10014: 21 08 90 04 addi $8, $8, -28668 # CHECK-NEXT: 10018: 8f 88 80 20 lw $8, -32736($gp) # CHECK-NEXT: 1001c: 21 08 10 04 addi $8, $8, 4100 -# CHECK-NEXT: 10020: 8f 88 80 24 lw $8, -32732($gp) +# CHECK-NEXT: 10020: 8f 88 80 28 lw $8, -32728($gp) # CHECK-NEXT: 10024: 21 08 10 08 addi $8, $8, 4104 # CHECK-NEXT: 10028: 8f 88 80 2c lw $8, -32724($gp) # @@ -67,14 +67,14 @@ # GOT-NEXT: Entry { # GOT-NEXT: Address: 0x20014 # GOT-NEXT: Access: -32732 -# GOT-NEXT: Initial: 0x51008 -# ^-- 'bar' address +# GOT-NEXT: Initial: 0x0 +# ^-- redundant unused entry # GOT-NEXT: } # GOT-NEXT: Entry { # GOT-NEXT: Address: 0x20018 -# GOT-NEXT: Access: -32728 -# GOT-NEXT: Initial: 0x0 -# ^-- redundant unused entry +# GOT-NEXT: Access: -327 +# GOT-NEXT: Initial: 0x51008 +# ^-- 'bar' address # GOT-NEXT: } # GOT-NEXT: ] # GOT-NEXT: Global entries [ Index: test/ELF/mips-plt-copy.s =================================================================== --- test/ELF/mips-plt-copy.s +++ test/ELF/mips-plt-copy.s @@ -12,8 +12,8 @@ # CHECK: Relocations [ # CHECK-NEXT: Section ({{.*}}) .rel.dyn { -# CHECK-NEXT: 0x{{[0-9A-F]+}} R_MIPS_COPY data0 0x0 # CHECK-NEXT: 0x{{[0-9A-F]+}} R_MIPS_COPY data1 0x0 +# CHECK-NEXT: 0x{{[0-9A-F]+}} R_MIPS_COPY data0 0x0 # CHECK-NEXT: } # CHECK-NEXT: Section ({{.*}}) .rel.plt { # CHECK-NEXT: 0x{{[0-9A-F]+}} R_MIPS_JUMP_SLOT foo0 0x0 Index: test/ELF/mips-sto-plt.s =================================================================== --- test/ELF/mips-sto-plt.s +++ test/ELF/mips-sto-plt.s @@ -10,23 +10,23 @@ # REQUIRES: mips # CHECK: Symbol { -# CHECK: Name: foo0@ -# CHECK-NEXT: Value: 0x0 +# CHECK: Name: foo1@ +# CHECK-NEXT: Value: 0x20050 # CHECK-NEXT: Size: 0 # CHECK-NEXT: Binding: Global # CHECK-NEXT: Type: Function -# CHECK-NEXT: Other: 0 +# CHECK-NEXT: Other [ (0x8) +# CHECK-NEXT: STO_MIPS_PLT +# CHECK-NEXT: ] # CHECK-NEXT: Section: Undefined # CHECK-NEXT: } # CHECK: Symbol { -# CHECK: Name: foo1@ -# CHECK-NEXT: Value: 0x20050 +# CHECK: Name: foo0@ +# CHECK-NEXT: Value: 0x0 # CHECK-NEXT: Size: 0 # CHECK-NEXT: Binding: Global # CHECK-NEXT: Type: Function -# CHECK-NEXT: Other [ (0x8) -# CHECK-NEXT: STO_MIPS_PLT -# CHECK-NEXT: ] +# CHECK-NEXT: Other: 0 # CHECK-NEXT: Section: Undefined # CHECK-NEXT: }