//===- lib/ReaderWriter/ELF/ExecutableWriter.h ----------------------------===// // // The LLVM Linker // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef LLD_READER_WRITER_ELF_EXECUTABLE_WRITER_H #define LLD_READER_WRITER_ELF_EXECUTABLE_WRITER_H #include "OutputELFWriter.h" namespace lld { namespace elf { using namespace llvm; using namespace llvm::object; template class ExecutableWriter; //===----------------------------------------------------------------------===// // ExecutableWriter Class //===----------------------------------------------------------------------===// template class ExecutableWriter : public OutputELFWriter { public: ExecutableWriter(const ELFLinkingContext &context, TargetLayout &layout) : OutputELFWriter(context, layout), _runtimeFile(new CRuntimeFile(context)) {} protected: virtual void addDefaultAtoms(); virtual bool createImplicitFiles(std::vector > &); virtual void finalizeDefaultAtomValues(); virtual void createDefaultSections(); LLD_UNIQUE_BUMP_PTR(InterpSection) _interpSection; std::unique_ptr > _runtimeFile; }; //===----------------------------------------------------------------------===// // ExecutableWriter //===----------------------------------------------------------------------===// /// \brief Add absolute symbols by default. These are linker added /// absolute symbols template void ExecutableWriter::addDefaultAtoms() { _runtimeFile->addUndefinedAtom(this->_context.entrySymbolName()); _runtimeFile->addAbsoluteAtom("__bss_start"); _runtimeFile->addAbsoluteAtom("__bss_end"); _runtimeFile->addAbsoluteAtom("_end"); _runtimeFile->addAbsoluteAtom("end"); _runtimeFile->addAbsoluteAtom("__preinit_array_start"); _runtimeFile->addAbsoluteAtom("__preinit_array_end"); _runtimeFile->addAbsoluteAtom("__init_array_start"); _runtimeFile->addAbsoluteAtom("__init_array_end"); if (this->_context.isRelaOutputFormat()) { _runtimeFile->addAbsoluteAtom("__rela_iplt_start"); _runtimeFile->addAbsoluteAtom("__rela_iplt_end"); } else { _runtimeFile->addAbsoluteAtom("__rel_iplt_start"); _runtimeFile->addAbsoluteAtom("__rel_iplt_end"); } _runtimeFile->addAbsoluteAtom("__fini_array_start"); _runtimeFile->addAbsoluteAtom("__fini_array_end"); } /// \brief Hook in lld to add CRuntime file template bool ExecutableWriter::createImplicitFiles( std::vector > &result) { // Add the default atoms as defined by executables ExecutableWriter::addDefaultAtoms(); OutputELFWriter::createImplicitFiles(result); result.push_back(std::move(_runtimeFile)); return true; } template void ExecutableWriter::createDefaultSections() { OutputELFWriter::createDefaultSections(); if (this->_context.isDynamic()) { _interpSection.reset(new (this->_alloc) InterpSection( this->_context, ".interp", DefaultLayout::ORDER_INTERP, this->_context.getInterpreter())); this->_layout.addSection(_interpSection.get()); } } /// Finalize the value of all the absolute symbols that we /// created template void ExecutableWriter::finalizeDefaultAtomValues() { auto bssStartAtomIter = this->_layout.findAbsoluteAtom("__bss_start"); auto bssEndAtomIter = this->_layout.findAbsoluteAtom("__bss_end"); auto underScoreEndAtomIter = this->_layout.findAbsoluteAtom("_end"); auto endAtomIter = this->_layout.findAbsoluteAtom("end"); auto startEnd = [&](StringRef sym, StringRef sec) -> void { // TODO: This looks like a good place to use Twine... std::string start("__"), end("__"); start += sym; start += "_start"; end += sym; end += "_end"; auto s = this->_layout.findAbsoluteAtom(start); auto e = this->_layout.findAbsoluteAtom(end); auto section = this->_layout.findOutputSection(sec); if (section) { (*s)->_virtualAddr = section->virtualAddr(); (*e)->_virtualAddr = section->virtualAddr() + section->memSize(); } else { (*s)->_virtualAddr = 0; (*e)->_virtualAddr = 0; } }; startEnd("preinit_array", ".preinit_array"); startEnd("init_array", ".init_array"); if (this->_context.isRelaOutputFormat()) startEnd("rela_iplt", ".rela.plt"); else startEnd("rel_iplt", ".rel.plt"); startEnd("fini_array", ".fini_array"); assert(!(bssStartAtomIter == this->_layout.absoluteAtoms().end() || bssEndAtomIter == this->_layout.absoluteAtoms().end() || underScoreEndAtomIter == this->_layout.absoluteAtoms().end() || endAtomIter == this->_layout.absoluteAtoms().end()) && "Unable to find the absolute atoms that have been added by lld"); auto bssSection = this->_layout.findOutputSection(".bss"); // If we don't find a bss section, then don't set these values if (bssSection) { (*bssStartAtomIter)->_virtualAddr = bssSection->virtualAddr(); (*bssEndAtomIter)->_virtualAddr = bssSection->virtualAddr() + bssSection->memSize(); (*underScoreEndAtomIter)->_virtualAddr = (*bssEndAtomIter)->_virtualAddr; (*endAtomIter)->_virtualAddr = (*bssEndAtomIter)->_virtualAddr; } else if (auto dataSection = this->_layout.findOutputSection(".data")) { (*underScoreEndAtomIter)->_virtualAddr = dataSection->virtualAddr() + dataSection->memSize(); (*endAtomIter)->_virtualAddr = (*underScoreEndAtomIter)->_virtualAddr; } } } // namespace elf } // namespace lld #endif // LLD_READER_WRITER_ELF_EXECUTABLE_WRITER_H