blob: 49ad47f7facc1ce436c942001ce42c52d50f0751 [file] [log] [blame]
Quentin Colombetba2a0162016-02-16 19:26:02 +00001//===-- llvm/lib/Target/AArch64/AArch64CallLowering.cpp - Call lowering ---===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9///
10/// \file
11/// This file implements the lowering of LLVM calls to machine code calls for
12/// GlobalISel.
13///
14//===----------------------------------------------------------------------===//
15
16#include "AArch64CallLowering.h"
17#include "AArch64ISelLowering.h"
18
19#include "llvm/CodeGen/GlobalISel/MachineIRBuilder.h"
Tim Northoverb18ea162016-09-20 15:20:36 +000020#include "llvm/CodeGen/Analysis.h"
Quentin Colombetba2a0162016-02-16 19:26:02 +000021#include "llvm/CodeGen/MachineInstrBuilder.h"
Tim Northoverb18ea162016-09-20 15:20:36 +000022#include "llvm/CodeGen/MachineRegisterInfo.h"
Tim Northover406024a2016-08-10 21:44:01 +000023#include "llvm/Target/TargetRegisterInfo.h"
24#include "llvm/Target/TargetSubtargetInfo.h"
Quentin Colombetba2a0162016-02-16 19:26:02 +000025using namespace llvm;
26
Quentin Colombet789ad562016-04-07 20:47:51 +000027#ifndef LLVM_BUILD_GLOBAL_ISEL
Quentin Colombet6cc73ce2016-04-07 20:49:15 +000028#error "This shouldn't be built without GISel"
Quentin Colombet789ad562016-04-07 20:47:51 +000029#endif
30
Quentin Colombetba2a0162016-02-16 19:26:02 +000031AArch64CallLowering::AArch64CallLowering(const AArch64TargetLowering &TLI)
32 : CallLowering(&TLI) {
33}
34
Diana Picusf11f0422016-12-05 10:40:33 +000035unsigned CallLowering::ValueHandler::extendRegister(unsigned ValReg,
36 CCValAssign &VA) {
Tim Northovera5e38fa2016-09-22 13:49:25 +000037 LLT LocTy{VA.getLocVT()};
38 switch (VA.getLocInfo()) {
39 default: break;
40 case CCValAssign::Full:
41 case CCValAssign::BCvt:
42 // FIXME: bitconverting between vector types may or may not be a
43 // nop in big-endian situations.
44 return ValReg;
45 case CCValAssign::AExt:
46 assert(!VA.getLocVT().isVector() && "unexpected vector extend");
47 // Otherwise, it's a nop.
48 return ValReg;
49 case CCValAssign::SExt: {
50 unsigned NewReg = MRI.createGenericVirtualRegister(LocTy);
51 MIRBuilder.buildSExt(NewReg, ValReg);
52 return NewReg;
53 }
54 case CCValAssign::ZExt: {
55 unsigned NewReg = MRI.createGenericVirtualRegister(LocTy);
56 MIRBuilder.buildZExt(NewReg, ValReg);
57 return NewReg;
58 }
59 }
60 llvm_unreachable("unable to extend register");
61}
62
Diana Picusf11f0422016-12-05 10:40:33 +000063struct IncomingArgHandler : public CallLowering::ValueHandler {
Tim Northovera5e38fa2016-09-22 13:49:25 +000064 IncomingArgHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI)
65 : ValueHandler(MIRBuilder, MRI) {}
66
67 unsigned getStackAddress(uint64_t Size, int64_t Offset,
68 MachinePointerInfo &MPO) override {
69 auto &MFI = MIRBuilder.getMF().getFrameInfo();
70 int FI = MFI.CreateFixedObject(Size, Offset, true);
71 MPO = MachinePointerInfo::getFixedStack(MIRBuilder.getMF(), FI);
72 unsigned AddrReg = MRI.createGenericVirtualRegister(LLT::pointer(0, 64));
73 MIRBuilder.buildFrameIndex(AddrReg, FI);
74 return AddrReg;
75 }
76
77 void assignValueToReg(unsigned ValVReg, unsigned PhysReg,
78 CCValAssign &VA) override {
79 markPhysRegUsed(PhysReg);
80 MIRBuilder.buildCopy(ValVReg, PhysReg);
81 // FIXME: assert extension
82 }
83
84 void assignValueToAddress(unsigned ValVReg, unsigned Addr, uint64_t Size,
85 MachinePointerInfo &MPO, CCValAssign &VA) override {
86 auto MMO = MIRBuilder.getMF().getMachineMemOperand(
87 MPO, MachineMemOperand::MOLoad | MachineMemOperand::MOInvariant, Size,
88 0);
89 MIRBuilder.buildLoad(ValVReg, Addr, *MMO);
90 }
91
92 /// How the physical register gets marked varies between formal
93 /// parameters (it's a basic-block live-in), and a call instruction
94 /// (it's an implicit-def of the BL).
95 virtual void markPhysRegUsed(unsigned PhysReg) = 0;
96};
97
98struct FormalArgHandler : public IncomingArgHandler {
99 FormalArgHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI)
100 : IncomingArgHandler(MIRBuilder, MRI) {}
101
102 void markPhysRegUsed(unsigned PhysReg) override {
103 MIRBuilder.getMBB().addLiveIn(PhysReg);
104 }
105};
106
107struct CallReturnHandler : public IncomingArgHandler {
108 CallReturnHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI,
109 MachineInstrBuilder MIB)
110 : IncomingArgHandler(MIRBuilder, MRI), MIB(MIB) {}
111
112 void markPhysRegUsed(unsigned PhysReg) override {
113 MIB.addDef(PhysReg, RegState::Implicit);
114 }
115
116 MachineInstrBuilder MIB;
117};
118
Diana Picusf11f0422016-12-05 10:40:33 +0000119struct OutgoingArgHandler : public CallLowering::ValueHandler {
Tim Northovera5e38fa2016-09-22 13:49:25 +0000120 OutgoingArgHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI,
121 MachineInstrBuilder MIB)
122 : ValueHandler(MIRBuilder, MRI), MIB(MIB) {}
123
124 unsigned getStackAddress(uint64_t Size, int64_t Offset,
125 MachinePointerInfo &MPO) override {
126 LLT p0 = LLT::pointer(0, 64);
127 LLT s64 = LLT::scalar(64);
128 unsigned SPReg = MRI.createGenericVirtualRegister(p0);
129 MIRBuilder.buildCopy(SPReg, AArch64::SP);
130
131 unsigned OffsetReg = MRI.createGenericVirtualRegister(s64);
132 MIRBuilder.buildConstant(OffsetReg, Offset);
133
134 unsigned AddrReg = MRI.createGenericVirtualRegister(p0);
135 MIRBuilder.buildGEP(AddrReg, SPReg, OffsetReg);
136
137 MPO = MachinePointerInfo::getStack(MIRBuilder.getMF(), Offset);
138 return AddrReg;
139 }
140
141 void assignValueToReg(unsigned ValVReg, unsigned PhysReg,
142 CCValAssign &VA) override {
143 MIB.addUse(PhysReg, RegState::Implicit);
144 unsigned ExtReg = extendRegister(ValVReg, VA);
145 MIRBuilder.buildCopy(PhysReg, ExtReg);
146 }
147
148 void assignValueToAddress(unsigned ValVReg, unsigned Addr, uint64_t Size,
149 MachinePointerInfo &MPO, CCValAssign &VA) override {
150 auto MMO = MIRBuilder.getMF().getMachineMemOperand(
151 MPO, MachineMemOperand::MOStore, Size, 0);
152 MIRBuilder.buildStore(ValVReg, Addr, *MMO);
153 }
154
155 MachineInstrBuilder MIB;
156};
157
Tim Northover9a467182016-09-21 12:57:45 +0000158void AArch64CallLowering::splitToValueTypes(const ArgInfo &OrigArg,
159 SmallVectorImpl<ArgInfo> &SplitArgs,
160 const DataLayout &DL,
161 MachineRegisterInfo &MRI,
162 SplitArgTy PerformArgSplit) const {
Tim Northoverb18ea162016-09-20 15:20:36 +0000163 const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
Tim Northover9a467182016-09-21 12:57:45 +0000164 LLVMContext &Ctx = OrigArg.Ty->getContext();
Tim Northoverb18ea162016-09-20 15:20:36 +0000165
166 SmallVector<EVT, 4> SplitVTs;
167 SmallVector<uint64_t, 4> Offsets;
Tim Northover9a467182016-09-21 12:57:45 +0000168 ComputeValueVTs(TLI, DL, OrigArg.Ty, SplitVTs, &Offsets, 0);
Tim Northoverb18ea162016-09-20 15:20:36 +0000169
170 if (SplitVTs.size() == 1) {
Tim Northoverd1fd3832016-12-05 21:25:33 +0000171 // No splitting to do, but we want to replace the original type (e.g. [1 x
172 // double] -> double).
173 SplitArgs.emplace_back(OrigArg.Reg, SplitVTs[0].getTypeForEVT(Ctx),
174 OrigArg.Flags);
Tim Northoverb18ea162016-09-20 15:20:36 +0000175 return;
176 }
177
Tim Northover9a467182016-09-21 12:57:45 +0000178 unsigned FirstRegIdx = SplitArgs.size();
Tim Northoverb18ea162016-09-20 15:20:36 +0000179 for (auto SplitVT : SplitVTs) {
Tim Northover9a467182016-09-21 12:57:45 +0000180 // FIXME: set split flags if they're actually used (e.g. i128 on AAPCS).
Tim Northoverb18ea162016-09-20 15:20:36 +0000181 Type *SplitTy = SplitVT.getTypeForEVT(Ctx);
Tim Northover9a467182016-09-21 12:57:45 +0000182 SplitArgs.push_back(
183 ArgInfo{MRI.createGenericVirtualRegister(LLT{*SplitTy, DL}), SplitTy,
184 OrigArg.Flags});
Tim Northoverb18ea162016-09-20 15:20:36 +0000185 }
186
187 SmallVector<uint64_t, 4> BitOffsets;
188 for (auto Offset : Offsets)
189 BitOffsets.push_back(Offset * 8);
190
Tim Northover9a467182016-09-21 12:57:45 +0000191 SmallVector<unsigned, 8> SplitRegs;
192 for (auto I = &SplitArgs[FirstRegIdx]; I != SplitArgs.end(); ++I)
193 SplitRegs.push_back(I->Reg);
194
195 PerformArgSplit(SplitRegs, BitOffsets);
Tim Northoverb18ea162016-09-20 15:20:36 +0000196}
197
198bool AArch64CallLowering::lowerReturn(MachineIRBuilder &MIRBuilder,
199 const Value *Val, unsigned VReg) const {
200 MachineFunction &MF = MIRBuilder.getMF();
201 const Function &F = *MF.getFunction();
202
Tim Northover05cc4852016-12-07 21:05:38 +0000203 auto MIB = MIRBuilder.buildInstrNoInsert(AArch64::RET_ReallyLR);
Tim Northoverb18ea162016-09-20 15:20:36 +0000204 assert(((Val && VReg) || (!Val && !VReg)) && "Return value without a vreg");
Tim Northover05cc4852016-12-07 21:05:38 +0000205 bool Success = true;
Tim Northoverb18ea162016-09-20 15:20:36 +0000206 if (VReg) {
Tim Northoverb18ea162016-09-20 15:20:36 +0000207 const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
208 CCAssignFn *AssignFn = TLI.CCAssignFnForReturn(F.getCallingConv());
209 MachineRegisterInfo &MRI = MF.getRegInfo();
210 auto &DL = F.getParent()->getDataLayout();
211
Tim Northover9a467182016-09-21 12:57:45 +0000212 ArgInfo OrigArg{VReg, Val->getType()};
213 setArgFlags(OrigArg, AttributeSet::ReturnIndex, DL, F);
214
215 SmallVector<ArgInfo, 8> SplitArgs;
216 splitToValueTypes(OrigArg, SplitArgs, DL, MRI,
Tim Northoverb18ea162016-09-20 15:20:36 +0000217 [&](ArrayRef<unsigned> Regs, ArrayRef<uint64_t> Offsets) {
218 MIRBuilder.buildExtract(Regs, Offsets, VReg);
219 });
220
Tim Northovera5e38fa2016-09-22 13:49:25 +0000221 OutgoingArgHandler Handler(MIRBuilder, MRI, MIB);
Tim Northover05cc4852016-12-07 21:05:38 +0000222 Success = handleAssignments(MIRBuilder, AssignFn, SplitArgs, Handler);
Tim Northoverb18ea162016-09-20 15:20:36 +0000223 }
Tim Northover05cc4852016-12-07 21:05:38 +0000224
225 MIRBuilder.insertInstr(MIB);
226 return Success;
Tim Northoverb18ea162016-09-20 15:20:36 +0000227}
228
Tim Northover862758ec2016-09-21 12:57:35 +0000229bool AArch64CallLowering::lowerFormalArguments(MachineIRBuilder &MIRBuilder,
230 const Function &F,
231 ArrayRef<unsigned> VRegs) const {
232 auto &Args = F.getArgumentList();
Tim Northover406024a2016-08-10 21:44:01 +0000233 MachineFunction &MF = MIRBuilder.getMF();
Tim Northoverb18ea162016-09-20 15:20:36 +0000234 MachineBasicBlock &MBB = MIRBuilder.getMBB();
235 MachineRegisterInfo &MRI = MF.getRegInfo();
Tim Northoverb18ea162016-09-20 15:20:36 +0000236 auto &DL = F.getParent()->getDataLayout();
Tim Northover406024a2016-08-10 21:44:01 +0000237
Tim Northover9a467182016-09-21 12:57:45 +0000238 SmallVector<ArgInfo, 8> SplitArgs;
Tim Northoverb18ea162016-09-20 15:20:36 +0000239 unsigned i = 0;
240 for (auto &Arg : Args) {
Tim Northover9a467182016-09-21 12:57:45 +0000241 ArgInfo OrigArg{VRegs[i], Arg.getType()};
242 setArgFlags(OrigArg, i + 1, DL, F);
243 splitToValueTypes(OrigArg, SplitArgs, DL, MRI,
Tim Northoverb18ea162016-09-20 15:20:36 +0000244 [&](ArrayRef<unsigned> Regs, ArrayRef<uint64_t> Offsets) {
245 MIRBuilder.buildSequence(VRegs[i], Regs, Offsets);
246 });
247 ++i;
248 }
249
250 if (!MBB.empty())
251 MIRBuilder.setInstr(*MBB.begin());
Tim Northover406024a2016-08-10 21:44:01 +0000252
253 const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
254 CCAssignFn *AssignFn =
255 TLI.CCAssignFnForCall(F.getCallingConv(), /*IsVarArg=*/false);
256
Tim Northovera5e38fa2016-09-22 13:49:25 +0000257 FormalArgHandler Handler(MIRBuilder, MRI);
258 if (!handleAssignments(MIRBuilder, AssignFn, SplitArgs, Handler))
Tim Northover9a467182016-09-21 12:57:45 +0000259 return false;
Tim Northoverb18ea162016-09-20 15:20:36 +0000260
261 // Move back to the end of the basic block.
262 MIRBuilder.setMBB(MBB);
263
Tim Northover9a467182016-09-21 12:57:45 +0000264 return true;
Tim Northover406024a2016-08-10 21:44:01 +0000265}
266
267bool AArch64CallLowering::lowerCall(MachineIRBuilder &MIRBuilder,
Tim Northover9a467182016-09-21 12:57:45 +0000268 const MachineOperand &Callee,
269 const ArgInfo &OrigRet,
270 ArrayRef<ArgInfo> OrigArgs) const {
Tim Northover406024a2016-08-10 21:44:01 +0000271 MachineFunction &MF = MIRBuilder.getMF();
272 const Function &F = *MF.getFunction();
Tim Northoverb18ea162016-09-20 15:20:36 +0000273 MachineRegisterInfo &MRI = MF.getRegInfo();
274 auto &DL = F.getParent()->getDataLayout();
275
Tim Northover9a467182016-09-21 12:57:45 +0000276 SmallVector<ArgInfo, 8> SplitArgs;
277 for (auto &OrigArg : OrigArgs) {
278 splitToValueTypes(OrigArg, SplitArgs, DL, MRI,
Tim Northoverb18ea162016-09-20 15:20:36 +0000279 [&](ArrayRef<unsigned> Regs, ArrayRef<uint64_t> Offsets) {
Tim Northover9a467182016-09-21 12:57:45 +0000280 MIRBuilder.buildExtract(Regs, Offsets, OrigArg.Reg);
Tim Northoverb18ea162016-09-20 15:20:36 +0000281 });
282 }
Tim Northover406024a2016-08-10 21:44:01 +0000283
Tim Northover406024a2016-08-10 21:44:01 +0000284 // Find out which ABI gets to decide where things go.
285 const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
286 CCAssignFn *CallAssignFn =
287 TLI.CCAssignFnForCall(F.getCallingConv(), /*IsVarArg=*/false);
288
Tim Northovera5e38fa2016-09-22 13:49:25 +0000289 // Create a temporarily-floating call instruction so we can add the implicit
290 // uses of arg registers.
291 auto MIB = MIRBuilder.buildInstrNoInsert(Callee.isReg() ? AArch64::BLR
292 : AArch64::BL);
Tim Northoverfe5f89b2016-08-29 19:07:08 +0000293 MIB.addOperand(Callee);
Tim Northover406024a2016-08-10 21:44:01 +0000294
295 // Tell the call which registers are clobbered.
296 auto TRI = MF.getSubtarget().getRegisterInfo();
297 MIB.addRegMask(TRI->getCallPreservedMask(MF, F.getCallingConv()));
298
Tim Northovera5e38fa2016-09-22 13:49:25 +0000299 // Do the actual argument marshalling.
300 SmallVector<unsigned, 8> PhysRegs;
301 OutgoingArgHandler Handler(MIRBuilder, MRI, MIB);
302 if (!handleAssignments(MIRBuilder, CallAssignFn, SplitArgs, Handler))
303 return false;
304
305 // Now we can add the actual call instruction to the correct basic block.
306 MIRBuilder.insertInstr(MIB);
Tim Northover406024a2016-08-10 21:44:01 +0000307
308 // Finally we can copy the returned value back into its virtual-register. In
309 // symmetry with the arugments, the physical register must be an
310 // implicit-define of the call instruction.
311 CCAssignFn *RetAssignFn = TLI.CCAssignFnForReturn(F.getCallingConv());
Tim Northover9a467182016-09-21 12:57:45 +0000312 if (OrigRet.Reg) {
313 SplitArgs.clear();
Tim Northoverb18ea162016-09-20 15:20:36 +0000314
315 SmallVector<uint64_t, 8> RegOffsets;
Tim Northover9a467182016-09-21 12:57:45 +0000316 SmallVector<unsigned, 8> SplitRegs;
317 splitToValueTypes(OrigRet, SplitArgs, DL, MRI,
Tim Northoverb18ea162016-09-20 15:20:36 +0000318 [&](ArrayRef<unsigned> Regs, ArrayRef<uint64_t> Offsets) {
319 std::copy(Offsets.begin(), Offsets.end(),
320 std::back_inserter(RegOffsets));
Tim Northover9a467182016-09-21 12:57:45 +0000321 std::copy(Regs.begin(), Regs.end(),
322 std::back_inserter(SplitRegs));
Tim Northoverb18ea162016-09-20 15:20:36 +0000323 });
324
Tim Northovera5e38fa2016-09-22 13:49:25 +0000325 CallReturnHandler Handler(MIRBuilder, MRI, MIB);
326 if (!handleAssignments(MIRBuilder, RetAssignFn, SplitArgs, Handler))
Tim Northover9a467182016-09-21 12:57:45 +0000327 return false;
Tim Northover406024a2016-08-10 21:44:01 +0000328
Tim Northoverb18ea162016-09-20 15:20:36 +0000329 if (!RegOffsets.empty())
Tim Northover9a467182016-09-21 12:57:45 +0000330 MIRBuilder.buildSequence(OrigRet.Reg, SplitRegs, RegOffsets);
Tim Northoverb18ea162016-09-20 15:20:36 +0000331 }
332
Tim Northover406024a2016-08-10 21:44:01 +0000333 return true;
334}