1 | //===-- MSP430InstrInfo.cpp - MSP430 Instruction Information --------------===// |
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 the MSP430 implementation of the TargetInstrInfo class. |
10 | // |
11 | //===----------------------------------------------------------------------===// |
12 | |
13 | #include "MSP430InstrInfo.h" |
14 | #include "MSP430.h" |
15 | #include "MSP430MachineFunctionInfo.h" |
16 | #include "MSP430TargetMachine.h" |
17 | #include "llvm/CodeGen/MachineFrameInfo.h" |
18 | #include "llvm/CodeGen/MachineInstrBuilder.h" |
19 | #include "llvm/CodeGen/MachineRegisterInfo.h" |
20 | #include "llvm/IR/Function.h" |
21 | #include "llvm/MC/TargetRegistry.h" |
22 | #include "llvm/Support/ErrorHandling.h" |
23 | |
24 | using namespace llvm; |
25 | |
26 | #define GET_INSTRINFO_CTOR_DTOR |
27 | #include "MSP430GenInstrInfo.inc" |
28 | |
29 | // Pin the vtable to this file. |
30 | void MSP430InstrInfo::anchor() {} |
31 | |
32 | MSP430InstrInfo::MSP430InstrInfo(MSP430Subtarget &STI) |
33 | : MSP430GenInstrInfo(MSP430::ADJCALLSTACKDOWN, MSP430::ADJCALLSTACKUP), |
34 | RI() {} |
35 | |
36 | void MSP430InstrInfo::storeRegToStackSlot( |
37 | MachineBasicBlock &MBB, MachineBasicBlock::iterator MI, Register SrcReg, |
38 | bool isKill, int FrameIdx, const TargetRegisterClass *RC, |
39 | const TargetRegisterInfo *TRI, Register VReg) const { |
40 | DebugLoc DL; |
41 | if (MI != MBB.end()) DL = MI->getDebugLoc(); |
42 | MachineFunction &MF = *MBB.getParent(); |
43 | MachineFrameInfo &MFI = MF.getFrameInfo(); |
44 | |
45 | MachineMemOperand *MMO = MF.getMachineMemOperand( |
46 | PtrInfo: MachinePointerInfo::getFixedStack(MF, FI: FrameIdx), |
47 | F: MachineMemOperand::MOStore, Size: MFI.getObjectSize(ObjectIdx: FrameIdx), |
48 | BaseAlignment: MFI.getObjectAlign(ObjectIdx: FrameIdx)); |
49 | |
50 | if (RC == &MSP430::GR16RegClass) |
51 | BuildMI(BB&: MBB, I: MI, MIMD: DL, MCID: get(Opcode: MSP430::MOV16mr)) |
52 | .addFrameIndex(Idx: FrameIdx).addImm(Val: 0) |
53 | .addReg(RegNo: SrcReg, flags: getKillRegState(B: isKill)).addMemOperand(MMO); |
54 | else if (RC == &MSP430::GR8RegClass) |
55 | BuildMI(BB&: MBB, I: MI, MIMD: DL, MCID: get(Opcode: MSP430::MOV8mr)) |
56 | .addFrameIndex(Idx: FrameIdx).addImm(Val: 0) |
57 | .addReg(RegNo: SrcReg, flags: getKillRegState(B: isKill)).addMemOperand(MMO); |
58 | else |
59 | llvm_unreachable("Cannot store this register to stack slot!" ); |
60 | } |
61 | |
62 | void MSP430InstrInfo::loadRegFromStackSlot(MachineBasicBlock &MBB, |
63 | MachineBasicBlock::iterator MI, |
64 | Register DestReg, int FrameIdx, |
65 | const TargetRegisterClass *RC, |
66 | const TargetRegisterInfo *TRI, |
67 | Register VReg) const { |
68 | DebugLoc DL; |
69 | if (MI != MBB.end()) DL = MI->getDebugLoc(); |
70 | MachineFunction &MF = *MBB.getParent(); |
71 | MachineFrameInfo &MFI = MF.getFrameInfo(); |
72 | |
73 | MachineMemOperand *MMO = MF.getMachineMemOperand( |
74 | PtrInfo: MachinePointerInfo::getFixedStack(MF, FI: FrameIdx), |
75 | F: MachineMemOperand::MOLoad, Size: MFI.getObjectSize(ObjectIdx: FrameIdx), |
76 | BaseAlignment: MFI.getObjectAlign(ObjectIdx: FrameIdx)); |
77 | |
78 | if (RC == &MSP430::GR16RegClass) |
79 | BuildMI(BB&: MBB, I: MI, MIMD: DL, MCID: get(Opcode: MSP430::MOV16rm)) |
80 | .addReg(RegNo: DestReg, flags: getDefRegState(B: true)).addFrameIndex(Idx: FrameIdx) |
81 | .addImm(Val: 0).addMemOperand(MMO); |
82 | else if (RC == &MSP430::GR8RegClass) |
83 | BuildMI(BB&: MBB, I: MI, MIMD: DL, MCID: get(Opcode: MSP430::MOV8rm)) |
84 | .addReg(RegNo: DestReg, flags: getDefRegState(B: true)).addFrameIndex(Idx: FrameIdx) |
85 | .addImm(Val: 0).addMemOperand(MMO); |
86 | else |
87 | llvm_unreachable("Cannot store this register to stack slot!" ); |
88 | } |
89 | |
90 | void MSP430InstrInfo::copyPhysReg(MachineBasicBlock &MBB, |
91 | MachineBasicBlock::iterator I, |
92 | const DebugLoc &DL, MCRegister DestReg, |
93 | MCRegister SrcReg, bool KillSrc) const { |
94 | unsigned Opc; |
95 | if (MSP430::GR16RegClass.contains(Reg1: DestReg, Reg2: SrcReg)) |
96 | Opc = MSP430::MOV16rr; |
97 | else if (MSP430::GR8RegClass.contains(Reg1: DestReg, Reg2: SrcReg)) |
98 | Opc = MSP430::MOV8rr; |
99 | else |
100 | llvm_unreachable("Impossible reg-to-reg copy" ); |
101 | |
102 | BuildMI(BB&: MBB, I, MIMD: DL, MCID: get(Opcode: Opc), DestReg) |
103 | .addReg(RegNo: SrcReg, flags: getKillRegState(B: KillSrc)); |
104 | } |
105 | |
106 | unsigned MSP430InstrInfo::removeBranch(MachineBasicBlock &MBB, |
107 | int *BytesRemoved) const { |
108 | assert(!BytesRemoved && "code size not handled" ); |
109 | |
110 | MachineBasicBlock::iterator I = MBB.end(); |
111 | unsigned Count = 0; |
112 | |
113 | while (I != MBB.begin()) { |
114 | --I; |
115 | if (I->isDebugInstr()) |
116 | continue; |
117 | if (I->getOpcode() != MSP430::JMP && |
118 | I->getOpcode() != MSP430::JCC && |
119 | I->getOpcode() != MSP430::Bi && |
120 | I->getOpcode() != MSP430::Br && |
121 | I->getOpcode() != MSP430::Bm) |
122 | break; |
123 | // Remove the branch. |
124 | I->eraseFromParent(); |
125 | I = MBB.end(); |
126 | ++Count; |
127 | } |
128 | |
129 | return Count; |
130 | } |
131 | |
132 | bool MSP430InstrInfo:: |
133 | reverseBranchCondition(SmallVectorImpl<MachineOperand> &Cond) const { |
134 | assert(Cond.size() == 1 && "Invalid Xbranch condition!" ); |
135 | |
136 | MSP430CC::CondCodes CC = static_cast<MSP430CC::CondCodes>(Cond[0].getImm()); |
137 | |
138 | switch (CC) { |
139 | default: llvm_unreachable("Invalid branch condition!" ); |
140 | case MSP430CC::COND_E: |
141 | CC = MSP430CC::COND_NE; |
142 | break; |
143 | case MSP430CC::COND_NE: |
144 | CC = MSP430CC::COND_E; |
145 | break; |
146 | case MSP430CC::COND_L: |
147 | CC = MSP430CC::COND_GE; |
148 | break; |
149 | case MSP430CC::COND_GE: |
150 | CC = MSP430CC::COND_L; |
151 | break; |
152 | case MSP430CC::COND_HS: |
153 | CC = MSP430CC::COND_LO; |
154 | break; |
155 | case MSP430CC::COND_LO: |
156 | CC = MSP430CC::COND_HS; |
157 | break; |
158 | } |
159 | |
160 | Cond[0].setImm(CC); |
161 | return false; |
162 | } |
163 | |
164 | bool MSP430InstrInfo::analyzeBranch(MachineBasicBlock &MBB, |
165 | MachineBasicBlock *&TBB, |
166 | MachineBasicBlock *&FBB, |
167 | SmallVectorImpl<MachineOperand> &Cond, |
168 | bool AllowModify) const { |
169 | // Start from the bottom of the block and work up, examining the |
170 | // terminator instructions. |
171 | MachineBasicBlock::iterator I = MBB.end(); |
172 | while (I != MBB.begin()) { |
173 | --I; |
174 | if (I->isDebugInstr()) |
175 | continue; |
176 | |
177 | // Working from the bottom, when we see a non-terminator |
178 | // instruction, we're done. |
179 | if (!isUnpredicatedTerminator(MI: *I)) |
180 | break; |
181 | |
182 | // A terminator that isn't a branch can't easily be handled |
183 | // by this analysis. |
184 | if (!I->isBranch()) |
185 | return true; |
186 | |
187 | // Cannot handle indirect branches. |
188 | if (I->getOpcode() == MSP430::Br || |
189 | I->getOpcode() == MSP430::Bm) |
190 | return true; |
191 | |
192 | // Handle unconditional branches. |
193 | if (I->getOpcode() == MSP430::JMP || I->getOpcode() == MSP430::Bi) { |
194 | if (!AllowModify) { |
195 | TBB = I->getOperand(i: 0).getMBB(); |
196 | continue; |
197 | } |
198 | |
199 | // If the block has any instructions after a JMP, delete them. |
200 | MBB.erase(I: std::next(x: I), E: MBB.end()); |
201 | Cond.clear(); |
202 | FBB = nullptr; |
203 | |
204 | // Delete the JMP if it's equivalent to a fall-through. |
205 | if (MBB.isLayoutSuccessor(MBB: I->getOperand(i: 0).getMBB())) { |
206 | TBB = nullptr; |
207 | I->eraseFromParent(); |
208 | I = MBB.end(); |
209 | continue; |
210 | } |
211 | |
212 | // TBB is used to indicate the unconditinal destination. |
213 | TBB = I->getOperand(i: 0).getMBB(); |
214 | continue; |
215 | } |
216 | |
217 | // Handle conditional branches. |
218 | assert(I->getOpcode() == MSP430::JCC && "Invalid conditional branch" ); |
219 | MSP430CC::CondCodes BranchCode = |
220 | static_cast<MSP430CC::CondCodes>(I->getOperand(i: 1).getImm()); |
221 | if (BranchCode == MSP430CC::COND_INVALID) |
222 | return true; // Can't handle weird stuff. |
223 | |
224 | // Working from the bottom, handle the first conditional branch. |
225 | if (Cond.empty()) { |
226 | FBB = TBB; |
227 | TBB = I->getOperand(i: 0).getMBB(); |
228 | Cond.push_back(Elt: MachineOperand::CreateImm(Val: BranchCode)); |
229 | continue; |
230 | } |
231 | |
232 | // Handle subsequent conditional branches. Only handle the case where all |
233 | // conditional branches branch to the same destination. |
234 | assert(Cond.size() == 1); |
235 | assert(TBB); |
236 | |
237 | // Only handle the case where all conditional branches branch to |
238 | // the same destination. |
239 | if (TBB != I->getOperand(i: 0).getMBB()) |
240 | return true; |
241 | |
242 | MSP430CC::CondCodes OldBranchCode = (MSP430CC::CondCodes)Cond[0].getImm(); |
243 | // If the conditions are the same, we can leave them alone. |
244 | if (OldBranchCode == BranchCode) |
245 | continue; |
246 | |
247 | return true; |
248 | } |
249 | |
250 | return false; |
251 | } |
252 | |
253 | unsigned MSP430InstrInfo::insertBranch(MachineBasicBlock &MBB, |
254 | MachineBasicBlock *TBB, |
255 | MachineBasicBlock *FBB, |
256 | ArrayRef<MachineOperand> Cond, |
257 | const DebugLoc &DL, |
258 | int *BytesAdded) const { |
259 | // Shouldn't be a fall through. |
260 | assert(TBB && "insertBranch must not be told to insert a fallthrough" ); |
261 | assert((Cond.size() == 1 || Cond.size() == 0) && |
262 | "MSP430 branch conditions have one component!" ); |
263 | assert(!BytesAdded && "code size not handled" ); |
264 | |
265 | if (Cond.empty()) { |
266 | // Unconditional branch? |
267 | assert(!FBB && "Unconditional branch with multiple successors!" ); |
268 | BuildMI(BB: &MBB, MIMD: DL, MCID: get(Opcode: MSP430::JMP)).addMBB(MBB: TBB); |
269 | return 1; |
270 | } |
271 | |
272 | // Conditional branch. |
273 | unsigned Count = 0; |
274 | BuildMI(BB: &MBB, MIMD: DL, MCID: get(Opcode: MSP430::JCC)).addMBB(MBB: TBB).addImm(Val: Cond[0].getImm()); |
275 | ++Count; |
276 | |
277 | if (FBB) { |
278 | // Two-way Conditional branch. Insert the second branch. |
279 | BuildMI(BB: &MBB, MIMD: DL, MCID: get(Opcode: MSP430::JMP)).addMBB(MBB: FBB); |
280 | ++Count; |
281 | } |
282 | return Count; |
283 | } |
284 | |
285 | /// GetInstSize - Return the number of bytes of code the specified |
286 | /// instruction may be. This returns the maximum number of bytes. |
287 | /// |
288 | unsigned MSP430InstrInfo::getInstSizeInBytes(const MachineInstr &MI) const { |
289 | const MCInstrDesc &Desc = MI.getDesc(); |
290 | |
291 | switch (Desc.getOpcode()) { |
292 | case TargetOpcode::CFI_INSTRUCTION: |
293 | case TargetOpcode::EH_LABEL: |
294 | case TargetOpcode::IMPLICIT_DEF: |
295 | case TargetOpcode::KILL: |
296 | case TargetOpcode::DBG_VALUE: |
297 | return 0; |
298 | case TargetOpcode::INLINEASM: |
299 | case TargetOpcode::INLINEASM_BR: { |
300 | const MachineFunction *MF = MI.getParent()->getParent(); |
301 | const TargetInstrInfo &TII = *MF->getSubtarget().getInstrInfo(); |
302 | return TII.getInlineAsmLength(Str: MI.getOperand(i: 0).getSymbolName(), |
303 | MAI: *MF->getTarget().getMCAsmInfo()); |
304 | } |
305 | } |
306 | |
307 | return Desc.getSize(); |
308 | } |
309 | |