Compute A-B when A or B is weak.

Similar to r235222, but for the weak symbol case.

In an "ideal" assembler/object format an expression would always refer to the
final value and A-B would only be computed from a section in the same
comdat as A and B with A and B strong.

Unfortunately that is not the case with debug info on ELF, so we need an
heuristic.  Since we need an heuristic, we may as well use the same one as
gas:

* call weak_sym : produces a relocation, even if in the same section.
* A - weak_sym and weak_sym -A: don't produce a relocation if we can
  compute it.

This fixes pr23272 and changes the fix of pr22815 to match what gas does.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@235227 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Rafael Espindola 2015-04-17 21:15:17 +00:00
parent b24498671a
commit fb118bd226
9 changed files with 36 additions and 40 deletions

View File

@ -264,7 +264,6 @@ public:
bool IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm, bool IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm,
const MCSymbolData &DataA, const MCSymbolData &DataA,
const MCSymbolData *DataB,
const MCFragment &FB, const MCFragment &FB,
bool InSet, bool InSet,
bool IsPCRel) const override; bool IsPCRel) const override;

View File

@ -94,7 +94,6 @@ public:
virtual bool IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm, virtual bool IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm,
const MCSymbolData &DataA, const MCSymbolData &DataA,
const MCSymbolData *DataB,
const MCFragment &FB, const MCFragment &FB,
bool InSet, bool InSet,
bool IsPCRel) const; bool IsPCRel) const;

View File

@ -275,7 +275,6 @@ class ELFObjectWriter : public MCObjectWriter {
bool bool
IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm, IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm,
const MCSymbolData &DataA, const MCSymbolData &DataA,
const MCSymbolData *DataB,
const MCFragment &FB, const MCFragment &FB,
bool InSet, bool InSet,
bool IsPCRel) const override; bool IsPCRel) const override;
@ -1669,13 +1668,15 @@ void ELFObjectWriter::WriteObject(MCAssembler &Asm,
} }
bool ELFObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl( bool ELFObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl(
const MCAssembler &Asm, const MCSymbolData &DataA, const MCAssembler &Asm, const MCSymbolData &DataA, const MCFragment &FB,
const MCSymbolData *DataB, const MCFragment &FB, bool InSet, bool InSet, bool IsPCRel) const {
bool IsPCRel) const { if (IsPCRel) {
if (!InSet && (::isWeak(DataA) || (DataB && ::isWeak(*DataB)))) assert(!InSet);
if (::isWeak(DataA))
return false; return false;
return MCObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl( }
Asm, DataA, DataB, FB, InSet, IsPCRel); return MCObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl(Asm, DataA, FB,
InSet, IsPCRel);
} }
bool ELFObjectWriter::isWeak(const MCSymbolData &SD) const { bool ELFObjectWriter::isWeak(const MCSymbolData &SD) const {

View File

@ -504,7 +504,7 @@ bool MCAssembler::evaluateFixup(const MCAsmLayout &Layout,
} else { } else {
const MCSymbolData &DataA = getSymbolData(SA); const MCSymbolData &DataA = getSymbolData(SA);
IsResolved = getWriter().IsSymbolRefDifferenceFullyResolvedImpl( IsResolved = getWriter().IsSymbolRefDifferenceFullyResolvedImpl(
*this, DataA, nullptr, *DF, false, true); *this, DataA, *DF, false, true);
} }
} }
} else { } else {

View File

@ -36,13 +36,12 @@ bool MCObjectWriter::IsSymbolRefDifferenceFullyResolved(
return false; return false;
return IsSymbolRefDifferenceFullyResolvedImpl( return IsSymbolRefDifferenceFullyResolvedImpl(
Asm, DataA, &DataB, *DataB.getFragment(), InSet, false); Asm, DataA, *DataB.getFragment(), InSet, false);
} }
bool MCObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl( bool MCObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl(
const MCAssembler &Asm, const MCSymbolData &DataA, const MCAssembler &Asm, const MCSymbolData &DataA, const MCFragment &FB,
const MCSymbolData *DataB, const MCFragment &FB, bool InSet, bool InSet, bool IsPCRel) const {
bool IsPCRel) const {
const MCSection &SecA = DataA.getSymbol().getSection(); const MCSection &SecA = DataA.getSymbol().getSection();
const MCSection &SecB = FB.getParent()->getSection(); const MCSection &SecB = FB.getParent()->getSection();
// On ELF and COFF A - B is absolute if A and B are in the same section. // On ELF and COFF A - B is absolute if A and B are in the same section.

View File

@ -669,13 +669,9 @@ void MachObjectWriter::ExecutePostLayoutBinding(MCAssembler &Asm,
BindIndirectSymbols(Asm); BindIndirectSymbols(Asm);
} }
bool MachObjectWriter:: bool MachObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl(
IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm, const MCAssembler &Asm, const MCSymbolData &DataA, const MCFragment &FB,
const MCSymbolData &DataA, bool InSet, bool IsPCRel) const {
const MCSymbolData *DataB,
const MCFragment &FB,
bool InSet,
bool IsPCRel) const {
if (InSet) if (InSet)
return true; return true;

View File

@ -172,7 +172,6 @@ public:
bool IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm, bool IsSymbolRefDifferenceFullyResolvedImpl(const MCAssembler &Asm,
const MCSymbolData &DataA, const MCSymbolData &DataA,
const MCSymbolData *DataB,
const MCFragment &FB, bool InSet, const MCFragment &FB, bool InSet,
bool IsPCRel) const override; bool IsPCRel) const override;
@ -649,17 +648,16 @@ void WinCOFFObjectWriter::ExecutePostLayoutBinding(MCAssembler &Asm,
} }
bool WinCOFFObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl( bool WinCOFFObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl(
const MCAssembler &Asm, const MCSymbolData &DataA, const MCAssembler &Asm, const MCSymbolData &DataA, const MCFragment &FB,
const MCSymbolData *DataB, const MCFragment &FB, bool InSet, bool InSet, bool IsPCRel) const {
bool IsPCRel) const {
// MS LINK expects to be able to replace all references to a function with a // MS LINK expects to be able to replace all references to a function with a
// thunk to implement their /INCREMENTAL feature. Make sure we don't optimize // thunk to implement their /INCREMENTAL feature. Make sure we don't optimize
// away any relocations to functions. // away any relocations to functions.
if ((((DataA.getFlags() & COFF::SF_TypeMask) >> COFF::SF_TypeShift) >> if ((((DataA.getFlags() & COFF::SF_TypeMask) >> COFF::SF_TypeShift) >>
COFF::SCT_COMPLEX_TYPE_SHIFT) == COFF::IMAGE_SYM_DTYPE_FUNCTION) COFF::SCT_COMPLEX_TYPE_SHIFT) == COFF::IMAGE_SYM_DTYPE_FUNCTION)
return false; return false;
return MCObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl( return MCObjectWriter::IsSymbolRefDifferenceFullyResolvedImpl(Asm, DataA, FB,
Asm, DataA, DataB, FB, InSet, IsPCRel); InSet, IsPCRel);
} }
bool WinCOFFObjectWriter::isWeak(const MCSymbolData &SD) const { bool WinCOFFObjectWriter::isWeak(const MCSymbolData &SD) const {

View File

@ -1,6 +1,10 @@
// RUN: not llvm-mc -filetype=obj -triple x86_64-pc-linux-gnu %s -o %t 2>&1 | FileCheck %s // RUN: llvm-mc -filetype=obj -triple x86_64-pc-linux-gnu < %s | llvm-readobj -r | FileCheck %s
// CHECK: error: Cannot represent a subtraction with a weak symbol // CHECK: Relocations [
// CHECK-NEXT: Section ({{.*}}) .rela.text {
// CHECK-NEXT: 0x1D R_X86_64_PC32 f2 0xFFFFFFFFFFFFFFFC
// CHECK-NEXT: }
// CHECK-NEXT: ]
.weak f .weak f
.weak g .weak g
@ -10,3 +14,13 @@ g:
nop nop
.quad g - f .quad g - f
.weak f2
f2:
nop
g2:
nop
.quad g2 - f2
.quad f2 - g2
call f2

View File

@ -1,10 +0,0 @@
// RUN: not llvm-mc -filetype=obj -triple x86_64-pc-linux-gnu %s -o %t 2>&1 | FileCheck %s
// CHECK: error: Cannot represent a subtraction with a weak symbol
.weak f
f:
nop
g:
nop
.quad g - f