| //===- subzero/src/IceELFSection.cpp - Representation of ELF sections -----===// |
| // |
| // The Subzero Code Generator |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| /// |
| /// \file |
| /// \brief Defines how ELF sections are represented. |
| /// |
| //===----------------------------------------------------------------------===// |
| |
| #include "IceELFSection.h" |
| |
| #include "IceDefs.h" |
| #include "IceELFStreamer.h" |
| #include "llvm/Support/MathExtras.h" |
| |
| using namespace llvm::ELF; |
| |
| namespace Ice { |
| |
| // Text sections. |
| |
| void ELFTextSection::appendData(ELFStreamer &Str, |
| const llvm::StringRef MoreData) { |
| Str.writeBytes(MoreData); |
| Header.sh_size += MoreData.size(); |
| } |
| |
| // Data sections. |
| |
| void ELFDataSection::appendData(ELFStreamer &Str, |
| const llvm::StringRef MoreData) { |
| Str.writeBytes(MoreData); |
| Header.sh_size += MoreData.size(); |
| } |
| |
| void ELFDataSection::appendZeros(ELFStreamer &Str, SizeT NumBytes) { |
| Str.writeZeroPadding(NumBytes); |
| Header.sh_size += NumBytes; |
| } |
| |
| void ELFDataSection::appendRelocationOffset(ELFStreamer &Str, bool IsRela, |
| RelocOffsetT RelocOffset) { |
| const SizeT RelocAddrSize = typeWidthInBytes(getPointerType()); |
| if (IsRela) { |
| appendZeros(Str, RelocAddrSize); |
| return; |
| } |
| assert(RelocAddrSize == 4 && " writeLE32 assumes RelocAddrSize is 4"); |
| Str.writeLE32(RelocOffset); |
| Header.sh_size += RelocAddrSize; |
| } |
| |
| void ELFDataSection::padToAlignment(ELFStreamer &Str, Elf64_Xword Align) { |
| assert(llvm::isPowerOf2_32(Align)); |
| Elf64_Xword AlignDiff = Utils::OffsetToAlignment(Header.sh_size, Align); |
| if (AlignDiff == 0) |
| return; |
| if (Header.sh_type != llvm::ELF::SHT_NOBITS) |
| Str.writeZeroPadding(AlignDiff); |
| Header.sh_size += AlignDiff; |
| } |
| |
| // Relocation sections. |
| |
| void ELFRelocationSection::addRelocations(RelocOffsetT BaseOff, |
| const FixupRefList &FixupRefs, |
| ELFSymbolTableSection *SymTab) { |
| for (const AssemblerFixup *FR : FixupRefs) { |
| Fixups.push_back(*FR); |
| AssemblerFixup &F = Fixups.back(); |
| F.set_position(BaseOff + F.position()); |
| assert(!F.valueIsSymbol()); |
| if (!F.isNullSymbol()) { |
| // Do an early lookup in the symbol table. If the symbol is found, |
| // replace the Constant in the symbol with the ELFSym, and calculate the |
| // final value of the addend. As such, a local label allocated from the |
| // Assembler arena will be converted to a symbol before the Assembler |
| // arena goes away. |
| if (const ELFSym *Sym = SymTab->findSymbol(F.symbol())) { |
| F.set_addend(F.offset()); |
| F.set_value(Sym); |
| } |
| } |
| } |
| } |
| |
| size_t ELFRelocationSection::getSectionDataSize() const { |
| return Fixups.size() * Header.sh_entsize; |
| } |
| |
| // Symbol tables. |
| |
| void ELFSymbolTableSection::createNullSymbol(ELFSection *NullSection, |
| GlobalContext *Ctx) { |
| // The first entry in the symbol table should be a NULL entry, so make sure |
| // the map is still empty. |
| assert(LocalSymbols.empty()); |
| // Explicitly set the null symbol name to the empty string, so that |
| // GlobalString::operator<() orders the null string first. |
| NullSymbolName = GlobalString::createWithString(Ctx, ""); |
| createDefinedSym(NullSymbolName, STT_NOTYPE, STB_LOCAL, NullSection, 0, 0); |
| NullSymbol = findSymbol(NullSymbolName); |
| } |
| |
| void ELFSymbolTableSection::createDefinedSym(GlobalString Name, uint8_t Type, |
| uint8_t Binding, |
| ELFSection *Section, |
| RelocOffsetT Offset, SizeT Size) { |
| ELFSym NewSymbol = ELFSym(); |
| NewSymbol.Sym.setBindingAndType(Binding, Type); |
| NewSymbol.Sym.st_value = Offset; |
| NewSymbol.Sym.st_size = Size; |
| NewSymbol.Section = Section; |
| NewSymbol.Number = ELFSym::UnknownNumber; |
| bool Unique; |
| if (Binding == STB_LOCAL) |
| Unique = LocalSymbols.insert(std::make_pair(Name, NewSymbol)).second; |
| else |
| Unique = GlobalSymbols.insert(std::make_pair(Name, NewSymbol)).second; |
| assert(Unique); |
| (void)Unique; |
| } |
| |
| void ELFSymbolTableSection::noteUndefinedSym(GlobalString Name, |
| ELFSection *NullSection) { |
| ELFSym NewSymbol = ELFSym(); |
| NewSymbol.Sym.setBindingAndType(STB_GLOBAL, STT_NOTYPE); |
| NewSymbol.Section = NullSection; |
| NewSymbol.Number = ELFSym::UnknownNumber; |
| bool Unique = GlobalSymbols.insert(std::make_pair(Name, NewSymbol)).second; |
| if (!Unique) { |
| std::string Buffer; |
| llvm::raw_string_ostream StrBuf(Buffer); |
| StrBuf << "Symbol external and defined: " << Name; |
| llvm::report_fatal_error(StrBuf.str()); |
| } |
| (void)Unique; |
| } |
| |
| const ELFSym *ELFSymbolTableSection::findSymbol(GlobalString Name) const { |
| auto I = LocalSymbols.find(Name); |
| if (I != LocalSymbols.end()) |
| return &I->second; |
| I = GlobalSymbols.find(Name); |
| if (I != GlobalSymbols.end()) |
| return &I->second; |
| return nullptr; |
| } |
| |
| void ELFSymbolTableSection::updateIndices(const ELFStringTableSection *StrTab) { |
| SizeT SymNumber = 0; |
| for (auto &KeyValue : LocalSymbols) { |
| GlobalString Name = KeyValue.first; |
| ELFSection *Section = KeyValue.second.Section; |
| Elf64_Sym &SymInfo = KeyValue.second.Sym; |
| if (Name != NullSymbolName && Name.hasStdString()) |
| SymInfo.st_name = StrTab->getIndex(Name.toString()); |
| SymInfo.st_shndx = Section->getNumber(); |
| KeyValue.second.setNumber(SymNumber++); |
| } |
| for (auto &KeyValue : GlobalSymbols) { |
| const std::string &Name = KeyValue.first.toString(); |
| ELFSection *Section = KeyValue.second.Section; |
| Elf64_Sym &SymInfo = KeyValue.second.Sym; |
| if (!Name.empty()) |
| SymInfo.st_name = StrTab->getIndex(Name); |
| SymInfo.st_shndx = Section->getNumber(); |
| KeyValue.second.setNumber(SymNumber++); |
| } |
| } |
| |
| void ELFSymbolTableSection::writeData(ELFStreamer &Str, bool IsELF64) { |
| if (IsELF64) { |
| writeSymbolMap<true>(Str, LocalSymbols); |
| writeSymbolMap<true>(Str, GlobalSymbols); |
| } else { |
| writeSymbolMap<false>(Str, LocalSymbols); |
| writeSymbolMap<false>(Str, GlobalSymbols); |
| } |
| } |
| |
| // String tables. |
| |
| void ELFStringTableSection::add(const std::string &Str) { |
| assert(!isLaidOut()); |
| assert(!Str.empty()); |
| StringToIndexMap.insert(std::make_pair(Str, UnknownIndex)); |
| } |
| |
| size_t ELFStringTableSection::getIndex(const std::string &Str) const { |
| assert(isLaidOut()); |
| StringToIndexType::const_iterator It = StringToIndexMap.find(Str); |
| if (It == StringToIndexMap.end()) { |
| llvm::report_fatal_error("String index not found: " + Str); |
| return UnknownIndex; |
| } |
| return It->second; |
| } |
| |
| bool ELFStringTableSection::SuffixComparator::operator()( |
| const std::string &StrA, const std::string &StrB) const { |
| size_t LenA = StrA.size(); |
| size_t LenB = StrB.size(); |
| size_t CommonLen = std::min(LenA, LenB); |
| // If there is a difference in the common suffix, use that diff to sort. |
| for (size_t i = 0; i < CommonLen; ++i) { |
| char a = StrA[LenA - i - 1]; |
| char b = StrB[LenB - i - 1]; |
| if (a != b) |
| return a > b; |
| } |
| // If the common suffixes are completely equal, let the longer one come |
| // first, so that it can be laid out first and its characters shared. |
| return LenA > LenB; |
| } |
| |
| void ELFStringTableSection::doLayout() { |
| assert(!isLaidOut()); |
| llvm::StringRef Prev; |
| |
| // String table starts with 0 byte. |
| StringData.push_back(0); |
| |
| for (auto &StringIndex : StringToIndexMap) { |
| assert(StringIndex.second == UnknownIndex); |
| llvm::StringRef Cur = llvm::StringRef(StringIndex.first); |
| if (Prev.endswith(Cur)) { |
| // Prev is already in the StringData, and Cur is shorter than Prev based |
| // on the sort. |
| StringIndex.second = StringData.size() - Cur.size() - 1; |
| continue; |
| } |
| StringIndex.second = StringData.size(); |
| std::copy(Cur.begin(), Cur.end(), back_inserter(StringData)); |
| StringData.push_back(0); |
| Prev = Cur; |
| } |
| } |
| |
| } // end of namespace Ice |