LLVM 22.0.0git
M68kExpandPseudo.cpp
Go to the documentation of this file.
1//===-- M68kExpandPseudo.cpp - Expand pseudo instructions -------*- C++ -*-===//
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/// \file
10/// This file contains a pass that expands pseudo instructions into target
11/// instructions to allow proper scheduling, if-conversion, other late
12/// optimizations, or simply the encoding of the instructions.
13///
14//===----------------------------------------------------------------------===//
15
16#include "M68k.h"
17#include "M68kFrameLowering.h"
18#include "M68kInstrInfo.h"
19#include "M68kMachineFunction.h"
20#include "M68kSubtarget.h"
21
25#include "llvm/CodeGen/Passes.h" // For IDs of passes that are preserved.
27#include "llvm/IR/GlobalValue.h"
28
29using namespace llvm;
30
31#define DEBUG_TYPE "m68k-expand-pseudo"
32#define PASS_NAME "M68k pseudo instruction expansion pass"
33
34namespace {
35class M68kExpandPseudo : public MachineFunctionPass {
36public:
37 static char ID;
38 M68kExpandPseudo() : MachineFunctionPass(ID) {}
39
40 void getAnalysisUsage(AnalysisUsage &AU) const override {
41 AU.setPreservesCFG();
45 }
46
47 const M68kSubtarget *STI;
48 const M68kInstrInfo *TII;
49 const M68kRegisterInfo *TRI;
50 const M68kMachineFunctionInfo *MFI;
51 const M68kFrameLowering *FL;
52
53 bool runOnMachineFunction(MachineFunction &Fn) override;
54
55 MachineFunctionProperties getRequiredProperties() const override {
56 return MachineFunctionProperties().setNoVRegs();
57 }
58
59private:
61 bool ExpandMBB(MachineBasicBlock &MBB);
62};
63char M68kExpandPseudo::ID = 0;
64} // End anonymous namespace.
65
66INITIALIZE_PASS(M68kExpandPseudo, DEBUG_TYPE, PASS_NAME, false, false)
67
68/// If \p MBBI is a pseudo instruction, this method expands
69/// it to the corresponding (sequence of) actual instruction(s).
70/// \returns true if \p MBBI has been expanded.
71bool M68kExpandPseudo::ExpandMI(MachineBasicBlock &MBB,
74 MachineInstrBuilder MIB(*MI.getParent()->getParent(), MI);
75 unsigned Opcode = MI.getOpcode();
76 DebugLoc DL = MBBI->getDebugLoc();
77 /// TODO infer argument size to create less switch cases
78 switch (Opcode) {
79 default:
80 return false;
81
82 case M68k::MOVI8di:
83 return TII->ExpandMOVI(MIB, MVT::i8);
84 case M68k::MOVI16ri:
85 return TII->ExpandMOVI(MIB, MVT::i16);
86 case M68k::MOVI32ri:
87 return TII->ExpandMOVI(MIB, MVT::i32);
88
89 case M68k::MOVXd16d8:
90 return TII->ExpandMOVX_RR(MIB, MVT::i16, MVT::i8);
91 case M68k::MOVXd32d8:
92 return TII->ExpandMOVX_RR(MIB, MVT::i32, MVT::i8);
93 case M68k::MOVXd32d16:
94 return TII->ExpandMOVX_RR(MIB, MVT::i32, MVT::i16);
95
96 case M68k::MOVSXd16d8:
97 return TII->ExpandMOVSZX_RR(MIB, true, MVT::i16, MVT::i8);
98 case M68k::MOVSXd32d8:
99 return TII->ExpandMOVSZX_RR(MIB, true, MVT::i32, MVT::i8);
100 case M68k::MOVSXd32d16:
101 return TII->ExpandMOVSZX_RR(MIB, true, MVT::i32, MVT::i16);
102
103 case M68k::MOVZXd16d8:
104 return TII->ExpandMOVSZX_RR(MIB, false, MVT::i16, MVT::i8);
105 case M68k::MOVZXd32d8:
106 return TII->ExpandMOVSZX_RR(MIB, false, MVT::i32, MVT::i8);
107 case M68k::MOVZXd32d16:
108 return TII->ExpandMOVSZX_RR(MIB, false, MVT::i32, MVT::i16);
109
110 case M68k::MOVSXd16j8:
111 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV8dj), MVT::i16,
112 MVT::i8);
113 case M68k::MOVSXd32j8:
114 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV8dj), MVT::i32,
115 MVT::i8);
116 case M68k::MOVSXd32j16:
117 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV16rj), MVT::i32,
118 MVT::i16);
119
120 case M68k::MOVZXd16j8:
121 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV8dj), MVT::i16,
122 MVT::i8);
123 case M68k::MOVZXd32j8:
124 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV8dj), MVT::i32,
125 MVT::i8);
126 case M68k::MOVZXd32j16:
127 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV16rj), MVT::i32,
128 MVT::i16);
129
130 case M68k::MOVSXd16p8:
131 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV8dp), MVT::i16,
132 MVT::i8);
133 case M68k::MOVSXd32p8:
134 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV8dp), MVT::i32,
135 MVT::i8);
136 case M68k::MOVSXd32p16:
137 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV16rp), MVT::i32,
138 MVT::i16);
139
140 case M68k::MOVZXd16p8:
141 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV8dp), MVT::i16,
142 MVT::i8);
143 case M68k::MOVZXd32p8:
144 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV8dp), MVT::i32,
145 MVT::i8);
146 case M68k::MOVZXd32p16:
147 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV16rp), MVT::i32,
148 MVT::i16);
149
150 case M68k::MOVSXd16f8:
151 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV8df), MVT::i16,
152 MVT::i8);
153 case M68k::MOVSXd32f8:
154 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV8df), MVT::i32,
155 MVT::i8);
156 case M68k::MOVSXd32f16:
157 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV16rf), MVT::i32,
158 MVT::i16);
159
160 case M68k::MOVZXd16f8:
161 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV8df), MVT::i16,
162 MVT::i8);
163 case M68k::MOVZXd32f8:
164 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV8df), MVT::i32,
165 MVT::i8);
166 case M68k::MOVZXd32f16:
167 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV16rf), MVT::i32,
168 MVT::i16);
169
170 case M68k::MOVSXd16q8:
171 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV8dq), MVT::i16,
172 MVT::i8);
173 case M68k::MOVSXd32q8:
174 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV8dq), MVT::i32,
175 MVT::i8);
176 case M68k::MOVSXd32q16:
177 return TII->ExpandMOVSZX_RM(MIB, true, TII->get(M68k::MOV16dq), MVT::i32,
178 MVT::i16);
179
180 case M68k::MOVZXd16q8:
181 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV8dq), MVT::i16,
182 MVT::i8);
183 case M68k::MOVZXd32q8:
184 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV8dq), MVT::i32,
185 MVT::i8);
186 case M68k::MOVZXd32q16:
187 return TII->ExpandMOVSZX_RM(MIB, false, TII->get(M68k::MOV16dq), MVT::i32,
188 MVT::i16);
189
190 case M68k::MOVM16jm_P:
191 return TII->ExpandMOVEM(MIB, TII->get(M68k::MOVM16jm), /*IsRM=*/false);
192 case M68k::MOVM32jm_P:
193 return TII->ExpandMOVEM(MIB, TII->get(M68k::MOVM32jm), /*IsRM=*/false);
194
195 case M68k::MOVM16pm_P:
196 return TII->ExpandMOVEM(MIB, TII->get(M68k::MOVM16pm), /*IsRM=*/false);
197 case M68k::MOVM32pm_P:
198 return TII->ExpandMOVEM(MIB, TII->get(M68k::MOVM32pm), /*IsRM=*/false);
199
200 case M68k::MOVM16mj_P:
201 return TII->ExpandMOVEM(MIB, TII->get(M68k::MOVM16mj), /*IsRM=*/true);
202 case M68k::MOVM32mj_P:
203 return TII->ExpandMOVEM(MIB, TII->get(M68k::MOVM32mj), /*IsRM=*/true);
204
205 case M68k::MOVM16mp_P:
206 return TII->ExpandMOVEM(MIB, TII->get(M68k::MOVM16mp), /*IsRM=*/true);
207 case M68k::MOVM32mp_P:
208 return TII->ExpandMOVEM(MIB, TII->get(M68k::MOVM32mp), /*IsRM=*/true);
209
210 case M68k::TCRETURNq:
211 case M68k::TCRETURNj: {
212 MachineOperand &JumpTarget = MI.getOperand(0);
213 MachineOperand &StackAdjust = MI.getOperand(1);
214 assert(StackAdjust.isImm() && "Expecting immediate value.");
215
216 // Adjust stack pointer.
217 int StackAdj = StackAdjust.getImm();
218 int MaxTCDelta = MFI->getTCReturnAddrDelta();
219 int Offset = 0;
220 assert(MaxTCDelta <= 0 && "MaxTCDelta should never be positive");
221
222 // Incoporate the retaddr area.
223 Offset = StackAdj - MaxTCDelta;
224 assert(Offset >= 0 && "Offset should never be negative");
225
226 if (Offset) {
227 // Check for possible merge with preceding ADD instruction.
228 Offset += FL->mergeSPUpdates(MBB, MBBI, true);
229 FL->emitSPUpdate(MBB, MBBI, Offset, /*InEpilogue=*/true);
230 }
231
232 // Jump to label or value in register.
233 if (Opcode == M68k::TCRETURNq) {
235 BuildMI(MBB, MBBI, DL, TII->get(M68k::TAILJMPq));
236 if (JumpTarget.isGlobal()) {
237 MIB.addGlobalAddress(JumpTarget.getGlobal(), JumpTarget.getOffset(),
238 JumpTarget.getTargetFlags());
239 } else {
240 assert(JumpTarget.isSymbol());
241 MIB.addExternalSymbol(JumpTarget.getSymbolName(),
242 JumpTarget.getTargetFlags());
243 }
244 } else {
245 BuildMI(MBB, MBBI, DL, TII->get(M68k::TAILJMPj))
246 .addReg(JumpTarget.getReg(), RegState::Kill);
247 }
248
249 MachineInstr &NewMI = *std::prev(MBBI);
250 NewMI.copyImplicitOps(*MBBI->getParent()->getParent(), *MBBI);
251
252 // Delete the pseudo instruction TCRETURN.
253 MBB.erase(MBBI);
254
255 return true;
256 }
257 case M68k::RET: {
258 if (MBB.getParent()->getFunction().getCallingConv() ==
260 BuildMI(MBB, MBBI, DL, TII->get(M68k::RTE));
261 } else if (int64_t StackAdj = MBBI->getOperand(0).getImm(); StackAdj == 0) {
262 BuildMI(MBB, MBBI, DL, TII->get(M68k::RTS));
263 } else {
264 // Copy return address from stack to a free address(A0 or A1) register
265 // TODO check if pseudo expand uses free address register
266 BuildMI(MBB, MBBI, DL, TII->get(M68k::MOV32aj), M68k::A1)
267 .addReg(M68k::SP);
268
269 // Adjust SP
270 FL->emitSPUpdate(MBB, MBBI, StackAdj, /*InEpilogue=*/true);
271
272 // Put the return address on stack
273 BuildMI(MBB, MBBI, DL, TII->get(M68k::MOV32ja))
274 .addReg(M68k::SP)
275 .addReg(M68k::A1);
276
277 // RTS
278 BuildMI(MBB, MBBI, DL, TII->get(M68k::RTS));
279 }
280
281 // FIXME: Can rest of the operands be ignored, if there is any?
282 MBB.erase(MBBI);
283 return true;
284 }
285 }
286 llvm_unreachable("Previous switch has a fallthrough?");
287}
288
289/// Expand all pseudo instructions contained in \p MBB.
290/// \returns true if any expansion occurred for \p MBB.
291bool M68kExpandPseudo::ExpandMBB(MachineBasicBlock &MBB) {
292 bool Modified = false;
293
294 // MBBI may be invalidated by the expansion.
296 while (MBBI != E) {
297 MachineBasicBlock::iterator NMBBI = std::next(MBBI);
298 Modified |= ExpandMI(MBB, MBBI);
299 MBBI = NMBBI;
300 }
301
302 return Modified;
303}
304
305bool M68kExpandPseudo::runOnMachineFunction(MachineFunction &MF) {
306 STI = &MF.getSubtarget<M68kSubtarget>();
307 TII = STI->getInstrInfo();
308 TRI = STI->getRegisterInfo();
309 MFI = MF.getInfo<M68kMachineFunctionInfo>();
310 FL = STI->getFrameLowering();
311
312 bool Modified = false;
313 for (MachineBasicBlock &MBB : MF)
314 Modified |= ExpandMBB(MBB);
315 return Modified;
316}
317
318/// Returns an instance of the pseudo instruction expansion pass.
320 return new M68kExpandPseudo();
321}
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
MachineBasicBlock & MBB
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
MachineBasicBlock MachineBasicBlock::iterator MBBI
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
#define DEBUG_TYPE
const HexagonInstrInfo * TII
IRTranslator LLVM IR MI
This file contains the M68k declaration of TargetFrameLowering class.
This file contains the M68k implementation of the TargetInstrInfo class.
This file declares the M68k specific subclass of MachineFunctionInfo.
This file declares the M68k specific subclass of TargetSubtargetInfo.
This file contains the entry points for global functions defined in the M68k target library,...
Register const TargetRegisterInfo * TRI
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
Definition PassSupport.h:56
#define PASS_NAME
Represent the analysis usage information of a pass.
AnalysisUsage & addPreservedID(const void *ID)
LLVM_ABI void setPreservesCFG()
This function should be called by the pass, iff they do not:
Definition Pass.cpp:270
A debug info location.
Definition DebugLoc.h:123
FunctionPass class - This class is used to implement most global optimizations.
Definition Pass.h:314
const M68kInstrInfo * getInstrInfo() const override
const M68kRegisterInfo * getRegisterInfo() const override
const M68kFrameLowering * getFrameLowering() const override
MachineInstrBundleIterator< MachineInstr > iterator
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - Subclasses that override getAnalysisUsage must call this.
Properties which a MachineFunction may have at a given point in time.
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
Ty * getInfo()
getInfo - Keep track of various per-function pieces of information for backends that would like to do...
const MachineInstrBuilder & addExternalSymbol(const char *FnName, unsigned TargetFlags=0) const
const MachineInstrBuilder & addGlobalAddress(const GlobalValue *GV, int64_t Offset=0, unsigned TargetFlags=0) const
const MachineInstrBuilder & addReg(Register RegNo, unsigned flags=0, unsigned SubReg=0) const
Add a new virtual register operand.
Representation of each machine instruction.
LLVM_ABI void copyImplicitOps(MachineFunction &MF, const MachineInstr &MI)
Copy implicit register operands from specified instruction to this instruction.
MachineOperand class - Representation of each machine instruction operand.
const GlobalValue * getGlobal() const
int64_t getImm() const
bool isImm() const
isImm - Tests if this is a MO_Immediate operand.
bool isSymbol() const
isSymbol - Tests if this is a MO_ExternalSymbol operand.
unsigned getTargetFlags() const
bool isGlobal() const
isGlobal - Tests if this is a MO_GlobalAddress operand.
const char * getSymbolName() const
Register getReg() const
getReg - Returns the register number.
int64_t getOffset() const
Return the offset from the symbol in this operand.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition CallingConv.h:24
@ M68k_INTR
Used for M68k interrupt routines.
@ Kill
The last use of a register.
This is an optimization pass for GlobalISel generic memory operations.
@ Offset
Definition DWP.cpp:532
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
LLVM_ABI char & MachineDominatorsID
MachineDominators - This pass is a machine dominators analysis pass.
LLVM_ABI char & MachineLoopInfoID
MachineLoopInfo - This pass is a loop analysis pass.
FunctionPass * createM68kExpandPseudoPass()
Return a Machine IR pass that expands M68k-specific pseudo instructions into a sequence of actual ins...