blob: e9b7b995eb7cc1718173f295632cee08d3a78273 [file] [log] [blame]
//===- ELFObjectWriter.cpp ------------------------------------------------===//
//
// The MCLinker Project
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include <mcld/LD/ELFObjectWriter.h>
#include <mcld/Module.h>
#include <mcld/LinkerConfig.h>
#include <mcld/Target/GNULDBackend.h>
#include <mcld/Support/MemoryArea.h>
#include <llvm/Support/system_error.h>
using namespace llvm;
using namespace mcld;
//===----------------------------------------------------------------------===//
// ELFObjectWriter
//===----------------------------------------------------------------------===//
ELFObjectWriter::ELFObjectWriter(GNULDBackend& pBackend,
const LinkerConfig& pConfig)
: ObjectWriter(pBackend), ELFWriter(pBackend),
m_Config(pConfig) {
}
ELFObjectWriter::~ELFObjectWriter()
{
}
llvm::error_code ELFObjectWriter::writeObject(Module& pModule,
MemoryArea& 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::Note:
if ((*sect)->getSectionData() == NULL)
continue;
// Fall through
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::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::Note:
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_Config, **sect, *region);
break;
case LDFileFormat::Target:
target().emitSectionData(**sect, *region);
break;
default:
continue;
}
} // end of for loop
emitELFShStrTab(target().getOutputFormat()->getShStrTab(),
pModule,
pOutput);
if (m_Config.targets().is32Bits()) {
// Write out ELF header
// Write out section header table
writeELF32Header(m_Config,
pModule,
pOutput);
emitELF32SectionHeader(pModule, m_Config, pOutput);
}
else if (m_Config.targets().is64Bits()) {
// Write out ELF header
// Write out section header table
writeELF64Header(m_Config,
pModule,
pOutput);
emitELF64SectionHeader(pModule, m_Config, pOutput);
}
else
return make_error_code(errc::not_supported);
pOutput.clear();
return llvm::make_error_code(llvm::errc::success);
}