| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 1 | //===-- LowerSubregs.cpp - Subregister Lowering instruction pass ----------===// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
| Chris Lattner | f3ebc3f | 2007-12-29 20:36:04 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
| Dan Gohman | 382e2ec | 2008-09-24 23:44:12 +0000 | [diff] [blame] | 9 | // | 
|  | 10 | // This file defines a MachineFunction pass which runs after register | 
|  | 11 | // allocation that turns subreg insert/extract instructions into register | 
|  | 12 | // copies, as needed. This ensures correct codegen even if the coalescer | 
|  | 13 | // isn't able to remove all subreg instructions. | 
|  | 14 | // | 
|  | 15 | //===----------------------------------------------------------------------===// | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 16 |  | 
|  | 17 | #define DEBUG_TYPE "lowersubregs" | 
|  | 18 | #include "llvm/CodeGen/Passes.h" | 
|  | 19 | #include "llvm/Function.h" | 
|  | 20 | #include "llvm/CodeGen/MachineFunctionPass.h" | 
|  | 21 | #include "llvm/CodeGen/MachineInstr.h" | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 22 | #include "llvm/CodeGen/MachineInstrBuilder.h" | 
| Chris Lattner | a10fff5 | 2007-12-31 04:13:23 +0000 | [diff] [blame] | 23 | #include "llvm/CodeGen/MachineRegisterInfo.h" | 
| Dan Gohman | 3a4be0f | 2008-02-10 18:45:23 +0000 | [diff] [blame] | 24 | #include "llvm/Target/TargetRegisterInfo.h" | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 25 | #include "llvm/Target/TargetInstrInfo.h" | 
|  | 26 | #include "llvm/Target/TargetMachine.h" | 
|  | 27 | #include "llvm/Support/Debug.h" | 
| Daniel Dunbar | 0dd5e1e | 2009-07-25 00:23:56 +0000 | [diff] [blame] | 28 | #include "llvm/Support/raw_ostream.h" | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 29 | using namespace llvm; | 
|  | 30 |  | 
|  | 31 | namespace { | 
| Nick Lewycky | 02d5f77 | 2009-10-25 06:33:48 +0000 | [diff] [blame] | 32 | struct LowerSubregsInstructionPass : public MachineFunctionPass { | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 33 | private: | 
|  | 34 | const TargetRegisterInfo *TRI; | 
|  | 35 | const TargetInstrInfo *TII; | 
|  | 36 |  | 
|  | 37 | public: | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 38 | static char ID; // Pass identification, replacement for typeid | 
| Dan Gohman | a79db30 | 2008-09-04 17:05:41 +0000 | [diff] [blame] | 39 | LowerSubregsInstructionPass() : MachineFunctionPass(&ID) {} | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 40 |  | 
|  | 41 | const char *getPassName() const { | 
|  | 42 | return "Subregister lowering instruction pass"; | 
|  | 43 | } | 
|  | 44 |  | 
| Evan Cheng | 168f8f3 | 2008-09-22 20:58:04 +0000 | [diff] [blame] | 45 | virtual void getAnalysisUsage(AnalysisUsage &AU) const { | 
| Dan Gohman | 0402315 | 2009-07-31 23:37:33 +0000 | [diff] [blame] | 46 | AU.setPreservesCFG(); | 
| Evan Cheng | 962c2cf | 2008-09-22 22:21:38 +0000 | [diff] [blame] | 47 | AU.addPreservedID(MachineLoopInfoID); | 
|  | 48 | AU.addPreservedID(MachineDominatorsID); | 
| Evan Cheng | 168f8f3 | 2008-09-22 20:58:04 +0000 | [diff] [blame] | 49 | MachineFunctionPass::getAnalysisUsage(AU); | 
|  | 50 | } | 
|  | 51 |  | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 52 | /// runOnMachineFunction - pass entry point | 
|  | 53 | bool runOnMachineFunction(MachineFunction&); | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 54 |  | 
|  | 55 | private: | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 56 | bool LowerExtract(MachineInstr *MI); | 
|  | 57 | bool LowerInsert(MachineInstr *MI); | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 58 | bool LowerSubregToReg(MachineInstr *MI); | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 59 |  | 
|  | 60 | void TransferDeadFlag(MachineInstr *MI, unsigned DstReg, | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 61 | const TargetRegisterInfo *TRI); | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 62 | void TransferKillFlag(MachineInstr *MI, unsigned SrcReg, | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 63 | const TargetRegisterInfo *TRI, | 
| Evan Cheng | 1f7b549 | 2009-08-05 02:25:11 +0000 | [diff] [blame] | 64 | bool AddIfNotFound = false); | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 65 | }; | 
|  | 66 |  | 
|  | 67 | char LowerSubregsInstructionPass::ID = 0; | 
|  | 68 | } | 
|  | 69 |  | 
|  | 70 | FunctionPass *llvm::createLowerSubregsPass() { | 
|  | 71 | return new LowerSubregsInstructionPass(); | 
|  | 72 | } | 
|  | 73 |  | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 74 | /// TransferDeadFlag - MI is a pseudo-instruction with DstReg dead, | 
|  | 75 | /// and the lowered replacement instructions immediately precede it. | 
|  | 76 | /// Mark the replacement instructions with the dead flag. | 
|  | 77 | void | 
|  | 78 | LowerSubregsInstructionPass::TransferDeadFlag(MachineInstr *MI, | 
|  | 79 | unsigned DstReg, | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 80 | const TargetRegisterInfo *TRI) { | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 81 | for (MachineBasicBlock::iterator MII = | 
|  | 82 | prior(MachineBasicBlock::iterator(MI)); ; --MII) { | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 83 | if (MII->addRegisterDead(DstReg, TRI)) | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 84 | break; | 
|  | 85 | assert(MII != MI->getParent()->begin() && | 
|  | 86 | "copyRegToReg output doesn't reference destination register!"); | 
|  | 87 | } | 
|  | 88 | } | 
|  | 89 |  | 
|  | 90 | /// TransferKillFlag - MI is a pseudo-instruction with SrcReg killed, | 
|  | 91 | /// and the lowered replacement instructions immediately precede it. | 
|  | 92 | /// Mark the replacement instructions with the kill flag. | 
|  | 93 | void | 
|  | 94 | LowerSubregsInstructionPass::TransferKillFlag(MachineInstr *MI, | 
|  | 95 | unsigned SrcReg, | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 96 | const TargetRegisterInfo *TRI, | 
| Evan Cheng | 1f7b549 | 2009-08-05 02:25:11 +0000 | [diff] [blame] | 97 | bool AddIfNotFound) { | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 98 | for (MachineBasicBlock::iterator MII = | 
|  | 99 | prior(MachineBasicBlock::iterator(MI)); ; --MII) { | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 100 | if (MII->addRegisterKilled(SrcReg, TRI, AddIfNotFound)) | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 101 | break; | 
|  | 102 | assert(MII != MI->getParent()->begin() && | 
|  | 103 | "copyRegToReg output doesn't reference source register!"); | 
|  | 104 | } | 
|  | 105 | } | 
|  | 106 |  | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 107 | bool LowerSubregsInstructionPass::LowerExtract(MachineInstr *MI) { | 
| Dan Gohman | 451afdd | 2008-12-18 22:06:01 +0000 | [diff] [blame] | 108 | MachineBasicBlock *MBB = MI->getParent(); | 
| Jakob Stoklund Olesen | 6304369 | 2009-08-04 20:01:11 +0000 | [diff] [blame] | 109 |  | 
| Dan Gohman | 451afdd | 2008-12-18 22:06:01 +0000 | [diff] [blame] | 110 | assert(MI->getOperand(0).isReg() && MI->getOperand(0).isDef() && | 
|  | 111 | MI->getOperand(1).isReg() && MI->getOperand(1).isUse() && | 
|  | 112 | MI->getOperand(2).isImm() && "Malformed extract_subreg"); | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 113 |  | 
| Dan Gohman | 451afdd | 2008-12-18 22:06:01 +0000 | [diff] [blame] | 114 | unsigned DstReg   = MI->getOperand(0).getReg(); | 
|  | 115 | unsigned SuperReg = MI->getOperand(1).getReg(); | 
|  | 116 | unsigned SubIdx   = MI->getOperand(2).getImm(); | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 117 | unsigned SrcReg   = TRI->getSubReg(SuperReg, SubIdx); | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 118 |  | 
| Dan Gohman | 451afdd | 2008-12-18 22:06:01 +0000 | [diff] [blame] | 119 | assert(TargetRegisterInfo::isPhysicalRegister(SuperReg) && | 
|  | 120 | "Extract supperg source must be a physical register"); | 
|  | 121 | assert(TargetRegisterInfo::isPhysicalRegister(DstReg) && | 
| Dan Gohman | d38c00c | 2008-12-18 22:07:25 +0000 | [diff] [blame] | 122 | "Extract destination must be in a physical register"); | 
| Evan Cheng | 3794292 | 2009-08-05 03:53:14 +0000 | [diff] [blame] | 123 | assert(SrcReg && "invalid subregister index for register"); | 
| Jakob Stoklund Olesen | 6304369 | 2009-08-04 20:01:11 +0000 | [diff] [blame] | 124 |  | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 125 | DEBUG(dbgs() << "subreg: CONVERTING: " << *MI); | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 126 |  | 
| Dan Gohman | c4ce336 | 2008-12-18 22:11:34 +0000 | [diff] [blame] | 127 | if (SrcReg == DstReg) { | 
| Jakob Stoklund Olesen | 6304369 | 2009-08-04 20:01:11 +0000 | [diff] [blame] | 128 | // No need to insert an identity copy instruction. | 
|  | 129 | if (MI->getOperand(1).isKill()) { | 
| Jakob Stoklund Olesen | 0bb5af3 | 2009-09-28 20:32:46 +0000 | [diff] [blame] | 130 | // We must make sure the super-register gets killed. Replace the | 
|  | 131 | // instruction with KILL. | 
| Chris Lattner | b06015a | 2010-02-09 19:54:29 +0000 | [diff] [blame] | 132 | MI->setDesc(TII->get(TargetOpcode::KILL)); | 
| Jakob Stoklund Olesen | 6304369 | 2009-08-04 20:01:11 +0000 | [diff] [blame] | 133 | MI->RemoveOperand(2);     // SubIdx | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 134 | DEBUG(dbgs() << "subreg: replace by: " << *MI); | 
| Jakob Stoklund Olesen | 6304369 | 2009-08-04 20:01:11 +0000 | [diff] [blame] | 135 | return true; | 
|  | 136 | } | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 137 |  | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 138 | DEBUG(dbgs() << "subreg: eliminated!"); | 
| Dan Gohman | c4ce336 | 2008-12-18 22:11:34 +0000 | [diff] [blame] | 139 | } else { | 
|  | 140 | // Insert copy | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 141 | const TargetRegisterClass *TRCS = TRI->getPhysicalRegisterRegClass(DstReg); | 
|  | 142 | const TargetRegisterClass *TRCD = TRI->getPhysicalRegisterRegClass(SrcReg); | 
|  | 143 | bool Emitted = TII->copyRegToReg(*MBB, MI, DstReg, SrcReg, TRCD, TRCS); | 
| Anton Korobeynikov | 12996ea | 2009-07-16 13:55:26 +0000 | [diff] [blame] | 144 | (void)Emitted; | 
|  | 145 | assert(Emitted && "Subreg and Dst must be of compatible register class"); | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 146 | // Transfer the kill/dead flags, if needed. | 
|  | 147 | if (MI->getOperand(0).isDead()) | 
|  | 148 | TransferDeadFlag(MI, DstReg, TRI); | 
|  | 149 | if (MI->getOperand(1).isKill()) | 
| Evan Cheng | 1f7b549 | 2009-08-05 02:25:11 +0000 | [diff] [blame] | 150 | TransferKillFlag(MI, SuperReg, TRI, true); | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 151 | DEBUG({ | 
|  | 152 | MachineBasicBlock::iterator dMI = MI; | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 153 | dbgs() << "subreg: " << *(--dMI); | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 154 | }); | 
| Dan Gohman | 451afdd | 2008-12-18 22:06:01 +0000 | [diff] [blame] | 155 | } | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 156 |  | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 157 | DEBUG(dbgs() << '\n'); | 
| Dan Gohman | 451afdd | 2008-12-18 22:06:01 +0000 | [diff] [blame] | 158 | MBB->erase(MI); | 
|  | 159 | return true; | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 160 | } | 
|  | 161 |  | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 162 | bool LowerSubregsInstructionPass::LowerSubregToReg(MachineInstr *MI) { | 
|  | 163 | MachineBasicBlock *MBB = MI->getParent(); | 
| Dan Gohman | 0d1e9a8 | 2008-10-03 15:45:36 +0000 | [diff] [blame] | 164 | assert((MI->getOperand(0).isReg() && MI->getOperand(0).isDef()) && | 
|  | 165 | MI->getOperand(1).isImm() && | 
|  | 166 | (MI->getOperand(2).isReg() && MI->getOperand(2).isUse()) && | 
|  | 167 | MI->getOperand(3).isImm() && "Invalid subreg_to_reg"); | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 168 |  | 
|  | 169 | unsigned DstReg  = MI->getOperand(0).getReg(); | 
|  | 170 | unsigned InsReg  = MI->getOperand(2).getReg(); | 
| Evan Cheng | 47c9750 | 2009-03-23 07:19:58 +0000 | [diff] [blame] | 171 | unsigned InsSIdx = MI->getOperand(2).getSubReg(); | 
|  | 172 | unsigned SubIdx  = MI->getOperand(3).getImm(); | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 173 |  | 
|  | 174 | assert(SubIdx != 0 && "Invalid index for insert_subreg"); | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 175 | unsigned DstSubReg = TRI->getSubReg(DstReg, SubIdx); | 
| Evan Cheng | 47c9750 | 2009-03-23 07:19:58 +0000 | [diff] [blame] | 176 |  | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 177 | assert(TargetRegisterInfo::isPhysicalRegister(DstReg) && | 
|  | 178 | "Insert destination must be in a physical register"); | 
|  | 179 | assert(TargetRegisterInfo::isPhysicalRegister(InsReg) && | 
|  | 180 | "Inserted value must be in a physical register"); | 
|  | 181 |  | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 182 | DEBUG(dbgs() << "subreg: CONVERTING: " << *MI); | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 183 |  | 
| Evan Cheng | 47c9750 | 2009-03-23 07:19:58 +0000 | [diff] [blame] | 184 | if (DstSubReg == InsReg && InsSIdx == 0) { | 
| Dan Gohman | 527ca7e | 2008-08-07 02:54:50 +0000 | [diff] [blame] | 185 | // No need to insert an identify copy instruction. | 
| Evan Cheng | 47c9750 | 2009-03-23 07:19:58 +0000 | [diff] [blame] | 186 | // Watch out for case like this: | 
|  | 187 | // %RAX<def> = ... | 
|  | 188 | // %RAX<def> = SUBREG_TO_REG 0, %EAX:3<kill>, 3 | 
|  | 189 | // The first def is defining RAX, not EAX so the top bits were not | 
|  | 190 | // zero extended. | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 191 | DEBUG(dbgs() << "subreg: eliminated!"); | 
| Dan Gohman | 527ca7e | 2008-08-07 02:54:50 +0000 | [diff] [blame] | 192 | } else { | 
|  | 193 | // Insert sub-register copy | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 194 | const TargetRegisterClass *TRC0= TRI->getPhysicalRegisterRegClass(DstSubReg); | 
|  | 195 | const TargetRegisterClass *TRC1= TRI->getPhysicalRegisterRegClass(InsReg); | 
|  | 196 | bool Emitted = TII->copyRegToReg(*MBB, MI, DstSubReg, InsReg, TRC0, TRC1); | 
| Anton Korobeynikov | b4a1347 | 2009-10-24 00:27:00 +0000 | [diff] [blame] | 197 | (void)Emitted; | 
|  | 198 | assert(Emitted && "Subreg and Dst must be of compatible register class"); | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 199 | // Transfer the kill/dead flags, if needed. | 
|  | 200 | if (MI->getOperand(0).isDead()) | 
|  | 201 | TransferDeadFlag(MI, DstSubReg, TRI); | 
|  | 202 | if (MI->getOperand(2).isKill()) | 
|  | 203 | TransferKillFlag(MI, InsReg, TRI); | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 204 | DEBUG({ | 
|  | 205 | MachineBasicBlock::iterator dMI = MI; | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 206 | dbgs() << "subreg: " << *(--dMI); | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 207 | }); | 
| Dan Gohman | 527ca7e | 2008-08-07 02:54:50 +0000 | [diff] [blame] | 208 | } | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 209 |  | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 210 | DEBUG(dbgs() << '\n'); | 
| Dan Gohman | 0ece943 | 2008-07-17 23:49:46 +0000 | [diff] [blame] | 211 | MBB->erase(MI); | 
| Anton Korobeynikov | b4a1347 | 2009-10-24 00:27:00 +0000 | [diff] [blame] | 212 | return true; | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 213 | } | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 214 |  | 
|  | 215 | bool LowerSubregsInstructionPass::LowerInsert(MachineInstr *MI) { | 
|  | 216 | MachineBasicBlock *MBB = MI->getParent(); | 
| Dan Gohman | 0d1e9a8 | 2008-10-03 15:45:36 +0000 | [diff] [blame] | 217 | assert((MI->getOperand(0).isReg() && MI->getOperand(0).isDef()) && | 
|  | 218 | (MI->getOperand(1).isReg() && MI->getOperand(1).isUse()) && | 
|  | 219 | (MI->getOperand(2).isReg() && MI->getOperand(2).isUse()) && | 
|  | 220 | MI->getOperand(3).isImm() && "Invalid insert_subreg"); | 
| Christopher Lamb | aa7c210 | 2008-03-11 10:09:17 +0000 | [diff] [blame] | 221 |  | 
|  | 222 | unsigned DstReg = MI->getOperand(0).getReg(); | 
| Devang Patel | cb181bb | 2008-11-21 20:00:59 +0000 | [diff] [blame] | 223 | #ifndef NDEBUG | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 224 | unsigned SrcReg = MI->getOperand(1).getReg(); | 
| Devang Patel | cb181bb | 2008-11-21 20:00:59 +0000 | [diff] [blame] | 225 | #endif | 
| Christopher Lamb | aa7c210 | 2008-03-11 10:09:17 +0000 | [diff] [blame] | 226 | unsigned InsReg = MI->getOperand(2).getReg(); | 
|  | 227 | unsigned SubIdx = MI->getOperand(3).getImm(); | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 228 |  | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 229 | assert(DstReg == SrcReg && "insert_subreg not a two-address instruction?"); | 
|  | 230 | assert(SubIdx != 0 && "Invalid index for insert_subreg"); | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 231 | unsigned DstSubReg = TRI->getSubReg(DstReg, SubIdx); | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 232 | assert(DstSubReg && "invalid subregister index for register"); | 
| Dan Gohman | 3a4be0f | 2008-02-10 18:45:23 +0000 | [diff] [blame] | 233 | assert(TargetRegisterInfo::isPhysicalRegister(SrcReg) && | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 234 | "Insert superreg source must be in a physical register"); | 
| Dan Gohman | 3a4be0f | 2008-02-10 18:45:23 +0000 | [diff] [blame] | 235 | assert(TargetRegisterInfo::isPhysicalRegister(InsReg) && | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 236 | "Inserted value must be in a physical register"); | 
|  | 237 |  | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 238 | DEBUG(dbgs() << "subreg: CONVERTING: " << *MI); | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 239 |  | 
| Evan Cheng | 1cde1f8 | 2008-06-16 22:52:53 +0000 | [diff] [blame] | 240 | if (DstSubReg == InsReg) { | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 241 | // No need to insert an identity copy instruction. If the SrcReg was | 
| Jakob Stoklund Olesen | 0bb5af3 | 2009-09-28 20:32:46 +0000 | [diff] [blame] | 242 | // <undef>, we need to make sure it is alive by inserting a KILL | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 243 | if (MI->getOperand(1).isUndef() && !MI->getOperand(0).isDead()) { | 
| Evan Cheng | 6376367 | 2009-08-05 01:57:22 +0000 | [diff] [blame] | 244 | MachineInstrBuilder MIB = BuildMI(*MBB, MI, MI->getDebugLoc(), | 
| Chris Lattner | b06015a | 2010-02-09 19:54:29 +0000 | [diff] [blame] | 245 | TII->get(TargetOpcode::KILL), DstReg); | 
| Evan Cheng | 6376367 | 2009-08-05 01:57:22 +0000 | [diff] [blame] | 246 | if (MI->getOperand(2).isUndef()) | 
| Jakob Stoklund Olesen | 0bb5af3 | 2009-09-28 20:32:46 +0000 | [diff] [blame] | 247 | MIB.addReg(InsReg, RegState::Undef); | 
| Evan Cheng | 6376367 | 2009-08-05 01:57:22 +0000 | [diff] [blame] | 248 | else | 
| Jakob Stoklund Olesen | 0bb5af3 | 2009-09-28 20:32:46 +0000 | [diff] [blame] | 249 | MIB.addReg(InsReg, RegState::Kill); | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 250 | } else { | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 251 | DEBUG(dbgs() << "subreg: eliminated!\n"); | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 252 | MBB->erase(MI); | 
|  | 253 | return true; | 
|  | 254 | } | 
| Evan Cheng | 1cde1f8 | 2008-06-16 22:52:53 +0000 | [diff] [blame] | 255 | } else { | 
|  | 256 | // Insert sub-register copy | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 257 | const TargetRegisterClass *TRC0= TRI->getPhysicalRegisterRegClass(DstSubReg); | 
|  | 258 | const TargetRegisterClass *TRC1= TRI->getPhysicalRegisterRegClass(InsReg); | 
| Evan Cheng | cdb125c | 2009-08-05 01:29:24 +0000 | [diff] [blame] | 259 | if (MI->getOperand(2).isUndef()) | 
| Jakob Stoklund Olesen | 0bb5af3 | 2009-09-28 20:32:46 +0000 | [diff] [blame] | 260 | // If the source register being inserted is undef, then this becomes a | 
|  | 261 | // KILL. | 
| Evan Cheng | cdb125c | 2009-08-05 01:29:24 +0000 | [diff] [blame] | 262 | BuildMI(*MBB, MI, MI->getDebugLoc(), | 
| Chris Lattner | b06015a | 2010-02-09 19:54:29 +0000 | [diff] [blame] | 263 | TII->get(TargetOpcode::KILL), DstSubReg); | 
| Anton Korobeynikov | b4a1347 | 2009-10-24 00:27:00 +0000 | [diff] [blame] | 264 | else { | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 265 | bool Emitted = TII->copyRegToReg(*MBB, MI, DstSubReg, InsReg, TRC0, TRC1); | 
| Anton Korobeynikov | b4a1347 | 2009-10-24 00:27:00 +0000 | [diff] [blame] | 266 | (void)Emitted; | 
|  | 267 | assert(Emitted && "Subreg and Dst must be of compatible register class"); | 
|  | 268 | } | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 269 | MachineBasicBlock::iterator CopyMI = MI; | 
|  | 270 | --CopyMI; | 
|  | 271 |  | 
| Jakob Stoklund Olesen | 8e96c6a | 2009-08-08 13:19:10 +0000 | [diff] [blame] | 272 | // INSERT_SUBREG is a two-address instruction so it implicitly kills SrcReg. | 
|  | 273 | if (!MI->getOperand(1).isUndef()) | 
|  | 274 | CopyMI->addOperand(MachineOperand::CreateReg(DstReg, false, true, true)); | 
|  | 275 |  | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 276 | // Transfer the kill/dead flags, if needed. | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 277 | if (MI->getOperand(0).isDead()) { | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 278 | TransferDeadFlag(MI, DstSubReg, TRI); | 
| Jakob Stoklund Olesen | 8e96c6a | 2009-08-08 13:19:10 +0000 | [diff] [blame] | 279 | } else { | 
|  | 280 | // Make sure the full DstReg is live after this replacement. | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 281 | CopyMI->addOperand(MachineOperand::CreateReg(DstReg, true, true)); | 
|  | 282 | } | 
|  | 283 |  | 
|  | 284 | // Make sure the inserted register gets killed | 
| Evan Cheng | cdb125c | 2009-08-05 01:29:24 +0000 | [diff] [blame] | 285 | if (MI->getOperand(2).isKill() && !MI->getOperand(2).isUndef()) | 
| Dan Gohman | 9abd04b | 2008-12-18 22:14:08 +0000 | [diff] [blame] | 286 | TransferKillFlag(MI, InsReg, TRI); | 
| Jakob Stoklund Olesen | 5d8ace0 | 2009-08-03 20:08:18 +0000 | [diff] [blame] | 287 | } | 
| Dan Gohman | c4ce336 | 2008-12-18 22:11:34 +0000 | [diff] [blame] | 288 |  | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 289 | DEBUG({ | 
|  | 290 | MachineBasicBlock::iterator dMI = MI; | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 291 | dbgs() << "subreg: " << *(--dMI) << "\n"; | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 292 | }); | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 293 |  | 
| Dan Gohman | 0ece943 | 2008-07-17 23:49:46 +0000 | [diff] [blame] | 294 | MBB->erase(MI); | 
| Jakob Stoklund Olesen | 8e96c6a | 2009-08-08 13:19:10 +0000 | [diff] [blame] | 295 | return true; | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 296 | } | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 297 |  | 
|  | 298 | /// runOnMachineFunction - Reduce subregister inserts and extracts to register | 
|  | 299 | /// copies. | 
|  | 300 | /// | 
|  | 301 | bool LowerSubregsInstructionPass::runOnMachineFunction(MachineFunction &MF) { | 
| David Greene | c4878b1 | 2010-01-04 23:06:47 +0000 | [diff] [blame] | 302 | DEBUG(dbgs() << "Machine Function\n" | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 303 | << "********** LOWERING SUBREG INSTRS **********\n" | 
|  | 304 | << "********** Function: " | 
|  | 305 | << MF.getFunction()->getName() << '\n'); | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 306 | TRI = MF.getTarget().getRegisterInfo(); | 
|  | 307 | TII = MF.getTarget().getInstrInfo(); | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 308 |  | 
| Bill Wendling | 8d64226 | 2009-08-22 20:23:49 +0000 | [diff] [blame] | 309 | bool MadeChange = false; | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 310 |  | 
|  | 311 | for (MachineFunction::iterator mbbi = MF.begin(), mbbe = MF.end(); | 
|  | 312 | mbbi != mbbe; ++mbbi) { | 
|  | 313 | for (MachineBasicBlock::iterator mi = mbbi->begin(), me = mbbi->end(); | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 314 | mi != me;) { | 
| Chris Lattner | a48f44d | 2009-12-03 00:50:42 +0000 | [diff] [blame] | 315 | MachineBasicBlock::iterator nmi = llvm::next(mi); | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 316 | MachineInstr *MI = mi; | 
| Chris Lattner | b06015a | 2010-02-09 19:54:29 +0000 | [diff] [blame] | 317 | if (MI->isExtractSubreg()) { | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 318 | MadeChange |= LowerExtract(MI); | 
| Chris Lattner | b06015a | 2010-02-09 19:54:29 +0000 | [diff] [blame] | 319 | } else if (MI->isInsertSubreg()) { | 
| Christopher Lamb | 2e5fb9f | 2007-08-06 16:33:56 +0000 | [diff] [blame] | 320 | MadeChange |= LowerInsert(MI); | 
| Chris Lattner | b06015a | 2010-02-09 19:54:29 +0000 | [diff] [blame] | 321 | } else if (MI->isSubregToReg()) { | 
| Christopher Lamb | d3d0ad3 | 2008-03-16 03:12:01 +0000 | [diff] [blame] | 322 | MadeChange |= LowerSubregToReg(MI); | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 323 | } | 
| Evan Cheng | 5d2245b | 2009-10-25 07:49:57 +0000 | [diff] [blame] | 324 | mi = nmi; | 
| Christopher Lamb | e9d738c | 2007-07-26 08:18:32 +0000 | [diff] [blame] | 325 | } | 
|  | 326 | } | 
|  | 327 |  | 
|  | 328 | return MadeChange; | 
|  | 329 | } |