mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-21 03:32:21 +00:00
54e853b8a6
The algorithm it used before wasn't 100% correct, we now use an iterative expansion model. This fixes assembler errors when compiling 403.gcc with tail merging enabled. Change the way the branch selector works overall: Now, the isel generates PPC::BCC instructions (as it used to) directly, and these BCC instructions are emitted to the output or jitted directly if branches don't need expansion. Only if branches need expansion are instructions rewritten and created. This should make branch select faster, and eliminates the Bxx instructions from the .td file. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@31837 91177308-0d34-0410-b5e6-96231b3b80d8
192 lines
6.5 KiB
C++
192 lines
6.5 KiB
C++
//===-- PPCBranchSelector.cpp - Emit long conditional branches-----*- C++ -*-=//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file was developed by Nate Baegeman and is distributed under the
|
|
// University of Illinois Open Source License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file contains a pass that scans a machine function to determine which
|
|
// conditional branches need more than 16 bits of displacement to reach their
|
|
// target basic block. It does this in two passes; a calculation of basic block
|
|
// positions pass, and a branch psuedo op to machine branch opcode pass. This
|
|
// pass should be run last, just before the assembly printer.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "PPC.h"
|
|
#include "PPCInstrBuilder.h"
|
|
#include "PPCInstrInfo.h"
|
|
#include "PPCPredicates.h"
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
|
#include "llvm/Target/TargetMachine.h"
|
|
#include "llvm/Target/TargetAsmInfo.h"
|
|
#include "llvm/ADT/Statistic.h"
|
|
#include "llvm/Support/Compiler.h"
|
|
#include "llvm/Support/MathExtras.h"
|
|
using namespace llvm;
|
|
|
|
static Statistic<> NumExpanded("ppc-branch-select",
|
|
"Num branches expanded to long format");
|
|
|
|
namespace {
|
|
struct VISIBILITY_HIDDEN PPCBSel : public MachineFunctionPass {
|
|
/// BlockSizes - The sizes of the basic blocks in the function.
|
|
std::vector<unsigned> BlockSizes;
|
|
|
|
virtual bool runOnMachineFunction(MachineFunction &Fn);
|
|
|
|
virtual const char *getPassName() const {
|
|
return "PowerPC Branch Selector";
|
|
}
|
|
};
|
|
}
|
|
|
|
/// createPPCBranchSelectionPass - returns an instance of the Branch Selection
|
|
/// Pass
|
|
///
|
|
FunctionPass *llvm::createPPCBranchSelectionPass() {
|
|
return new PPCBSel();
|
|
}
|
|
|
|
/// getNumBytesForInstruction - Return the number of bytes of code the specified
|
|
/// instruction may be. This returns the maximum number of bytes.
|
|
///
|
|
static unsigned getNumBytesForInstruction(MachineInstr *MI) {
|
|
switch (MI->getOpcode()) {
|
|
case PPC::IMPLICIT_DEF_GPRC: // no asm emitted
|
|
case PPC::IMPLICIT_DEF_G8RC: // no asm emitted
|
|
case PPC::IMPLICIT_DEF_F4: // no asm emitted
|
|
case PPC::IMPLICIT_DEF_F8: // no asm emitted
|
|
case PPC::IMPLICIT_DEF_VRRC: // no asm emitted
|
|
return 0;
|
|
case PPC::INLINEASM: { // Inline Asm: Variable size.
|
|
MachineFunction *MF = MI->getParent()->getParent();
|
|
const char *AsmStr = MI->getOperand(0).getSymbolName();
|
|
return MF->getTarget().getTargetAsmInfo()->getInlineAsmLength(AsmStr);
|
|
}
|
|
default:
|
|
return 4; // PowerPC instructions are all 4 bytes
|
|
}
|
|
}
|
|
|
|
|
|
bool PPCBSel::runOnMachineFunction(MachineFunction &Fn) {
|
|
// Give the blocks of the function a dense, in-order, numbering.
|
|
Fn.RenumberBlocks();
|
|
BlockSizes.resize(Fn.getNumBlockIDs());
|
|
|
|
// Measure each MBB and compute a size for the entire function.
|
|
unsigned FuncSize = 0;
|
|
for (MachineFunction::iterator MFI = Fn.begin(), E = Fn.end(); MFI != E;
|
|
++MFI) {
|
|
MachineBasicBlock *MBB = MFI;
|
|
|
|
unsigned BlockSize = 0;
|
|
for (MachineBasicBlock::iterator MBBI = MBB->begin(), EE = MBB->end();
|
|
MBBI != EE; ++MBBI)
|
|
BlockSize += getNumBytesForInstruction(MBBI);
|
|
|
|
BlockSizes[MBB->getNumber()] = BlockSize;
|
|
FuncSize += BlockSize;
|
|
}
|
|
|
|
// If the entire function is smaller than the displacement of a branch field,
|
|
// we know we don't need to shrink any branches in this function. This is a
|
|
// common case.
|
|
if (FuncSize < (1 << 15)) {
|
|
BlockSizes.clear();
|
|
return false;
|
|
}
|
|
|
|
// For each conditional branch, if the offset to its destination is larger
|
|
// than the offset field allows, transform it into a long branch sequence
|
|
// like this:
|
|
// short branch:
|
|
// bCC MBB
|
|
// long branch:
|
|
// b!CC $PC+8
|
|
// b MBB
|
|
//
|
|
bool MadeChange = true;
|
|
bool EverMadeChange = false;
|
|
while (MadeChange) {
|
|
// Iteratively expand branches until we reach a fixed point.
|
|
MadeChange = false;
|
|
|
|
for (MachineFunction::iterator MFI = Fn.begin(), E = Fn.end(); MFI != E;
|
|
++MFI) {
|
|
MachineBasicBlock &MBB = *MFI;
|
|
unsigned MBBStartOffset = 0;
|
|
for (MachineBasicBlock::iterator I = MBB.begin(), E = MBB.end();
|
|
I != E; ++I) {
|
|
if (I->getOpcode() != PPC::BCC || I->getOperand(2).isImm()) {
|
|
MBBStartOffset += getNumBytesForInstruction(I);
|
|
continue;
|
|
}
|
|
|
|
// Determine the offset from the current branch to the destination
|
|
// block.
|
|
MachineBasicBlock *Dest = I->getOperand(2).getMachineBasicBlock();
|
|
|
|
int BranchSize;
|
|
if (Dest->getNumber() <= MBB.getNumber()) {
|
|
// If this is a backwards branch, the delta is the offset from the
|
|
// start of this block to this branch, plus the sizes of all blocks
|
|
// from this block to the dest.
|
|
BranchSize = MBBStartOffset;
|
|
|
|
for (unsigned i = Dest->getNumber(), e = MBB.getNumber(); i != e; ++i)
|
|
BranchSize += BlockSizes[i];
|
|
} else {
|
|
// Otherwise, add the size of the blocks between this block and the
|
|
// dest to the number of bytes left in this block.
|
|
BranchSize = -MBBStartOffset;
|
|
|
|
for (unsigned i = MBB.getNumber(), e = Dest->getNumber(); i != e; ++i)
|
|
BranchSize += BlockSizes[i];
|
|
}
|
|
|
|
// If this branch is in range, ignore it.
|
|
if (isInt16(BranchSize)) {
|
|
MBBStartOffset += 4;
|
|
continue;
|
|
}
|
|
|
|
// Otherwise, we have to expand it to a long branch.
|
|
// The BCC operands are:
|
|
// 0. PPC branch predicate
|
|
// 1. CR register
|
|
// 2. Target MBB
|
|
PPC::Predicate Pred = (PPC::Predicate)I->getOperand(0).getImm();
|
|
unsigned CRReg = I->getOperand(1).getReg();
|
|
|
|
MachineInstr *OldBranch = I;
|
|
|
|
// Jump over the uncond branch inst (i.e. $PC+8) on opposite condition.
|
|
BuildMI(MBB, I, PPC::BCC, 3)
|
|
.addImm(PPC::InvertPredicate(Pred)).addReg(CRReg).addImm(2);
|
|
|
|
// Uncond branch to the real destination.
|
|
I = BuildMI(MBB, I, PPC::B, 1).addMBB(Dest);
|
|
|
|
// Remove the old branch from the function.
|
|
OldBranch->eraseFromParent();
|
|
|
|
// Remember that this instruction is 8-bytes, increase the size of the
|
|
// block by 4, remember to iterate.
|
|
BlockSizes[MBB.getNumber()] += 4;
|
|
MBBStartOffset += 8;
|
|
++NumExpanded;
|
|
MadeChange = true;
|
|
}
|
|
}
|
|
EverMadeChange |= MadeChange;
|
|
}
|
|
|
|
BlockSizes.clear();
|
|
return true;
|
|
}
|
|
|