//===- subzero/src/IceELFObjectWriter.h - ELF object writer -----*- C++ -*-===//
//
// The Subzero Code Generator
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
///
/// \file
/// \brief Abstraction for a writer that is responsible for writing an ELF file.
///
//===----------------------------------------------------------------------===//
#ifndef SUBZERO_SRC_ICEELFOBJECTWRITER_H
#define SUBZERO_SRC_ICEELFOBJECTWRITER_H
#include "IceDefs.h"
#include "IceELFSection.h"
#include "IceELFStreamer.h"
#include "IceTypes.h"
using namespace llvm::ELF;
namespace Ice {
using VariableDeclarationPartition = std::vector<VariableDeclaration *>;
/// Higher level ELF object writer. Manages section information and writes the
/// final ELF object. The object writer will write to file the code and data as
/// it is being defined (rather than keep a copy). After all definitions are
/// written out, it will finalize the bookkeeping sections and write them out.
/// Expected usage:
///
/// (1) writeInitialELFHeader (invoke once)
/// (2) writeDataSection (may be invoked multiple times, as long as
/// SectionSuffix is unique)
/// (3) writeFunctionCode (must invoke once per function)
/// (4) writeConstantPool (must invoke once per pooled primitive type)
/// (5) setUndefinedSyms (invoke once)
/// (6) writeNonUserSections (invoke once)
///
/// The requirement for writeDataSection to be invoked only once can be relaxed
/// if using -fdata-sections. The requirement to invoke only once without
/// -fdata-sections is so that variables that belong to each possible
/// SectionType are contiguous in the file. With -fdata-sections, each global
/// variable is in a separate section and therefore the sections will be
/// trivially contiguous.
class ELFObjectWriter {
ELFObjectWriter() = delete;
ELFObjectWriter(const ELFObjectWriter &) = delete;
ELFObjectWriter &operator=(const ELFObjectWriter &) = delete;
public:
ELFObjectWriter(GlobalContext &Ctx, ELFStreamer &Out);
/// Write the initial ELF header. This is just to reserve space in the ELF
/// file. Reserving space allows the other functions to write text and data
/// directly to the file and get the right file offsets.
void writeInitialELFHeader();
/// Copy initializer data for globals to file and note the offset and size of
/// each global's definition in the symbol table. Use the given target's
/// RelocationKind for any relocations.
void writeDataSection(const VariableDeclarationList &Vars,
FixupKind RelocationKind,
const std::string &SectionSuffix, bool IsPIC);
/// Copy data of a function's text section to file and note the offset of the
/// symbol's definition in the symbol table. Copy the text fixups for use
/// after all functions are written. The text buffer and fixups are extracted
/// from the Assembler object.
void writeFunctionCode(GlobalString FuncName, bool IsInternal,
Assembler *Asm);
/// Queries the GlobalContext for constant pools of the given type and writes
/// out read-only data sections for those constants. This also fills the
/// symbol table with labels for each constant pool entry.
template <typename ConstType> void writeConstantPool(Type Ty);
/// Write a jump table and register fixups for the target addresses.
void writeJumpTable(const JumpTableData &JT, FixupKind RelocationKind,
bool IsPIC);
/// Populate the symbol table with a list of external/undefined symbols.
void setUndefinedSyms(const ConstantList &UndefSyms);
/// Do final layout and write out the rest of the object file. Finally, patch
/// up the initial ELF header with the final info.
void writeNonUserSections();
/// Which type of ELF section a global variable initializer belongs to. This
/// is used as an array index so should start at 0 and be contiguous.
enum SectionType { ROData = 0, Data, BSS, NumSectionTypes };
/// Create target specific section with the given information about section.
void writeTargetRODataSection(const std::string &Name, Elf64_Word ShType,
Elf64_Xword ShFlags, Elf64_Xword ShAddralign,
Elf64_Xword ShEntsize,
const llvm::StringRef &SecData);
private:
GlobalContext &Ctx;
ELFStreamer &Str;
bool SectionNumbersAssigned = false;
bool ELF64;
// All created sections, separated into different pools.
using SectionList = std::vector<ELFSection *>;
using TextSectionList = std::vector<ELFTextSection *>;
using DataSectionList = std::vector<ELFDataSection *>;
using RelSectionList = std::vector<ELFRelocationSection *>;
TextSectionList TextSections;
RelSectionList RelTextSections;
DataSectionList DataSections;
RelSectionList RelDataSections;
DataSectionList RODataSections;
RelSectionList RelRODataSections;
DataSectionList BSSSections;
// Handles to special sections that need incremental bookkeeping.
ELFSection *NullSection;
ELFStringTableSection *ShStrTab;
ELFSymbolTableSection *SymTab;
ELFStringTableSection *StrTab;
template <typename T>
T *createSection(const std::string &Name, Elf64_Word ShType,
Elf64_Xword ShFlags, Elf64_Xword ShAddralign,
Elf64_Xword ShEntsize);
/// Create a relocation section, given the related section (e.g., .text,
/// .data., .rodata).
ELFRelocationSection *
createRelocationSection(const ELFSection *RelatedSection);
/// Align the file position before writing out a section's data, and return
/// the position of the file.
Elf64_Off alignFileOffset(Elf64_Xword Align);
/// Assign an ordering / section numbers to each section. Fill in other
/// information that is only known near the end (such as the size, if it
/// wasn't already incrementally updated). This then collects all sections in
/// the decided order, into one vector, for conveniently writing out all of
/// the section headers.
void assignSectionNumbersInfo(SectionList &AllSections);
/// This function assigns .foo and .rel.foo consecutive section numbers. It
/// also sets the relocation section's sh_info field to the related section's
/// number.
template <typename UserSectionList>
void assignRelSectionNumInPairs(SizeT &CurSectionNumber,
UserSectionList &UserSections,
RelSectionList &RelSections,
SectionList &AllSections);
/// Link the relocation sections to the symbol table.
void assignRelLinkNum(SizeT SymTabNumber, RelSectionList &RelSections);
/// Helper function for writeDataSection. Writes a data section of type
/// SectionType, given the global variables Vars belonging to that
/// SectionType.
void writeDataOfType(SectionType SectionType,
const VariableDeclarationPartition &Vars,
FixupKind RelocationKind,
const std::string &SectionSuffix, bool IsPIC);
/// Write the final relocation sections given the final symbol table. May also
/// be able to seek around the file and resolve function calls that are for
/// functions within the same section.
void writeAllRelocationSections();
void writeRelocationSections(RelSectionList &RelSections);
/// Write the ELF file header with the given information about sections.
template <bool IsELF64>
void writeELFHeaderInternal(Elf64_Off SectionHeaderOffset,
SizeT SectHeaderStrIndex, SizeT NumSections);
};
} // end of namespace Ice
#endif // SUBZERO_SRC_ICEELFOBJECTWRITER_H