2010-06-16 22:04:22 +02:00
|
|
|
//===- lib/MC/MCObjectStreamer.cpp - Object File MCStreamer Interface -----===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/MC/MCObjectStreamer.h"
|
2012-03-26 08:58:25 +02:00
|
|
|
#include "llvm/MC/MCAsmBackend.h"
|
|
|
|
#include "llvm/MC/MCAsmInfo.h"
|
2010-06-16 22:04:22 +02:00
|
|
|
#include "llvm/MC/MCAssembler.h"
|
2010-07-29 19:48:06 +02:00
|
|
|
#include "llvm/MC/MCCodeEmitter.h"
|
2010-12-10 08:39:47 +01:00
|
|
|
#include "llvm/MC/MCContext.h"
|
2010-11-01 17:27:31 +01:00
|
|
|
#include "llvm/MC/MCDwarf.h"
|
2010-07-19 08:13:10 +02:00
|
|
|
#include "llvm/MC/MCExpr.h"
|
2012-03-26 08:58:25 +02:00
|
|
|
#include "llvm/MC/MCObjectWriter.h"
|
2010-11-28 18:18:55 +01:00
|
|
|
#include "llvm/MC/MCSymbol.h"
|
2012-03-26 08:58:25 +02:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2010-06-16 22:04:22 +02:00
|
|
|
using namespace llvm;
|
|
|
|
|
2011-07-26 01:24:55 +02:00
|
|
|
MCObjectStreamer::MCObjectStreamer(MCContext &Context, MCAsmBackend &TAB,
|
2010-12-17 03:45:41 +01:00
|
|
|
raw_ostream &OS, MCCodeEmitter *Emitter_)
|
|
|
|
: MCStreamer(Context),
|
|
|
|
Assembler(new MCAssembler(Context, TAB,
|
|
|
|
*Emitter_, *TAB.createObjectWriter(OS),
|
|
|
|
OS)),
|
2010-06-16 22:04:25 +02:00
|
|
|
CurSectionData(0)
|
2010-06-16 22:04:22 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-07-26 01:24:55 +02:00
|
|
|
MCObjectStreamer::MCObjectStreamer(MCContext &Context, MCAsmBackend &TAB,
|
2011-03-09 18:33:05 +01:00
|
|
|
raw_ostream &OS, MCCodeEmitter *Emitter_,
|
|
|
|
MCAssembler *_Assembler)
|
|
|
|
: MCStreamer(Context), Assembler(_Assembler), CurSectionData(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-06-16 22:04:22 +02:00
|
|
|
MCObjectStreamer::~MCObjectStreamer() {
|
2010-07-29 19:48:06 +02:00
|
|
|
delete &Assembler->getBackend();
|
|
|
|
delete &Assembler->getEmitter();
|
2010-12-17 03:45:41 +01:00
|
|
|
delete &Assembler->getWriter();
|
2010-06-16 22:04:22 +02:00
|
|
|
delete Assembler;
|
|
|
|
}
|
2010-06-16 22:04:25 +02:00
|
|
|
|
2010-07-19 08:13:10 +02:00
|
|
|
MCFragment *MCObjectStreamer::getCurrentFragment() const {
|
|
|
|
assert(getCurrentSectionData() && "No current section!");
|
|
|
|
|
|
|
|
if (!getCurrentSectionData()->empty())
|
|
|
|
return &getCurrentSectionData()->getFragmentList().back();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
MCDataFragment *MCObjectStreamer::getOrCreateDataFragment() const {
|
|
|
|
MCDataFragment *F = dyn_cast_or_null<MCDataFragment>(getCurrentFragment());
|
|
|
|
if (!F)
|
|
|
|
F = new MCDataFragment(getCurrentSectionData());
|
|
|
|
return F;
|
|
|
|
}
|
|
|
|
|
|
|
|
const MCExpr *MCObjectStreamer::AddValueSymbols(const MCExpr *Value) {
|
|
|
|
switch (Value->getKind()) {
|
2011-01-13 08:58:56 +01:00
|
|
|
case MCExpr::Target:
|
|
|
|
cast<MCTargetExpr>(Value)->AddValueSymbols(Assembler);
|
|
|
|
break;
|
|
|
|
|
2010-07-19 08:13:10 +02:00
|
|
|
case MCExpr::Constant:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MCExpr::Binary: {
|
|
|
|
const MCBinaryExpr *BE = cast<MCBinaryExpr>(Value);
|
|
|
|
AddValueSymbols(BE->getLHS());
|
|
|
|
AddValueSymbols(BE->getRHS());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case MCExpr::SymbolRef:
|
|
|
|
Assembler->getOrCreateSymbolData(cast<MCSymbolRefExpr>(Value)->getSymbol());
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MCExpr::Unary:
|
|
|
|
AddValueSymbols(cast<MCUnaryExpr>(Value)->getSubExpr());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Value;
|
|
|
|
}
|
|
|
|
|
2010-12-10 08:39:47 +01:00
|
|
|
void MCObjectStreamer::EmitValueImpl(const MCExpr *Value, unsigned Size,
|
2011-05-01 05:50:49 +02:00
|
|
|
unsigned AddrSpace) {
|
2010-11-29 00:08:47 +01:00
|
|
|
assert(AddrSpace == 0 && "Address space must be 0!");
|
|
|
|
MCDataFragment *DF = getOrCreateDataFragment();
|
|
|
|
|
|
|
|
// Avoid fixups when possible.
|
|
|
|
int64_t AbsValue;
|
2010-12-18 04:57:21 +01:00
|
|
|
if (AddValueSymbols(Value)->EvaluateAsAbsolute(AbsValue, getAssembler())) {
|
2010-12-03 03:54:21 +01:00
|
|
|
EmitIntValue(AbsValue, Size, AddrSpace);
|
|
|
|
return;
|
2010-11-29 00:08:47 +01:00
|
|
|
}
|
2010-12-03 03:54:21 +01:00
|
|
|
DF->addFixup(MCFixup::Create(DF->getContents().size(),
|
2010-12-22 23:04:28 +01:00
|
|
|
Value,
|
2011-05-01 05:50:49 +02:00
|
|
|
MCFixup::getKindForSize(Size, false)));
|
2010-12-03 03:54:21 +01:00
|
|
|
DF->getContents().resize(DF->getContents().size() + Size, 0);
|
2010-11-29 00:08:47 +01:00
|
|
|
}
|
|
|
|
|
2012-01-07 23:42:19 +01:00
|
|
|
void MCObjectStreamer::EmitCFIStartProcImpl(MCDwarfFrameInfo &Frame) {
|
|
|
|
RecordProcStart(Frame);
|
|
|
|
}
|
|
|
|
|
2012-01-09 01:17:29 +01:00
|
|
|
void MCObjectStreamer::EmitCFIEndProcImpl(MCDwarfFrameInfo &Frame) {
|
|
|
|
RecordProcEnd(Frame);
|
|
|
|
}
|
|
|
|
|
2010-11-28 18:18:55 +01:00
|
|
|
void MCObjectStreamer::EmitLabel(MCSymbol *Symbol) {
|
2011-04-27 17:21:19 +02:00
|
|
|
MCStreamer::EmitLabel(Symbol);
|
2010-11-28 18:18:55 +01:00
|
|
|
|
|
|
|
MCSymbolData &SD = getAssembler().getOrCreateSymbolData(*Symbol);
|
|
|
|
|
|
|
|
// FIXME: This is wasteful, we don't necessarily need to create a data
|
|
|
|
// fragment. Instead, we should mark the symbol as pointing into the data
|
|
|
|
// fragment if it exists, otherwise we should just queue the label and set its
|
|
|
|
// fragment pointer when we emit the next fragment.
|
|
|
|
MCDataFragment *F = getOrCreateDataFragment();
|
|
|
|
assert(!SD.getFragment() && "Unexpected fragment on symbol data!");
|
|
|
|
SD.setFragment(F);
|
|
|
|
SD.setOffset(F->getContents().size());
|
|
|
|
}
|
|
|
|
|
2011-04-22 01:39:26 +02:00
|
|
|
void MCObjectStreamer::EmitULEB128Value(const MCExpr *Value) {
|
2010-12-03 02:19:49 +01:00
|
|
|
int64_t IntValue;
|
2010-12-18 04:57:21 +01:00
|
|
|
if (Value->EvaluateAsAbsolute(IntValue, getAssembler())) {
|
2011-04-22 01:39:26 +02:00
|
|
|
EmitULEB128IntValue(IntValue);
|
2010-12-03 02:19:49 +01:00
|
|
|
return;
|
|
|
|
}
|
2011-05-19 23:40:34 +02:00
|
|
|
Value = ForceExpAbs(Value);
|
2010-11-02 18:22:24 +01:00
|
|
|
new MCLEBFragment(*Value, false, getCurrentSectionData());
|
|
|
|
}
|
|
|
|
|
2011-04-22 01:39:26 +02:00
|
|
|
void MCObjectStreamer::EmitSLEB128Value(const MCExpr *Value) {
|
2010-12-03 02:19:49 +01:00
|
|
|
int64_t IntValue;
|
2010-12-18 04:57:21 +01:00
|
|
|
if (Value->EvaluateAsAbsolute(IntValue, getAssembler())) {
|
2011-04-22 01:39:26 +02:00
|
|
|
EmitSLEB128IntValue(IntValue);
|
2010-12-03 02:19:49 +01:00
|
|
|
return;
|
|
|
|
}
|
2011-05-19 23:40:34 +02:00
|
|
|
Value = ForceExpAbs(Value);
|
2010-11-02 18:22:24 +01:00
|
|
|
new MCLEBFragment(*Value, true, getCurrentSectionData());
|
|
|
|
}
|
|
|
|
|
2010-11-01 15:28:48 +01:00
|
|
|
void MCObjectStreamer::EmitWeakReference(MCSymbol *Alias,
|
|
|
|
const MCSymbol *Symbol) {
|
|
|
|
report_fatal_error("This file format doesn't support weak aliases.");
|
|
|
|
}
|
|
|
|
|
2011-02-16 02:08:29 +01:00
|
|
|
void MCObjectStreamer::ChangeSection(const MCSection *Section) {
|
2010-06-16 22:04:25 +02:00
|
|
|
assert(Section && "Cannot switch to a null section!");
|
|
|
|
|
|
|
|
CurSectionData = &getAssembler().getOrCreateSectionData(*Section);
|
|
|
|
}
|
|
|
|
|
2010-11-01 17:27:31 +01:00
|
|
|
void MCObjectStreamer::EmitInstruction(const MCInst &Inst) {
|
|
|
|
// Scan for values.
|
|
|
|
for (unsigned i = Inst.getNumOperands(); i--; )
|
|
|
|
if (Inst.getOperand(i).isExpr())
|
|
|
|
AddValueSymbols(Inst.getOperand(i).getExpr());
|
|
|
|
|
|
|
|
getCurrentSectionData()->setHasInstructions(true);
|
|
|
|
|
|
|
|
// Now that a machine instruction has been assembled into this section, make
|
|
|
|
// a line entry for any .loc directive that has been seen.
|
|
|
|
MCLineEntry::Make(this, getCurrentSection());
|
|
|
|
|
|
|
|
// If this instruction doesn't need relaxation, just emit it as data.
|
2012-01-18 19:52:16 +01:00
|
|
|
if (!getAssembler().getBackend().mayNeedRelaxation(Inst)) {
|
2010-11-01 17:27:31 +01:00
|
|
|
EmitInstToData(Inst);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, if we are relaxing everything, relax the instruction as much as
|
|
|
|
// possible and emit it as data.
|
|
|
|
if (getAssembler().getRelaxAll()) {
|
|
|
|
MCInst Relaxed;
|
2012-01-18 19:52:16 +01:00
|
|
|
getAssembler().getBackend().relaxInstruction(Inst, Relaxed);
|
|
|
|
while (getAssembler().getBackend().mayNeedRelaxation(Relaxed))
|
|
|
|
getAssembler().getBackend().relaxInstruction(Relaxed, Relaxed);
|
2010-11-01 17:27:31 +01:00
|
|
|
EmitInstToData(Relaxed);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise emit to a separate fragment.
|
|
|
|
EmitInstToFragment(Inst);
|
|
|
|
}
|
|
|
|
|
2010-12-02 06:44:06 +01:00
|
|
|
void MCObjectStreamer::EmitInstToFragment(const MCInst &Inst) {
|
|
|
|
MCInstFragment *IF = new MCInstFragment(Inst, getCurrentSectionData());
|
|
|
|
|
2011-04-18 22:54:46 +02:00
|
|
|
SmallString<128> Code;
|
|
|
|
raw_svector_ostream VecOS(Code);
|
2010-12-02 06:44:06 +01:00
|
|
|
getAssembler().getEmitter().EncodeInstruction(Inst, VecOS, IF->getFixups());
|
2011-04-18 22:54:46 +02:00
|
|
|
VecOS.flush();
|
|
|
|
IF->getCode().append(Code.begin(), Code.end());
|
2010-12-02 06:44:06 +01:00
|
|
|
}
|
|
|
|
|
2010-12-03 01:55:40 +01:00
|
|
|
void MCObjectStreamer::EmitDwarfAdvanceLineAddr(int64_t LineDelta,
|
|
|
|
const MCSymbol *LastLabel,
|
2011-07-14 07:43:07 +02:00
|
|
|
const MCSymbol *Label,
|
|
|
|
unsigned PointerSize) {
|
2010-12-03 01:55:40 +01:00
|
|
|
if (!LastLabel) {
|
|
|
|
EmitDwarfSetLineAddr(LineDelta, Label, PointerSize);
|
|
|
|
return;
|
|
|
|
}
|
2010-12-28 06:39:27 +01:00
|
|
|
const MCExpr *AddrDelta = BuildSymbolDiff(getContext(), Label, LastLabel);
|
2010-12-03 01:55:40 +01:00
|
|
|
int64_t Res;
|
2010-12-18 04:57:21 +01:00
|
|
|
if (AddrDelta->EvaluateAsAbsolute(Res, getAssembler())) {
|
2010-12-03 01:55:40 +01:00
|
|
|
MCDwarfLineAddr::Emit(this, LineDelta, Res);
|
|
|
|
return;
|
|
|
|
}
|
2011-05-19 23:40:34 +02:00
|
|
|
AddrDelta = ForceExpAbs(AddrDelta);
|
2010-12-03 01:55:40 +01:00
|
|
|
new MCDwarfLineAddrFragment(LineDelta, *AddrDelta, getCurrentSectionData());
|
|
|
|
}
|
|
|
|
|
2010-12-28 06:39:27 +01:00
|
|
|
void MCObjectStreamer::EmitDwarfAdvanceFrameAddr(const MCSymbol *LastLabel,
|
|
|
|
const MCSymbol *Label) {
|
|
|
|
const MCExpr *AddrDelta = BuildSymbolDiff(getContext(), Label, LastLabel);
|
|
|
|
int64_t Res;
|
|
|
|
if (AddrDelta->EvaluateAsAbsolute(Res, getAssembler())) {
|
|
|
|
MCDwarfFrameEmitter::EmitAdvanceLoc(*this, Res);
|
|
|
|
return;
|
|
|
|
}
|
2011-05-19 23:40:34 +02:00
|
|
|
AddrDelta = ForceExpAbs(AddrDelta);
|
2010-12-28 06:39:27 +01:00
|
|
|
new MCDwarfCallFrameFragment(*AddrDelta, getCurrentSectionData());
|
|
|
|
}
|
|
|
|
|
2012-01-27 01:37:08 +01:00
|
|
|
bool MCObjectStreamer::EmitValueToOffset(const MCExpr *Offset,
|
2012-01-27 00:47:45 +01:00
|
|
|
unsigned char Value) {
|
2011-02-20 21:20:07 +01:00
|
|
|
int64_t Res;
|
|
|
|
if (Offset->EvaluateAsAbsolute(Res, getAssembler())) {
|
|
|
|
new MCOrgFragment(*Offset, Value, getCurrentSectionData());
|
2012-01-27 01:37:08 +01:00
|
|
|
return false;
|
2011-02-20 21:20:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
MCSymbol *CurrentPos = getContext().CreateTempSymbol();
|
|
|
|
EmitLabel(CurrentPos);
|
|
|
|
MCSymbolRefExpr::VariantKind Variant = MCSymbolRefExpr::VK_None;
|
|
|
|
const MCExpr *Ref =
|
|
|
|
MCSymbolRefExpr::Create(CurrentPos, Variant, getContext());
|
|
|
|
const MCExpr *Delta =
|
|
|
|
MCBinaryExpr::Create(MCBinaryExpr::Sub, Offset, Ref, getContext());
|
|
|
|
|
|
|
|
if (!Delta->EvaluateAsAbsolute(Res, getAssembler()))
|
2012-01-27 01:37:08 +01:00
|
|
|
return true;
|
2011-02-20 21:20:07 +01:00
|
|
|
EmitFill(Res, Value, 0);
|
2012-01-27 01:37:08 +01:00
|
|
|
return false;
|
2010-12-02 06:59:38 +01:00
|
|
|
}
|
|
|
|
|
2011-11-23 23:18:04 +01:00
|
|
|
// Associate GPRel32 fixup with data and resize data area
|
|
|
|
void MCObjectStreamer::EmitGPRel32Value(const MCExpr *Value) {
|
|
|
|
MCDataFragment *DF = getOrCreateDataFragment();
|
|
|
|
|
|
|
|
DF->addFixup(MCFixup::Create(DF->getContents().size(),
|
|
|
|
Value,
|
|
|
|
FK_GPRel_4));
|
|
|
|
DF->getContents().resize(DF->getContents().size() + 4, 0);
|
|
|
|
}
|
|
|
|
|
2012-01-07 04:13:18 +01:00
|
|
|
void MCObjectStreamer::FinishImpl() {
|
2010-12-10 08:39:47 +01:00
|
|
|
// Dump out the dwarf file & directory tables and line tables.
|
2012-03-03 15:24:15 +01:00
|
|
|
const MCSymbol *LineSectionSymbol = NULL;
|
2010-12-10 08:39:47 +01:00
|
|
|
if (getContext().hasDwarfFiles())
|
2012-02-28 22:13:05 +01:00
|
|
|
LineSectionSymbol = MCDwarfFileTable::Emit(this);
|
2010-12-10 08:39:47 +01:00
|
|
|
|
2011-12-09 19:09:40 +01:00
|
|
|
// If we are generating dwarf for assembly source files dump out the sections.
|
|
|
|
if (getContext().getGenDwarfForAssembly())
|
2012-02-28 22:13:05 +01:00
|
|
|
MCGenDwarfInfo::Emit(this, LineSectionSymbol);
|
2011-12-09 19:09:40 +01:00
|
|
|
|
2010-06-16 22:04:25 +02:00
|
|
|
getAssembler().Finish();
|
|
|
|
}
|