mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-04-04 10:30:01 +00:00
Reuse data fragments while lowering. Patch by David Meyer.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@119999 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
cafe0b4e1f
commit
0e11dfc4a6
@ -59,8 +59,15 @@ public:
|
||||
/// the delta from the old size.
|
||||
void UpdateForSlide(MCFragment *F, int SlideAmount);
|
||||
|
||||
/// \brief Update the layout because a fragment has been replaced.
|
||||
void FragmentReplaced(MCFragment *Src, MCFragment *Dst);
|
||||
/// \brief Update the layout, replacing Src with Dst. The contents
|
||||
/// of Src and Dst are not modified, and must be copied by the caller.
|
||||
/// Src will be removed from the layout, but not deleted.
|
||||
void ReplaceFragment(MCFragment *Src, MCFragment *Dst);
|
||||
|
||||
/// \brief Update the layout to coalesce Src into Dst. The contents
|
||||
/// of Src and Dst are not modified, and must be coalesced by the caller.
|
||||
/// Src will be removed from the layout, but not deleted.
|
||||
void CoalesceFragments(MCFragment *Src, MCFragment *Dst);
|
||||
|
||||
/// \brief Perform a full layout.
|
||||
void LayoutFile();
|
||||
|
@ -113,12 +113,40 @@ void MCAsmLayout::EnsureValid(const MCFragment *F) const {
|
||||
}
|
||||
}
|
||||
|
||||
void MCAsmLayout::FragmentReplaced(MCFragment *Src, MCFragment *Dst) {
|
||||
void MCAsmLayout::ReplaceFragment(MCFragment *Src, MCFragment *Dst) {
|
||||
MCSectionData *SD = Src->getParent();
|
||||
|
||||
// Insert Dst immediately before Src
|
||||
SD->getFragmentList().insert(Src, Dst);
|
||||
|
||||
// Set the data fragment's layout data.
|
||||
Dst->setParent(Src->getParent());
|
||||
Dst->setAtom(Src->getAtom());
|
||||
Dst->setLayoutOrder(Src->getLayoutOrder());
|
||||
|
||||
if (LastValidFragment == Src)
|
||||
LastValidFragment = Dst;
|
||||
|
||||
Dst->Offset = Src->Offset;
|
||||
Dst->EffectiveSize = Src->EffectiveSize;
|
||||
|
||||
// Remove Src, but don't delete it yet.
|
||||
SD->getFragmentList().remove(Src);
|
||||
}
|
||||
|
||||
void MCAsmLayout::CoalesceFragments(MCFragment *Src, MCFragment *Dst) {
|
||||
assert(Src->getPrevNode() == Dst);
|
||||
|
||||
if (isFragmentUpToDate(Src)) {
|
||||
if (LastValidFragment == Src)
|
||||
LastValidFragment = Dst;
|
||||
Dst->EffectiveSize += Src->EffectiveSize;
|
||||
} else {
|
||||
// We don't know the effective size of Src, so we have to invalidate Dst.
|
||||
UpdateForSlide(Dst, 0);
|
||||
}
|
||||
// Remove Src, but don't delete it yet.
|
||||
Src->getParent()->getFragmentList().remove(Src);
|
||||
}
|
||||
|
||||
uint64_t MCAsmLayout::getFragmentAddress(const MCFragment *F) const {
|
||||
@ -890,6 +918,22 @@ bool MCAssembler::LayoutOnce(const MCObjectWriter &Writer,
|
||||
return WasRelaxed;
|
||||
}
|
||||
|
||||
static void LowerInstFragment(MCInstFragment *IF,
|
||||
MCDataFragment *DF) {
|
||||
|
||||
uint64_t DataOffset = DF->getContents().size();
|
||||
|
||||
// Copy in the data
|
||||
DF->getContents().append(IF->getCode().begin(), IF->getCode().end());
|
||||
|
||||
// Adjust the fixup offsets and add them to the data fragment.
|
||||
for (unsigned i = 0, e = IF->getFixups().size(); i != e; ++i) {
|
||||
MCFixup &F = IF->getFixups()[i];
|
||||
F.setOffset(DataOffset + F.getOffset());
|
||||
DF->getFixups().push_back(F);
|
||||
}
|
||||
}
|
||||
|
||||
void MCAssembler::FinishLayout(MCAsmLayout &Layout) {
|
||||
// Lower out any instruction fragments, to simplify the fixup application and
|
||||
// output.
|
||||
@ -898,35 +942,42 @@ void MCAssembler::FinishLayout(MCAsmLayout &Layout) {
|
||||
// cheap (we will mostly end up eliminating fragments and appending on to data
|
||||
// fragments), so the extra complexity downstream isn't worth it. Evaluate
|
||||
// this assumption.
|
||||
for (iterator it = begin(), ie = end(); it != ie; ++it) {
|
||||
MCSectionData &SD = *it;
|
||||
unsigned FragmentIndex = 0;
|
||||
for (unsigned i = 0, e = Layout.getSectionOrder().size(); i != e; ++i) {
|
||||
MCSectionData &SD = *Layout.getSectionOrder()[i];
|
||||
MCDataFragment *CurDF = NULL;
|
||||
|
||||
for (MCSectionData::iterator it2 = SD.begin(),
|
||||
ie2 = SD.end(); it2 != ie2; ++it2) {
|
||||
MCInstFragment *IF = dyn_cast<MCInstFragment>(it2);
|
||||
if (!IF)
|
||||
continue;
|
||||
switch (it2->getKind()) {
|
||||
default:
|
||||
CurDF = NULL;
|
||||
break;
|
||||
case MCFragment::FT_Data:
|
||||
CurDF = cast<MCDataFragment>(it2);
|
||||
break;
|
||||
case MCFragment::FT_Inst: {
|
||||
MCInstFragment *IF = cast<MCInstFragment>(it2);
|
||||
// Use the existing data fragment if possible.
|
||||
if (CurDF && CurDF->getAtom() == IF->getAtom()) {
|
||||
Layout.CoalesceFragments(IF, CurDF);
|
||||
} else {
|
||||
// Otherwise, create a new data fragment.
|
||||
CurDF = new MCDataFragment();
|
||||
Layout.ReplaceFragment(IF, CurDF);
|
||||
}
|
||||
|
||||
// Create a new data fragment for the instruction.
|
||||
//
|
||||
// FIXME-PERF: Reuse previous data fragment if possible.
|
||||
MCDataFragment *DF = new MCDataFragment();
|
||||
SD.getFragmentList().insert(it2, DF);
|
||||
// Lower the Instruction Fragment
|
||||
LowerInstFragment(IF, CurDF);
|
||||
|
||||
// Update the data fragments layout data.
|
||||
DF->setParent(IF->getParent());
|
||||
DF->setAtom(IF->getAtom());
|
||||
DF->setLayoutOrder(IF->getLayoutOrder());
|
||||
Layout.FragmentReplaced(IF, DF);
|
||||
|
||||
// Copy in the data and the fixups.
|
||||
DF->getContents().append(IF->getCode().begin(), IF->getCode().end());
|
||||
for (unsigned i = 0, e = IF->getFixups().size(); i != e; ++i)
|
||||
DF->getFixups().push_back(IF->getFixups()[i]);
|
||||
|
||||
// Delete the instruction fragment and update the iterator.
|
||||
SD.getFragmentList().erase(IF);
|
||||
it2 = DF;
|
||||
// Delete the instruction fragment and update the iterator.
|
||||
delete IF;
|
||||
it2 = CurDF;
|
||||
break;
|
||||
}
|
||||
}
|
||||
// Since we may have merged fragments, fix the layout order.
|
||||
it2->setLayoutOrder(FragmentIndex++);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user