| 1 | //===-- MSP430BranchSelector.cpp - Emit long conditional branches ---------===// | 
|---|
| 2 | // | 
|---|
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|---|
| 4 | // See https://llvm.org/LICENSE.txt for license information. | 
|---|
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|---|
| 6 | // | 
|---|
| 7 | //===----------------------------------------------------------------------===// | 
|---|
| 8 | // | 
|---|
| 9 | // This file contains a pass that scans a machine function to determine which | 
|---|
| 10 | // conditional branches need more than 10 bits of displacement to reach their | 
|---|
| 11 | // target basic block.  It does this in two passes; a calculation of basic block | 
|---|
| 12 | // positions pass, and a branch pseudo op to machine branch opcode pass.  This | 
|---|
| 13 | // pass should be run last, just before the assembly printer. | 
|---|
| 14 | // | 
|---|
| 15 | //===----------------------------------------------------------------------===// | 
|---|
| 16 |  | 
|---|
| 17 | #include "MSP430.h" | 
|---|
| 18 | #include "MSP430InstrInfo.h" | 
|---|
| 19 | #include "MSP430Subtarget.h" | 
|---|
| 20 | #include "llvm/ADT/Statistic.h" | 
|---|
| 21 | #include "llvm/CodeGen/MachineFunctionPass.h" | 
|---|
| 22 | #include "llvm/CodeGen/MachineInstrBuilder.h" | 
|---|
| 23 | #include "llvm/Support/Debug.h" | 
|---|
| 24 | #include "llvm/Support/MathExtras.h" | 
|---|
| 25 | #include "llvm/Target/TargetMachine.h" | 
|---|
| 26 | using namespace llvm; | 
|---|
| 27 |  | 
|---|
| 28 | #define DEBUG_TYPE "msp430-branch-select" | 
|---|
| 29 |  | 
|---|
| 30 | static cl::opt<bool> | 
|---|
| 31 | BranchSelectEnabled( "msp430-branch-select", cl::Hidden, cl::init(Val: true), | 
|---|
| 32 | cl::desc( "Expand out of range branches")); | 
|---|
| 33 |  | 
|---|
| 34 | STATISTIC(NumSplit, "Number of machine basic blocks split"); | 
|---|
| 35 | STATISTIC(NumExpanded, "Number of branches expanded to long format"); | 
|---|
| 36 |  | 
|---|
| 37 | namespace { | 
|---|
| 38 | class MSP430BSel : public MachineFunctionPass { | 
|---|
| 39 |  | 
|---|
| 40 | typedef SmallVector<int, 16> OffsetVector; | 
|---|
| 41 |  | 
|---|
| 42 | MachineFunction *MF; | 
|---|
| 43 | const MSP430InstrInfo *TII; | 
|---|
| 44 |  | 
|---|
| 45 | unsigned measureFunction(OffsetVector &BlockOffsets, | 
|---|
| 46 | MachineBasicBlock *FromBB = nullptr); | 
|---|
| 47 | bool expandBranches(OffsetVector &BlockOffsets); | 
|---|
| 48 |  | 
|---|
| 49 | public: | 
|---|
| 50 | static char ID; | 
|---|
| 51 | MSP430BSel() : MachineFunctionPass(ID) {} | 
|---|
| 52 |  | 
|---|
| 53 | bool runOnMachineFunction(MachineFunction &MF) override; | 
|---|
| 54 |  | 
|---|
| 55 | MachineFunctionProperties getRequiredProperties() const override { | 
|---|
| 56 | return MachineFunctionProperties().setNoVRegs(); | 
|---|
| 57 | } | 
|---|
| 58 |  | 
|---|
| 59 | StringRef getPassName() const override { return "MSP430 Branch Selector"; } | 
|---|
| 60 | }; | 
|---|
| 61 | char MSP430BSel::ID = 0; | 
|---|
| 62 | } | 
|---|
| 63 |  | 
|---|
| 64 | static bool isInRage(int DistanceInBytes) { | 
|---|
| 65 | // According to CC430 Family User's Guide, Section 4.5.1.3, branch | 
|---|
| 66 | // instructions have the signed 10-bit word offset field, so first we need to | 
|---|
| 67 | // convert the distance from bytes to words, then check if it fits in 10-bit | 
|---|
| 68 | // signed integer. | 
|---|
| 69 | const int WordSize = 2; | 
|---|
| 70 |  | 
|---|
| 71 | assert((DistanceInBytes % WordSize == 0) && | 
|---|
| 72 | "Branch offset should be word aligned!"); | 
|---|
| 73 |  | 
|---|
| 74 | int Words = DistanceInBytes / WordSize; | 
|---|
| 75 | return isInt<10>(x: Words); | 
|---|
| 76 | } | 
|---|
| 77 |  | 
|---|
| 78 | /// Measure each basic block, fill the BlockOffsets, and return the size of | 
|---|
| 79 | /// the function, starting with BB | 
|---|
| 80 | unsigned MSP430BSel::measureFunction(OffsetVector &BlockOffsets, | 
|---|
| 81 | MachineBasicBlock *FromBB) { | 
|---|
| 82 | // Give the blocks of the function a dense, in-order, numbering. | 
|---|
| 83 | MF->RenumberBlocks(MBBFrom: FromBB); | 
|---|
| 84 |  | 
|---|
| 85 | MachineFunction::iterator Begin; | 
|---|
| 86 | if (FromBB == nullptr) { | 
|---|
| 87 | Begin = MF->begin(); | 
|---|
| 88 | } else { | 
|---|
| 89 | Begin = FromBB->getIterator(); | 
|---|
| 90 | } | 
|---|
| 91 |  | 
|---|
| 92 | BlockOffsets.resize(N: MF->getNumBlockIDs()); | 
|---|
| 93 |  | 
|---|
| 94 | unsigned TotalSize = BlockOffsets[Begin->getNumber()]; | 
|---|
| 95 | for (auto &MBB : make_range(x: Begin, y: MF->end())) { | 
|---|
| 96 | BlockOffsets[MBB.getNumber()] = TotalSize; | 
|---|
| 97 | for (MachineInstr &MI : MBB) { | 
|---|
| 98 | TotalSize += TII->getInstSizeInBytes(MI); | 
|---|
| 99 | } | 
|---|
| 100 | } | 
|---|
| 101 | return TotalSize; | 
|---|
| 102 | } | 
|---|
| 103 |  | 
|---|
| 104 | /// Do expand branches and split the basic blocks if necessary. | 
|---|
| 105 | /// Returns true if made any change. | 
|---|
| 106 | bool MSP430BSel::expandBranches(OffsetVector &BlockOffsets) { | 
|---|
| 107 | // For each conditional branch, if the offset to its destination is larger | 
|---|
| 108 | // than the offset field allows, transform it into a long branch sequence | 
|---|
| 109 | // like this: | 
|---|
| 110 | //   short branch: | 
|---|
| 111 | //     bCC MBB | 
|---|
| 112 | //   long branch: | 
|---|
| 113 | //     b!CC $PC+6 | 
|---|
| 114 | //     b MBB | 
|---|
| 115 | // | 
|---|
| 116 | bool MadeChange = false; | 
|---|
| 117 | for (auto MBB = MF->begin(), E = MF->end(); MBB != E; ++MBB) { | 
|---|
| 118 | unsigned MBBStartOffset = 0; | 
|---|
| 119 | for (auto MI = MBB->begin(), EE = MBB->end(); MI != EE; ++MI) { | 
|---|
| 120 | MBBStartOffset += TII->getInstSizeInBytes(MI: *MI); | 
|---|
| 121 |  | 
|---|
| 122 | // If this instruction is not a short branch then skip it. | 
|---|
| 123 | if (MI->getOpcode() != MSP430::JCC && MI->getOpcode() != MSP430::JMP) { | 
|---|
| 124 | continue; | 
|---|
| 125 | } | 
|---|
| 126 |  | 
|---|
| 127 | MachineBasicBlock *DestBB = MI->getOperand(i: 0).getMBB(); | 
|---|
| 128 | // Determine the distance from the current branch to the destination | 
|---|
| 129 | // block. MBBStartOffset already includes the size of the current branch | 
|---|
| 130 | // instruction. | 
|---|
| 131 | int BlockDistance = | 
|---|
| 132 | BlockOffsets[DestBB->getNumber()] - BlockOffsets[MBB->getNumber()]; | 
|---|
| 133 | int BranchDistance = BlockDistance - MBBStartOffset; | 
|---|
| 134 |  | 
|---|
| 135 | // If this branch is in range, ignore it. | 
|---|
| 136 | if (isInRage(DistanceInBytes: BranchDistance)) { | 
|---|
| 137 | continue; | 
|---|
| 138 | } | 
|---|
| 139 |  | 
|---|
| 140 | LLVM_DEBUG(dbgs() << "  Found a branch that needs expanding, " | 
|---|
| 141 | << printMBBReference(*DestBB) << ", Distance " | 
|---|
| 142 | << BranchDistance << "\n"); | 
|---|
| 143 |  | 
|---|
| 144 | // If JCC is not the last instruction we need to split the MBB. | 
|---|
| 145 | if (MI->getOpcode() == MSP430::JCC && std::next(x: MI) != EE) { | 
|---|
| 146 |  | 
|---|
| 147 | LLVM_DEBUG(dbgs() << "  Found a basic block that needs to be split, " | 
|---|
| 148 | << printMBBReference(*MBB) << "\n"); | 
|---|
| 149 |  | 
|---|
| 150 | // Create a new basic block. | 
|---|
| 151 | MachineBasicBlock *NewBB = | 
|---|
| 152 | MF->CreateMachineBasicBlock(BB: MBB->getBasicBlock()); | 
|---|
| 153 | MF->insert(MBBI: std::next(x: MBB), MBB: NewBB); | 
|---|
| 154 |  | 
|---|
| 155 | // Splice the instructions following MI over to the NewBB. | 
|---|
| 156 | NewBB->splice(Where: NewBB->end(), Other: &*MBB, From: std::next(x: MI), To: MBB->end()); | 
|---|
| 157 |  | 
|---|
| 158 | // Update the successor lists. | 
|---|
| 159 | for (MachineBasicBlock *Succ : MBB->successors()) { | 
|---|
| 160 | if (Succ == DestBB) { | 
|---|
| 161 | continue; | 
|---|
| 162 | } | 
|---|
| 163 | MBB->replaceSuccessor(Old: Succ, New: NewBB); | 
|---|
| 164 | NewBB->addSuccessor(Succ); | 
|---|
| 165 | } | 
|---|
| 166 |  | 
|---|
| 167 | // We introduced a new MBB so all following blocks should be numbered | 
|---|
| 168 | // and measured again. | 
|---|
| 169 | measureFunction(BlockOffsets, FromBB: &*MBB); | 
|---|
| 170 |  | 
|---|
| 171 | ++NumSplit; | 
|---|
| 172 |  | 
|---|
| 173 | // It may be not necessary to start all over at this point, but it's | 
|---|
| 174 | // safer do this anyway. | 
|---|
| 175 | return true; | 
|---|
| 176 | } | 
|---|
| 177 |  | 
|---|
| 178 | MachineInstr &OldBranch = *MI; | 
|---|
| 179 | DebugLoc dl = OldBranch.getDebugLoc(); | 
|---|
| 180 | int InstrSizeDiff = -TII->getInstSizeInBytes(MI: OldBranch); | 
|---|
| 181 |  | 
|---|
| 182 | if (MI->getOpcode() == MSP430::JCC) { | 
|---|
| 183 | MachineBasicBlock *NextMBB = &*std::next(x: MBB); | 
|---|
| 184 | assert(MBB->isSuccessor(NextMBB) && | 
|---|
| 185 | "This block must have a layout successor!"); | 
|---|
| 186 |  | 
|---|
| 187 | // The BCC operands are: | 
|---|
| 188 | // 0. Target MBB | 
|---|
| 189 | // 1. MSP430 branch predicate | 
|---|
| 190 | SmallVector<MachineOperand, 1> Cond; | 
|---|
| 191 | Cond.push_back(Elt: MI->getOperand(i: 1)); | 
|---|
| 192 |  | 
|---|
| 193 | // Jump over the long branch on the opposite condition | 
|---|
| 194 | TII->reverseBranchCondition(Cond); | 
|---|
| 195 | MI = BuildMI(BB&: *MBB, I: MI, MIMD: dl, MCID: TII->get(Opcode: MSP430::JCC)) | 
|---|
| 196 | .addMBB(MBB: NextMBB) | 
|---|
| 197 | .add(MO: Cond[0]); | 
|---|
| 198 | InstrSizeDiff += TII->getInstSizeInBytes(MI: *MI); | 
|---|
| 199 | ++MI; | 
|---|
| 200 | } | 
|---|
| 201 |  | 
|---|
| 202 | // Unconditional branch to the real destination. | 
|---|
| 203 | MI = BuildMI(BB&: *MBB, I: MI, MIMD: dl, MCID: TII->get(Opcode: MSP430::Bi)).addMBB(MBB: DestBB); | 
|---|
| 204 | InstrSizeDiff += TII->getInstSizeInBytes(MI: *MI); | 
|---|
| 205 |  | 
|---|
| 206 | // Remove the old branch from the function. | 
|---|
| 207 | OldBranch.eraseFromParent(); | 
|---|
| 208 |  | 
|---|
| 209 | // The size of a new instruction is different from the old one, so we need | 
|---|
| 210 | // to correct all block offsets. | 
|---|
| 211 | for (int i = MBB->getNumber() + 1, e = BlockOffsets.size(); i < e; ++i) { | 
|---|
| 212 | BlockOffsets[i] += InstrSizeDiff; | 
|---|
| 213 | } | 
|---|
| 214 | MBBStartOffset += InstrSizeDiff; | 
|---|
| 215 |  | 
|---|
| 216 | ++NumExpanded; | 
|---|
| 217 | MadeChange = true; | 
|---|
| 218 | } | 
|---|
| 219 | } | 
|---|
| 220 | return MadeChange; | 
|---|
| 221 | } | 
|---|
| 222 |  | 
|---|
| 223 | bool MSP430BSel::runOnMachineFunction(MachineFunction &mf) { | 
|---|
| 224 | MF = &mf; | 
|---|
| 225 | TII = static_cast<const MSP430InstrInfo *>(MF->getSubtarget().getInstrInfo()); | 
|---|
| 226 |  | 
|---|
| 227 | // If the pass is disabled, just bail early. | 
|---|
| 228 | if (!BranchSelectEnabled) | 
|---|
| 229 | return false; | 
|---|
| 230 |  | 
|---|
| 231 | LLVM_DEBUG(dbgs() << "\n********** "<< getPassName() << " **********\n"); | 
|---|
| 232 |  | 
|---|
| 233 | // BlockOffsets - Contains the distance from the beginning of the function to | 
|---|
| 234 | // the beginning of each basic block. | 
|---|
| 235 | OffsetVector BlockOffsets; | 
|---|
| 236 |  | 
|---|
| 237 | unsigned FunctionSize = measureFunction(BlockOffsets); | 
|---|
| 238 | // If the entire function is smaller than the displacement of a branch field, | 
|---|
| 239 | // we know we don't need to expand any branches in this | 
|---|
| 240 | // function. This is a common case. | 
|---|
| 241 | if (isInRage(DistanceInBytes: FunctionSize)) { | 
|---|
| 242 | return false; | 
|---|
| 243 | } | 
|---|
| 244 |  | 
|---|
| 245 | // Iteratively expand branches until we reach a fixed point. | 
|---|
| 246 | bool MadeChange = false; | 
|---|
| 247 | while (expandBranches(BlockOffsets)) | 
|---|
| 248 | MadeChange = true; | 
|---|
| 249 |  | 
|---|
| 250 | return MadeChange; | 
|---|
| 251 | } | 
|---|
| 252 |  | 
|---|
| 253 | /// Returns an instance of the Branch Selection Pass | 
|---|
| 254 | FunctionPass *llvm::createMSP430BranchSelectionPass() { | 
|---|
| 255 | return new MSP430BSel(); | 
|---|
| 256 | } | 
|---|
| 257 |  | 
|---|