Pack the MCSymbolELF bit fields into MCSymbol's Flags.

This reduces MCSymolfELF from 64 bytes to 56 bytes on x86_64.

While at it, also make getOther/setOther easier to use by accepting unshifted
STO_* values.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@239006 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Rafael Espindola
2015-06-04 02:32:20 +00:00
parent dc967a97df
commit d90fd082f9
7 changed files with 154 additions and 76 deletions

View File

@@ -17,15 +17,9 @@ class MCSymbolELF : public MCSymbol {
/// symbol has no size this field will be NULL. /// symbol has no size this field will be NULL.
const MCExpr *SymbolSize = nullptr; const MCExpr *SymbolSize = nullptr;
mutable unsigned BindingSet : 1;
mutable unsigned UsedInReloc : 1;
mutable unsigned WeakrefUsedInReloc : 1;
mutable unsigned IsSignature : 1;
public: public:
MCSymbolELF(const StringMapEntry<bool> *Name, bool isTemporary) MCSymbolELF(const StringMapEntry<bool> *Name, bool isTemporary)
: MCSymbol(true, Name, isTemporary), BindingSet(false), : MCSymbol(true, Name, isTemporary) {}
UsedInReloc(false), WeakrefUsedInReloc(false), IsSignature(false) {}
void setSize(const MCExpr *SS) { SymbolSize = SS; } void setSize(const MCExpr *SS) { SymbolSize = SS; }
const MCExpr *getSize() const { return SymbolSize; } const MCExpr *getSize() const { return SymbolSize; }
@@ -42,7 +36,7 @@ public:
void setBinding(unsigned Binding) const; void setBinding(unsigned Binding) const;
unsigned getBinding() const; unsigned getBinding() const;
bool isBindingSet() const { return BindingSet; } bool isBindingSet() const;
void setUsedInReloc() const; void setUsedInReloc() const;
bool isUsedInReloc() const; bool isUsedInReloc() const;
@@ -54,6 +48,9 @@ public:
bool isSignature() const; bool isSignature() const;
static bool classof(const MCSymbol *S) { return S->isELF(); } static bool classof(const MCSymbol *S) { return S->isELF(); }
private:
void setIsBindingSet() const;
}; };
} }

View File

@@ -463,8 +463,7 @@ void ELFObjectWriter::writeSymbol(SymbolTableWriter &Writer,
// Other and Visibility share the same byte with Visibility using the lower // Other and Visibility share the same byte with Visibility using the lower
// 2 bits // 2 bits
uint8_t Visibility = Symbol.getVisibility(); uint8_t Visibility = Symbol.getVisibility();
uint8_t Other = Symbol.getOther() << 2; uint8_t Other = Symbol.getOther() | Visibility;
Other |= Visibility;
uint64_t Value = SymbolValue(*MSD.Symbol, Layout); uint64_t Value = SymbolValue(*MSD.Symbol, Layout);
uint64_t Size = 0; uint64_t Size = 0;

View File

@@ -16,27 +16,71 @@ namespace llvm {
namespace { namespace {
enum { enum {
ELF_STT_Shift = 0, // Shift value for STT_* flags // Shift value for STT_* flags. 7 possible values. 3 bits.
ELF_STB_Shift = 4, // Shift value for STB_* flags ELF_STT_Shift = 0,
ELF_STV_Shift = 8, // Shift value for STV_* flags
ELF_STO_Shift = 10 // Shift value for STO_* flags // Shift value for STB_* flags. 4 possible values, 2 bits.
ELF_STB_Shift = 3,
// Shift value for STV_* flags. 4 possible values, 2 bits.
ELF_STV_Shift = 5,
// Shift value for STO_* flags. 3 bits. All the values are between 0x20 and
// 0xe0, so we shift right by 5 before storing.
ELF_STO_Shift = 7,
// One bit.
ELF_IsSignature_Shift = 10,
// One bit.
ELF_WeakrefUsedInReloc_Shift = 11,
// One bit.
ELF_UsedInReloc_Shift = 12,
// One bit.
ELF_BindingSet_Shift = 13
}; };
} }
void MCSymbolELF::setBinding(unsigned Binding) const { void MCSymbolELF::setBinding(unsigned Binding) const {
BindingSet = true; setIsBindingSet();
assert(Binding == ELF::STB_LOCAL || Binding == ELF::STB_GLOBAL || unsigned Val;
Binding == ELF::STB_WEAK || Binding == ELF::STB_GNU_UNIQUE); switch (Binding) {
uint32_t OtherFlags = getFlags() & ~(0xf << ELF_STB_Shift); default:
setFlags(OtherFlags | (Binding << ELF_STB_Shift)); llvm_unreachable("Unsupported Binding");
case ELF::STB_LOCAL:
Val = 0;
break;
case ELF::STB_GLOBAL:
Val = 1;
break;
case ELF::STB_WEAK:
Val = 2;
break;
case ELF::STB_GNU_UNIQUE:
Val = 3;
break;
}
uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STB_Shift);
setFlags(OtherFlags | (Val << ELF_STB_Shift));
} }
unsigned MCSymbolELF::getBinding() const { unsigned MCSymbolELF::getBinding() const {
if (isBindingSet()) { if (isBindingSet()) {
uint32_t Binding = (getFlags() & (0xf << ELF_STB_Shift)) >> ELF_STB_Shift; uint32_t Val = (getFlags() & (0x3 << ELF_STB_Shift)) >> ELF_STB_Shift;
assert(Binding == ELF::STB_LOCAL || Binding == ELF::STB_GLOBAL || switch (Val) {
Binding == ELF::STB_WEAK || Binding == ELF::STB_GNU_UNIQUE); default:
return Binding; llvm_unreachable("Invalid value");
case 0:
return ELF::STB_LOCAL;
case 1:
return ELF::STB_GLOBAL;
case 2:
return ELF::STB_WEAK;
case 3:
return ELF::STB_GNU_UNIQUE;
}
} }
if (isDefined()) if (isDefined())
@@ -51,26 +95,58 @@ unsigned MCSymbolELF::getBinding() const {
} }
void MCSymbolELF::setType(unsigned Type) const { void MCSymbolELF::setType(unsigned Type) const {
assert(Type == ELF::STT_NOTYPE || Type == ELF::STT_OBJECT || unsigned Val;
Type == ELF::STT_FUNC || Type == ELF::STT_SECTION || switch (Type) {
Type == ELF::STT_COMMON || Type == ELF::STT_TLS || default:
Type == ELF::STT_GNU_IFUNC); llvm_unreachable("Unsupported Binding");
case ELF::STT_NOTYPE:
uint32_t OtherFlags = getFlags() & ~(0xf << ELF_STT_Shift); Val = 0;
setFlags(OtherFlags | (Type << ELF_STT_Shift)); break;
case ELF::STT_OBJECT:
Val = 1;
break;
case ELF::STT_FUNC:
Val = 2;
break;
case ELF::STT_SECTION:
Val = 3;
break;
case ELF::STT_COMMON:
Val = 4;
break;
case ELF::STT_TLS:
Val = 5;
break;
case ELF::STT_GNU_IFUNC:
Val = 6;
break;
}
uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STT_Shift);
setFlags(OtherFlags | (Val << ELF_STT_Shift));
} }
unsigned MCSymbolELF::getType() const { unsigned MCSymbolELF::getType() const {
uint32_t Type = (getFlags() & (0xf << ELF_STT_Shift)) >> ELF_STT_Shift; uint32_t Val = (getFlags() & (0x7 << ELF_STT_Shift)) >> ELF_STT_Shift;
assert(Type == ELF::STT_NOTYPE || Type == ELF::STT_OBJECT || switch (Val) {
Type == ELF::STT_FUNC || Type == ELF::STT_SECTION || default:
Type == ELF::STT_COMMON || Type == ELF::STT_TLS || llvm_unreachable("Invalid value");
Type == ELF::STT_GNU_IFUNC); case 0:
return Type; return ELF::STT_NOTYPE;
case 1:
return ELF::STT_OBJECT;
case 2:
return ELF::STT_FUNC;
case 3:
return ELF::STT_SECTION;
case 4:
return ELF::STT_COMMON;
case 5:
return ELF::STT_TLS;
case 6:
return ELF::STT_GNU_IFUNC;
}
} }
// Visibility is stored in the first two bits of st_other
// st_other values are stored in the second byte of get/setFlags
void MCSymbolELF::setVisibility(unsigned Visibility) { void MCSymbolELF::setVisibility(unsigned Visibility) {
assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL || assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL ||
Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED); Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED);
@@ -86,31 +162,52 @@ unsigned MCSymbolELF::getVisibility() const {
return Visibility; return Visibility;
} }
// Other is stored in the last six bits of st_other
// st_other values are stored in the second byte of get/setFlags
void MCSymbolELF::setOther(unsigned Other) { void MCSymbolELF::setOther(unsigned Other) {
uint32_t OtherFlags = getFlags() & ~(0x3f << ELF_STO_Shift); assert((Other & 0x1f) == 0);
Other >>= 5;
assert(Other <= 0x7);
uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STO_Shift);
setFlags(OtherFlags | (Other << ELF_STO_Shift)); setFlags(OtherFlags | (Other << ELF_STO_Shift));
} }
unsigned MCSymbolELF::getOther() const { unsigned MCSymbolELF::getOther() const {
unsigned Other = (getFlags() & (0x3f << ELF_STO_Shift)) >> ELF_STO_Shift; unsigned Other = (getFlags() & (0x3f << ELF_STO_Shift)) >> ELF_STO_Shift;
return Other; return Other << 5;
} }
void MCSymbolELF::setUsedInReloc() const { void MCSymbolELF::setUsedInReloc() const {
UsedInReloc = true; uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_UsedInReloc_Shift);
setFlags(OtherFlags | (1 << ELF_UsedInReloc_Shift));
} }
bool MCSymbolELF::isUsedInReloc() const { bool MCSymbolELF::isUsedInReloc() const {
return UsedInReloc; return getFlags() & (0x1 << ELF_UsedInReloc_Shift);
} }
void MCSymbolELF::setIsWeakrefUsedInReloc() const { WeakrefUsedInReloc = true; } void MCSymbolELF::setIsWeakrefUsedInReloc() const {
uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_WeakrefUsedInReloc_Shift);
bool MCSymbolELF::isWeakrefUsedInReloc() const { return WeakrefUsedInReloc; } setFlags(OtherFlags | (1 << ELF_WeakrefUsedInReloc_Shift));
}
void MCSymbolELF::setIsSignature() const { IsSignature = true; }
bool MCSymbolELF::isWeakrefUsedInReloc() const {
bool MCSymbolELF::isSignature() const { return IsSignature; } return getFlags() & (0x1 << ELF_WeakrefUsedInReloc_Shift);
}
void MCSymbolELF::setIsSignature() const {
uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_IsSignature_Shift);
setFlags(OtherFlags | (1 << ELF_IsSignature_Shift));
}
bool MCSymbolELF::isSignature() const {
return getFlags() & (0x1 << ELF_IsSignature_Shift);
}
void MCSymbolELF::setIsBindingSet() const {
uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_BindingSet_Shift);
setFlags(OtherFlags | (1 << ELF_BindingSet_Shift));
}
bool MCSymbolELF::isBindingSet() const {
return getFlags() & (0x1 << ELF_BindingSet_Shift);
}
} }

View File

@@ -384,7 +384,7 @@ bool MipsELFObjectWriter::needsRelocateWithSymbol(const MCSymbol &Sym,
return true; return true;
case ELF::R_MIPS_32: case ELF::R_MIPS_32:
if (cast<MCSymbolELF>(Sym).getOther() & (ELF::STO_MIPS_MICROMIPS >> 2)) if (cast<MCSymbolELF>(Sym).getOther() & ELF::STO_MIPS_MICROMIPS)
return true; return true;
// falltrough // falltrough
case ELF::R_MIPS_26: case ELF::R_MIPS_26:

View File

@@ -44,10 +44,7 @@ void MipsELFStreamer::createPendingLabelRelocs() {
for (auto *L : Labels) { for (auto *L : Labels) {
auto *Label = cast<MCSymbolELF>(L); auto *Label = cast<MCSymbolELF>(L);
getAssembler().registerSymbol(*Label); getAssembler().registerSymbol(*Label);
// The "other" values are stored in the last 6 bits of the second byte. Label->setOther(ELF::STO_MIPS_MICROMIPS);
// The traditional defines for STO values assume the full byte and thus
// the shift to pack it.
Label->setOther(ELF::STO_MIPS_MICROMIPS >> 2);
} }
} }

View File

@@ -462,10 +462,7 @@ void MipsTargetELFStreamer::emitLabel(MCSymbol *S) {
if (Type != ELF::STT_FUNC) if (Type != ELF::STT_FUNC)
return; return;
// The "other" values are stored in the last 6 bits of the second byte Symbol->setOther(ELF::STO_MIPS_MICROMIPS);
// The traditional defines for STO values assume the full byte and thus
// the shift to pack it.
Symbol->setOther(ELF::STO_MIPS_MICROMIPS >> 2);
} }
void MipsTargetELFStreamer::finish() { void MipsTargetELFStreamer::finish() {
@@ -527,13 +524,10 @@ void MipsTargetELFStreamer::emitAssignment(MCSymbol *S, const MCExpr *Value) {
const auto &RhsSym = cast<MCSymbolELF>( const auto &RhsSym = cast<MCSymbolELF>(
static_cast<const MCSymbolRefExpr *>(Value)->getSymbol()); static_cast<const MCSymbolRefExpr *>(Value)->getSymbol());
if (!(RhsSym.getOther() & (ELF::STO_MIPS_MICROMIPS >> 2))) if (!(RhsSym.getOther() & ELF::STO_MIPS_MICROMIPS))
return; return;
// The "other" values are stored in the last 6 bits of the second byte. Symbol->setOther(ELF::STO_MIPS_MICROMIPS);
// The traditional defines for STO values assume the full byte and thus
// the shift to pack it.
Symbol->setOther(ELF::STO_MIPS_MICROMIPS >> 2);
} }
MCELFStreamer &MipsTargetELFStreamer::getStreamer() { MCELFStreamer &MipsTargetELFStreamer::getStreamer() {

View File

@@ -169,13 +169,10 @@ public:
if (Res != ELF::decodePPC64LocalEntryOffset(Encoded)) if (Res != ELF::decodePPC64LocalEntryOffset(Encoded))
report_fatal_error(".localentry expression cannot be encoded."); report_fatal_error(".localentry expression cannot be encoded.");
// The "other" values are stored in the last 6 bits of the second byte. unsigned Other = S->getOther();
// The traditional defines for STO values assume the full byte and thus
// the shift to pack it.
unsigned Other = S->getOther() << 2;
Other &= ~ELF::STO_PPC64_LOCAL_MASK; Other &= ~ELF::STO_PPC64_LOCAL_MASK;
Other |= Encoded; Other |= Encoded;
S->setOther(Other >> 2); S->setOther(Other);
// For GAS compatibility, unless we already saw a .abiversion directive, // For GAS compatibility, unless we already saw a .abiversion directive,
// set e_flags to indicate ELFv2 ABI. // set e_flags to indicate ELFv2 ABI.
@@ -191,13 +188,10 @@ public:
return; return;
const auto &RhsSym = cast<MCSymbolELF>( const auto &RhsSym = cast<MCSymbolELF>(
static_cast<const MCSymbolRefExpr *>(Value)->getSymbol()); static_cast<const MCSymbolRefExpr *>(Value)->getSymbol());
// The "other" values are stored in the last 6 bits of the second byte. unsigned Other = Symbol->getOther();
// The traditional defines for STO values assume the full byte and thus
// the shift to pack it.
unsigned Other = Symbol->getOther() << 2;
Other &= ~ELF::STO_PPC64_LOCAL_MASK; Other &= ~ELF::STO_PPC64_LOCAL_MASK;
Other |= (RhsSym.getOther() << 2) & ELF::STO_PPC64_LOCAL_MASK; Other |= RhsSym.getOther() & ELF::STO_PPC64_LOCAL_MASK;
Symbol->setOther(Other >> 2); Symbol->setOther(Other);
} }
}; };