//===- lib/MC/MCSymbolELF.cpp ---------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "llvm/MC/MCSymbolELF.h" #include "llvm/BinaryFormat/ELF.h" #include "llvm/MC/MCFixupKindInfo.h" namespace llvm { namespace { enum { // Shift value for STT_* flags. 7 possible values. 3 bits. ELF_STT_Shift = 0, // 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_BindingSet_Shift = 12 }; } void MCSymbolELF::setBinding(unsigned Binding) const { setIsBindingSet(); if (getType() == ELF::STT_SECTION && Binding != ELF::STB_LOCAL) setType(ELF::STT_NOTYPE); unsigned Val; switch (Binding) { default: 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 { if (isBindingSet()) { uint32_t Val = (Flags >> ELF_STB_Shift) & 3; switch (Val) { default: 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()) return ELF::STB_LOCAL; if (isUsedInReloc()) return ELF::STB_GLOBAL; if (isWeakrefUsedInReloc()) return ELF::STB_WEAK; if (isSignature()) return ELF::STB_LOCAL; return ELF::STB_GLOBAL; } void MCSymbolELF::setType(unsigned Type) const { unsigned Val; if (Type == ELF::STT_SECTION && getBinding() != ELF::STB_LOCAL) return; switch (Type) { default: llvm_unreachable("Unsupported Binding"); case ELF::STT_NOTYPE: Val = 0; 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 { uint32_t Val = (Flags >> ELF_STT_Shift) & 7; switch (Val) { default: llvm_unreachable("Invalid value"); case 0: 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; } } void MCSymbolELF::setVisibility(unsigned Visibility) { assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL || Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED); uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STV_Shift); setFlags(OtherFlags | (Visibility << ELF_STV_Shift)); } unsigned MCSymbolELF::getVisibility() const { unsigned Visibility = (Flags >> ELF_STV_Shift) & 3; return Visibility; } void MCSymbolELF::setOther(unsigned Other) { assert((Other & 0x1f) == 0); Other >>= 5; assert(Other <= 0x7); uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STO_Shift); setFlags(OtherFlags | (Other << ELF_STO_Shift)); } unsigned MCSymbolELF::getOther() const { unsigned Other = (Flags >> ELF_STO_Shift) & 7; return Other << 5; } void MCSymbolELF::setIsWeakrefUsedInReloc() const { uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_WeakrefUsedInReloc_Shift); setFlags(OtherFlags | (1 << ELF_WeakrefUsedInReloc_Shift)); } bool MCSymbolELF::isWeakrefUsedInReloc() const { 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); } }