blob: dec5e26e369b702f3dc7dc34ea86c39b5f2189cc [file] [log] [blame]
//===- ELFExecWriter.cpp --------------------------------------------------===//
//
// The MCLinker Project
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include <mcld/LD/ELFExecWriter.h>
#include <mcld/Module.h>
#include <mcld/LinkerConfig.h>
#include <mcld/LD/LDSymbol.h>
#include <mcld/Target/GNULDBackend.h>
#include <mcld/Fragment/FragmentLinker.h>
#include <mcld/Support/MemoryArea.h>
#include <llvm/Support/ELF.h>
#include <vector>
using namespace llvm;
using namespace mcld;
//===----------------------------------------------------------------------===//
// ELFExecWriter
//===----------------------------------------------------------------------===//
ELFExecWriter::ELFExecWriter(GNULDBackend& pBackend, FragmentLinker& pLinker)
: ExecWriter(pBackend),
ELFWriter(pBackend),
m_Linker(pLinker) {
}
ELFExecWriter::~ELFExecWriter()
{
}
llvm::error_code ELFExecWriter::writeExecutable(Module& pModule,
MemoryArea& pOutput)
{
// write out the interpreter section: .interp
target().emitInterp(pOutput);
// Write out name pool sections: .dynsym, .dynstr, .hash
target().emitDynNamePools(pModule, pOutput);
// Write out name pool sections: .symtab, .strtab
target().emitRegNamePools(pModule, pOutput);
// Write out regular ELF sections
Module::iterator sect, sectEnd = pModule.end();
for (sect = pModule.begin(); sect != sectEnd; ++sect) {
MemoryRegion* region = NULL;
// request output region
switch((*sect)->kind()) {
case LDFileFormat::Regular:
case LDFileFormat::Relocation:
case LDFileFormat::Target:
case LDFileFormat::Debug:
case LDFileFormat::GCCExceptTable:
case LDFileFormat::EhFrame: {
region = pOutput.request((*sect)->offset(), (*sect)->size());
if (NULL == region) {
llvm::report_fatal_error(llvm::Twine("cannot get enough memory region for output section `") +
llvm::Twine((*sect)->name()) +
llvm::Twine("'.\n"));
}
break;
}
case LDFileFormat::Null:
case LDFileFormat::NamePool:
case LDFileFormat::BSS:
case LDFileFormat::Note:
case LDFileFormat::MetaData:
case LDFileFormat::Version:
case LDFileFormat::EhFrameHdr:
case LDFileFormat::StackNote:
// ignore these sections
continue;
default: {
llvm::errs() << "WARNING: unsupported section kind: "
<< (*sect)->kind()
<< " of section "
<< (*sect)->name()
<< ".\n";
continue;
}
}
// write out sections with data
switch((*sect)->kind()) {
case LDFileFormat::Regular:
case LDFileFormat::Debug:
case LDFileFormat::GCCExceptTable:
case LDFileFormat::EhFrame: {
// FIXME: if optimization of exception handling sections is enabled,
// then we should emit these sections by the other way.
emitSectionData(**sect, *region);
break;
}
case LDFileFormat::Relocation:
emitRelocation(m_Linker.getLDInfo(), **sect, *region);
break;
case LDFileFormat::Target:
target().emitSectionData(**sect, *region);
break;
default:
continue;
}
} // end of for loop
emitELFShStrTab(target().getOutputFormat()->getShStrTab(),
pModule,
pOutput);
if (32 == target().bitclass()) {
// Write out ELF header
// Write out section header table
writeELF32Header(m_Linker.getLDInfo(),
pModule,
pOutput);
emitELF32ProgramHeader(pOutput);
emitELF32SectionHeader(pModule, m_Linker.getLDInfo(), pOutput);
}
else if (64 == target().bitclass()) {
// Write out ELF header
// Write out section header table
writeELF64Header(m_Linker.getLDInfo(),
pModule,
pOutput);
emitELF64ProgramHeader(pOutput);
emitELF64SectionHeader(pModule, m_Linker.getLDInfo(), pOutput);
}
else
return make_error_code(errc::not_supported);
pOutput.clear();
return llvm::make_error_code(llvm::errc::success);
}