blob: b8644ba1432d9deab199cd018437e2362c06c07c [file] [log] [blame]
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +00001//===--- AArch64CallLowering.cpp - Call lowering --------------------------===//
Quentin Colombetba2a0162016-02-16 19:26:02 +00002//
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"
Tim Northovere9600d82017-02-08 17:57:27 +000018#include "AArch64MachineFunctionInfo.h"
19#include "AArch64Subtarget.h"
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +000020#include "llvm/ADT/ArrayRef.h"
21#include "llvm/ADT/SmallVector.h"
Tim Northoverb18ea162016-09-20 15:20:36 +000022#include "llvm/CodeGen/Analysis.h"
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +000023#include "llvm/CodeGen/CallingConvLower.h"
Quentin Colombetf38015e2016-12-22 21:56:31 +000024#include "llvm/CodeGen/GlobalISel/MachineIRBuilder.h"
Quentin Colombetf38015e2016-12-22 21:56:31 +000025#include "llvm/CodeGen/GlobalISel/Utils.h"
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +000026#include "llvm/CodeGen/LowLevelType.h"
27#include "llvm/CodeGen/MachineBasicBlock.h"
28#include "llvm/CodeGen/MachineFrameInfo.h"
29#include "llvm/CodeGen/MachineFunction.h"
Quentin Colombetba2a0162016-02-16 19:26:02 +000030#include "llvm/CodeGen/MachineInstrBuilder.h"
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +000031#include "llvm/CodeGen/MachineMemOperand.h"
32#include "llvm/CodeGen/MachineOperand.h"
Tim Northoverb18ea162016-09-20 15:20:36 +000033#include "llvm/CodeGen/MachineRegisterInfo.h"
David Blaikieb3bde2e2017-11-17 01:07:10 +000034#include "llvm/CodeGen/TargetRegisterInfo.h"
35#include "llvm/CodeGen/TargetSubtargetInfo.h"
Craig Topper2fa14362018-03-29 17:21:10 +000036#include "llvm/CodeGen/ValueTypes.h"
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +000037#include "llvm/IR/Argument.h"
38#include "llvm/IR/Attributes.h"
39#include "llvm/IR/Function.h"
40#include "llvm/IR/Type.h"
41#include "llvm/IR/Value.h"
David Blaikie13e77db2018-03-23 23:58:25 +000042#include "llvm/Support/MachineValueType.h"
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +000043#include <algorithm>
44#include <cassert>
45#include <cstdint>
46#include <iterator>
47
Quentin Colombetba2a0162016-02-16 19:26:02 +000048using namespace llvm;
49
50AArch64CallLowering::AArch64CallLowering(const AArch64TargetLowering &TLI)
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +000051 : CallLowering(&TLI) {}
Quentin Colombetba2a0162016-02-16 19:26:02 +000052
Benjamin Kramer49a49fe2017-08-20 13:03:48 +000053namespace {
Diana Picusf11f0422016-12-05 10:40:33 +000054struct IncomingArgHandler : public CallLowering::ValueHandler {
Tim Northoverd9433542017-01-17 22:30:10 +000055 IncomingArgHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI,
56 CCAssignFn *AssignFn)
Tim Northovere9600d82017-02-08 17:57:27 +000057 : ValueHandler(MIRBuilder, MRI, AssignFn), StackUsed(0) {}
Tim Northovera5e38fa2016-09-22 13:49:25 +000058
59 unsigned getStackAddress(uint64_t Size, int64_t Offset,
60 MachinePointerInfo &MPO) override {
61 auto &MFI = MIRBuilder.getMF().getFrameInfo();
62 int FI = MFI.CreateFixedObject(Size, Offset, true);
63 MPO = MachinePointerInfo::getFixedStack(MIRBuilder.getMF(), FI);
64 unsigned AddrReg = MRI.createGenericVirtualRegister(LLT::pointer(0, 64));
65 MIRBuilder.buildFrameIndex(AddrReg, FI);
Tim Northovere9600d82017-02-08 17:57:27 +000066 StackUsed = std::max(StackUsed, Size + Offset);
Tim Northovera5e38fa2016-09-22 13:49:25 +000067 return AddrReg;
68 }
69
70 void assignValueToReg(unsigned ValVReg, unsigned PhysReg,
71 CCValAssign &VA) override {
72 markPhysRegUsed(PhysReg);
Aditya Nandakumarc3bfc812017-10-09 20:07:43 +000073 switch (VA.getLocInfo()) {
74 default:
75 MIRBuilder.buildCopy(ValVReg, PhysReg);
76 break;
77 case CCValAssign::LocInfo::SExt:
78 case CCValAssign::LocInfo::ZExt:
79 case CCValAssign::LocInfo::AExt: {
80 auto Copy = MIRBuilder.buildCopy(LLT{VA.getLocVT()}, PhysReg);
81 MIRBuilder.buildTrunc(ValVReg, Copy);
82 break;
83 }
84 }
Tim Northovera5e38fa2016-09-22 13:49:25 +000085 }
86
87 void assignValueToAddress(unsigned ValVReg, unsigned Addr, uint64_t Size,
88 MachinePointerInfo &MPO, CCValAssign &VA) override {
89 auto MMO = MIRBuilder.getMF().getMachineMemOperand(
90 MPO, MachineMemOperand::MOLoad | MachineMemOperand::MOInvariant, Size,
91 0);
92 MIRBuilder.buildLoad(ValVReg, Addr, *MMO);
93 }
94
95 /// How the physical register gets marked varies between formal
96 /// parameters (it's a basic-block live-in), and a call instruction
97 /// (it's an implicit-def of the BL).
98 virtual void markPhysRegUsed(unsigned PhysReg) = 0;
Tim Northovere9600d82017-02-08 17:57:27 +000099
100 uint64_t StackUsed;
Tim Northovera5e38fa2016-09-22 13:49:25 +0000101};
102
103struct FormalArgHandler : public IncomingArgHandler {
Tim Northoverd9433542017-01-17 22:30:10 +0000104 FormalArgHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI,
105 CCAssignFn *AssignFn)
106 : IncomingArgHandler(MIRBuilder, MRI, AssignFn) {}
Tim Northovera5e38fa2016-09-22 13:49:25 +0000107
108 void markPhysRegUsed(unsigned PhysReg) override {
109 MIRBuilder.getMBB().addLiveIn(PhysReg);
110 }
111};
112
113struct CallReturnHandler : public IncomingArgHandler {
114 CallReturnHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI,
Tim Northoverd9433542017-01-17 22:30:10 +0000115 MachineInstrBuilder MIB, CCAssignFn *AssignFn)
116 : IncomingArgHandler(MIRBuilder, MRI, AssignFn), MIB(MIB) {}
Tim Northovera5e38fa2016-09-22 13:49:25 +0000117
118 void markPhysRegUsed(unsigned PhysReg) override {
119 MIB.addDef(PhysReg, RegState::Implicit);
120 }
121
122 MachineInstrBuilder MIB;
123};
124
Diana Picusf11f0422016-12-05 10:40:33 +0000125struct OutgoingArgHandler : public CallLowering::ValueHandler {
Tim Northovera5e38fa2016-09-22 13:49:25 +0000126 OutgoingArgHandler(MachineIRBuilder &MIRBuilder, MachineRegisterInfo &MRI,
Tim Northoverd9433542017-01-17 22:30:10 +0000127 MachineInstrBuilder MIB, CCAssignFn *AssignFn,
128 CCAssignFn *AssignFnVarArg)
129 : ValueHandler(MIRBuilder, MRI, AssignFn), MIB(MIB),
Tim Northover509091f2017-01-17 22:43:34 +0000130 AssignFnVarArg(AssignFnVarArg), StackSize(0) {}
Tim Northovera5e38fa2016-09-22 13:49:25 +0000131
132 unsigned getStackAddress(uint64_t Size, int64_t Offset,
133 MachinePointerInfo &MPO) override {
134 LLT p0 = LLT::pointer(0, 64);
135 LLT s64 = LLT::scalar(64);
136 unsigned SPReg = MRI.createGenericVirtualRegister(p0);
137 MIRBuilder.buildCopy(SPReg, AArch64::SP);
138
139 unsigned OffsetReg = MRI.createGenericVirtualRegister(s64);
140 MIRBuilder.buildConstant(OffsetReg, Offset);
141
142 unsigned AddrReg = MRI.createGenericVirtualRegister(p0);
143 MIRBuilder.buildGEP(AddrReg, SPReg, OffsetReg);
144
145 MPO = MachinePointerInfo::getStack(MIRBuilder.getMF(), Offset);
146 return AddrReg;
147 }
148
149 void assignValueToReg(unsigned ValVReg, unsigned PhysReg,
150 CCValAssign &VA) override {
151 MIB.addUse(PhysReg, RegState::Implicit);
152 unsigned ExtReg = extendRegister(ValVReg, VA);
153 MIRBuilder.buildCopy(PhysReg, ExtReg);
154 }
155
156 void assignValueToAddress(unsigned ValVReg, unsigned Addr, uint64_t Size,
157 MachinePointerInfo &MPO, CCValAssign &VA) override {
Amara Emerson846f2432018-07-02 16:39:09 +0000158 if (VA.getLocInfo() == CCValAssign::LocInfo::AExt)
159 Size = VA.getLocVT().getSizeInBits() / 8;
160
Tim Northovera5e38fa2016-09-22 13:49:25 +0000161 auto MMO = MIRBuilder.getMF().getMachineMemOperand(
162 MPO, MachineMemOperand::MOStore, Size, 0);
163 MIRBuilder.buildStore(ValVReg, Addr, *MMO);
164 }
165
Eugene Zelenkoc5eb8e22017-02-01 22:56:06 +0000166 bool assignArg(unsigned ValNo, MVT ValVT, MVT LocVT,
167 CCValAssign::LocInfo LocInfo,
168 const CallLowering::ArgInfo &Info,
169 CCState &State) override {
Tim Northovere80d6d12017-03-02 15:34:18 +0000170 bool Res;
Tim Northoverd9433542017-01-17 22:30:10 +0000171 if (Info.IsFixed)
Tim Northovere80d6d12017-03-02 15:34:18 +0000172 Res = AssignFn(ValNo, ValVT, LocVT, LocInfo, Info.Flags, State);
173 else
174 Res = AssignFnVarArg(ValNo, ValVT, LocVT, LocInfo, Info.Flags, State);
175
176 StackSize = State.getNextStackOffset();
177 return Res;
Tim Northoverd9433542017-01-17 22:30:10 +0000178 }
179
Tim Northovera5e38fa2016-09-22 13:49:25 +0000180 MachineInstrBuilder MIB;
Tim Northoverd9433542017-01-17 22:30:10 +0000181 CCAssignFn *AssignFnVarArg;
Tim Northover509091f2017-01-17 22:43:34 +0000182 uint64_t StackSize;
Tim Northovera5e38fa2016-09-22 13:49:25 +0000183};
Benjamin Kramer49a49fe2017-08-20 13:03:48 +0000184} // namespace
Tim Northovera5e38fa2016-09-22 13:49:25 +0000185
Benjamin Kramer061f4a52017-01-13 14:39:03 +0000186void AArch64CallLowering::splitToValueTypes(
187 const ArgInfo &OrigArg, SmallVectorImpl<ArgInfo> &SplitArgs,
Tim Northoveref1fc5a2017-08-21 21:56:11 +0000188 const DataLayout &DL, MachineRegisterInfo &MRI, CallingConv::ID CallConv,
Benjamin Kramer061f4a52017-01-13 14:39:03 +0000189 const SplitArgTy &PerformArgSplit) const {
Tim Northoverb18ea162016-09-20 15:20:36 +0000190 const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
Tim Northover9a467182016-09-21 12:57:45 +0000191 LLVMContext &Ctx = OrigArg.Ty->getContext();
Tim Northoverb18ea162016-09-20 15:20:36 +0000192
Amara Emerson0d6a26d2018-05-16 10:32:02 +0000193 if (OrigArg.Ty->isVoidTy())
194 return;
195
Tim Northoverb18ea162016-09-20 15:20:36 +0000196 SmallVector<EVT, 4> SplitVTs;
197 SmallVector<uint64_t, 4> Offsets;
Tim Northover9a467182016-09-21 12:57:45 +0000198 ComputeValueVTs(TLI, DL, OrigArg.Ty, SplitVTs, &Offsets, 0);
Tim Northoverb18ea162016-09-20 15:20:36 +0000199
200 if (SplitVTs.size() == 1) {
Tim Northoverd1fd3832016-12-05 21:25:33 +0000201 // No splitting to do, but we want to replace the original type (e.g. [1 x
202 // double] -> double).
203 SplitArgs.emplace_back(OrigArg.Reg, SplitVTs[0].getTypeForEVT(Ctx),
Tim Northoverd9433542017-01-17 22:30:10 +0000204 OrigArg.Flags, OrigArg.IsFixed);
Tim Northoverb18ea162016-09-20 15:20:36 +0000205 return;
206 }
207
Tim Northover9a467182016-09-21 12:57:45 +0000208 unsigned FirstRegIdx = SplitArgs.size();
Tim Northoveref1fc5a2017-08-21 21:56:11 +0000209 bool NeedsRegBlock = TLI.functionArgumentNeedsConsecutiveRegisters(
210 OrigArg.Ty, CallConv, false);
Tim Northoverb18ea162016-09-20 15:20:36 +0000211 for (auto SplitVT : SplitVTs) {
212 Type *SplitTy = SplitVT.getTypeForEVT(Ctx);
Tim Northover9a467182016-09-21 12:57:45 +0000213 SplitArgs.push_back(
Daniel Sanders52b4ce72017-03-07 23:20:35 +0000214 ArgInfo{MRI.createGenericVirtualRegister(getLLTForType(*SplitTy, DL)),
215 SplitTy, OrigArg.Flags, OrigArg.IsFixed});
Tim Northoveref1fc5a2017-08-21 21:56:11 +0000216 if (NeedsRegBlock)
217 SplitArgs.back().Flags.setInConsecutiveRegs();
Tim Northoverb18ea162016-09-20 15:20:36 +0000218 }
219
Tim Northoveref1fc5a2017-08-21 21:56:11 +0000220 SplitArgs.back().Flags.setInConsecutiveRegsLast();
221
Tim Northoverc2c545b2017-03-06 23:50:28 +0000222 for (unsigned i = 0; i < Offsets.size(); ++i)
223 PerformArgSplit(SplitArgs[FirstRegIdx + i].Reg, Offsets[i] * 8);
Tim Northoverb18ea162016-09-20 15:20:36 +0000224}
225
226bool AArch64CallLowering::lowerReturn(MachineIRBuilder &MIRBuilder,
227 const Value *Val, unsigned VReg) const {
228 MachineFunction &MF = MIRBuilder.getMF();
Matthias Braunf1caa282017-12-15 22:22:58 +0000229 const Function &F = MF.getFunction();
Tim Northoverb18ea162016-09-20 15:20:36 +0000230
Tim Northover05cc4852016-12-07 21:05:38 +0000231 auto MIB = MIRBuilder.buildInstrNoInsert(AArch64::RET_ReallyLR);
Tim Northoverb18ea162016-09-20 15:20:36 +0000232 assert(((Val && VReg) || (!Val && !VReg)) && "Return value without a vreg");
Tim Northover05cc4852016-12-07 21:05:38 +0000233 bool Success = true;
Tim Northoverb18ea162016-09-20 15:20:36 +0000234 if (VReg) {
Amara Emerson5a3bb682018-06-01 13:20:32 +0000235 MachineRegisterInfo &MRI = MF.getRegInfo();
236
237 // We zero-extend i1s to i8.
238 if (MRI.getType(VReg).getSizeInBits() == 1)
239 VReg = MIRBuilder.buildZExt(LLT::scalar(8), VReg)->getOperand(0).getReg();
240
Tim Northoverb18ea162016-09-20 15:20:36 +0000241 const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
242 CCAssignFn *AssignFn = TLI.CCAssignFnForReturn(F.getCallingConv());
Tim Northoverb18ea162016-09-20 15:20:36 +0000243 auto &DL = F.getParent()->getDataLayout();
244
Tim Northover9a467182016-09-21 12:57:45 +0000245 ArgInfo OrigArg{VReg, Val->getType()};
Reid Klecknerb5180542017-03-21 16:57:19 +0000246 setArgFlags(OrigArg, AttributeList::ReturnIndex, DL, F);
Tim Northover9a467182016-09-21 12:57:45 +0000247
248 SmallVector<ArgInfo, 8> SplitArgs;
Tim Northoveref1fc5a2017-08-21 21:56:11 +0000249 splitToValueTypes(OrigArg, SplitArgs, DL, MRI, F.getCallingConv(),
Tim Northoverc2c545b2017-03-06 23:50:28 +0000250 [&](unsigned Reg, uint64_t Offset) {
251 MIRBuilder.buildExtract(Reg, VReg, Offset);
Tim Northoverb18ea162016-09-20 15:20:36 +0000252 });
253
Tim Northoverd9433542017-01-17 22:30:10 +0000254 OutgoingArgHandler Handler(MIRBuilder, MRI, MIB, AssignFn, AssignFn);
255 Success = handleAssignments(MIRBuilder, SplitArgs, Handler);
Tim Northoverb18ea162016-09-20 15:20:36 +0000256 }
Tim Northover05cc4852016-12-07 21:05:38 +0000257
258 MIRBuilder.insertInstr(MIB);
259 return Success;
Tim Northoverb18ea162016-09-20 15:20:36 +0000260}
261
Tim Northover862758ec2016-09-21 12:57:35 +0000262bool AArch64CallLowering::lowerFormalArguments(MachineIRBuilder &MIRBuilder,
263 const Function &F,
264 ArrayRef<unsigned> VRegs) const {
Tim Northover406024a2016-08-10 21:44:01 +0000265 MachineFunction &MF = MIRBuilder.getMF();
Tim Northoverb18ea162016-09-20 15:20:36 +0000266 MachineBasicBlock &MBB = MIRBuilder.getMBB();
267 MachineRegisterInfo &MRI = MF.getRegInfo();
Tim Northoverb18ea162016-09-20 15:20:36 +0000268 auto &DL = F.getParent()->getDataLayout();
Tim Northover406024a2016-08-10 21:44:01 +0000269
Tim Northover9a467182016-09-21 12:57:45 +0000270 SmallVector<ArgInfo, 8> SplitArgs;
Tim Northoverb18ea162016-09-20 15:20:36 +0000271 unsigned i = 0;
Reid Kleckner45707d42017-03-16 22:59:15 +0000272 for (auto &Arg : F.args()) {
Amara Emersond78d65c2017-11-30 20:06:02 +0000273 if (DL.getTypeStoreSize(Arg.getType()) == 0)
274 continue;
Tim Northover9a467182016-09-21 12:57:45 +0000275 ArgInfo OrigArg{VRegs[i], Arg.getType()};
Reid Klecknera0b45f42017-05-03 18:17:31 +0000276 setArgFlags(OrigArg, i + AttributeList::FirstArgIndex, DL, F);
Tim Northoverc2c545b2017-03-06 23:50:28 +0000277 bool Split = false;
278 LLT Ty = MRI.getType(VRegs[i]);
279 unsigned Dst = VRegs[i];
280
Tim Northoveref1fc5a2017-08-21 21:56:11 +0000281 splitToValueTypes(OrigArg, SplitArgs, DL, MRI, F.getCallingConv(),
Tim Northoverc2c545b2017-03-06 23:50:28 +0000282 [&](unsigned Reg, uint64_t Offset) {
283 if (!Split) {
284 Split = true;
285 Dst = MRI.createGenericVirtualRegister(Ty);
286 MIRBuilder.buildUndef(Dst);
287 }
288 unsigned Tmp = MRI.createGenericVirtualRegister(Ty);
289 MIRBuilder.buildInsert(Tmp, Dst, Reg, Offset);
290 Dst = Tmp;
Tim Northoverb18ea162016-09-20 15:20:36 +0000291 });
Tim Northoverc2c545b2017-03-06 23:50:28 +0000292
293 if (Dst != VRegs[i])
294 MIRBuilder.buildCopy(VRegs[i], Dst);
Tim Northoverb18ea162016-09-20 15:20:36 +0000295 ++i;
296 }
297
298 if (!MBB.empty())
299 MIRBuilder.setInstr(*MBB.begin());
Tim Northover406024a2016-08-10 21:44:01 +0000300
301 const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
302 CCAssignFn *AssignFn =
303 TLI.CCAssignFnForCall(F.getCallingConv(), /*IsVarArg=*/false);
304
Tim Northoverd9433542017-01-17 22:30:10 +0000305 FormalArgHandler Handler(MIRBuilder, MRI, AssignFn);
306 if (!handleAssignments(MIRBuilder, SplitArgs, Handler))
Tim Northover9a467182016-09-21 12:57:45 +0000307 return false;
Tim Northoverb18ea162016-09-20 15:20:36 +0000308
Tim Northovere9600d82017-02-08 17:57:27 +0000309 if (F.isVarArg()) {
310 if (!MF.getSubtarget<AArch64Subtarget>().isTargetDarwin()) {
311 // FIXME: we need to reimplement saveVarArgsRegisters from
312 // AArch64ISelLowering.
313 return false;
314 }
315
316 // We currently pass all varargs at 8-byte alignment.
317 uint64_t StackOffset = alignTo(Handler.StackUsed, 8);
318
319 auto &MFI = MIRBuilder.getMF().getFrameInfo();
320 AArch64FunctionInfo *FuncInfo = MF.getInfo<AArch64FunctionInfo>();
321 FuncInfo->setVarArgsStackIndex(MFI.CreateFixedObject(4, StackOffset, true));
322 }
323
Tim Northoverb18ea162016-09-20 15:20:36 +0000324 // Move back to the end of the basic block.
325 MIRBuilder.setMBB(MBB);
326
Tim Northover9a467182016-09-21 12:57:45 +0000327 return true;
Tim Northover406024a2016-08-10 21:44:01 +0000328}
329
330bool AArch64CallLowering::lowerCall(MachineIRBuilder &MIRBuilder,
Diana Picusd79253a2017-03-20 14:40:18 +0000331 CallingConv::ID CallConv,
Tim Northover9a467182016-09-21 12:57:45 +0000332 const MachineOperand &Callee,
333 const ArgInfo &OrigRet,
334 ArrayRef<ArgInfo> OrigArgs) const {
Tim Northover406024a2016-08-10 21:44:01 +0000335 MachineFunction &MF = MIRBuilder.getMF();
Matthias Braunf1caa282017-12-15 22:22:58 +0000336 const Function &F = MF.getFunction();
Tim Northoverb18ea162016-09-20 15:20:36 +0000337 MachineRegisterInfo &MRI = MF.getRegInfo();
338 auto &DL = F.getParent()->getDataLayout();
339
Tim Northover9a467182016-09-21 12:57:45 +0000340 SmallVector<ArgInfo, 8> SplitArgs;
341 for (auto &OrigArg : OrigArgs) {
Tim Northoveref1fc5a2017-08-21 21:56:11 +0000342 splitToValueTypes(OrigArg, SplitArgs, DL, MRI, CallConv,
Tim Northoverc2c545b2017-03-06 23:50:28 +0000343 [&](unsigned Reg, uint64_t Offset) {
344 MIRBuilder.buildExtract(Reg, OrigArg.Reg, Offset);
Tim Northoverb18ea162016-09-20 15:20:36 +0000345 });
346 }
Tim Northover406024a2016-08-10 21:44:01 +0000347
Tim Northover406024a2016-08-10 21:44:01 +0000348 // Find out which ABI gets to decide where things go.
349 const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
Tim Northoverd9433542017-01-17 22:30:10 +0000350 CCAssignFn *AssignFnFixed =
Diana Picusd79253a2017-03-20 14:40:18 +0000351 TLI.CCAssignFnForCall(CallConv, /*IsVarArg=*/false);
Tim Northoverd9433542017-01-17 22:30:10 +0000352 CCAssignFn *AssignFnVarArg =
Diana Picusd79253a2017-03-20 14:40:18 +0000353 TLI.CCAssignFnForCall(CallConv, /*IsVarArg=*/true);
Tim Northover406024a2016-08-10 21:44:01 +0000354
Tim Northover509091f2017-01-17 22:43:34 +0000355 auto CallSeqStart = MIRBuilder.buildInstr(AArch64::ADJCALLSTACKDOWN);
356
Tim Northovera5e38fa2016-09-22 13:49:25 +0000357 // Create a temporarily-floating call instruction so we can add the implicit
358 // uses of arg registers.
359 auto MIB = MIRBuilder.buildInstrNoInsert(Callee.isReg() ? AArch64::BLR
360 : AArch64::BL);
Diana Picus116bbab2017-01-13 09:58:52 +0000361 MIB.add(Callee);
Tim Northover406024a2016-08-10 21:44:01 +0000362
363 // Tell the call which registers are clobbered.
364 auto TRI = MF.getSubtarget().getRegisterInfo();
365 MIB.addRegMask(TRI->getCallPreservedMask(MF, F.getCallingConv()));
366
Tim Northovera5e38fa2016-09-22 13:49:25 +0000367 // Do the actual argument marshalling.
368 SmallVector<unsigned, 8> PhysRegs;
Tim Northoverd9433542017-01-17 22:30:10 +0000369 OutgoingArgHandler Handler(MIRBuilder, MRI, MIB, AssignFnFixed,
370 AssignFnVarArg);
371 if (!handleAssignments(MIRBuilder, SplitArgs, Handler))
Tim Northovera5e38fa2016-09-22 13:49:25 +0000372 return false;
373
374 // Now we can add the actual call instruction to the correct basic block.
375 MIRBuilder.insertInstr(MIB);
Tim Northover406024a2016-08-10 21:44:01 +0000376
Quentin Colombetf38015e2016-12-22 21:56:31 +0000377 // If Callee is a reg, since it is used by a target specific
378 // instruction, it must have a register class matching the
379 // constraint of that instruction.
380 if (Callee.isReg())
381 MIB->getOperand(0).setReg(constrainOperandRegClass(
382 MF, *TRI, MRI, *MF.getSubtarget().getInstrInfo(),
Aditya Nandakumar59999052018-02-26 22:56:21 +0000383 *MF.getSubtarget().getRegBankInfo(), *MIB, MIB->getDesc(), Callee, 0));
Quentin Colombetf38015e2016-12-22 21:56:31 +0000384
Tim Northover406024a2016-08-10 21:44:01 +0000385 // Finally we can copy the returned value back into its virtual-register. In
386 // symmetry with the arugments, the physical register must be an
387 // implicit-define of the call instruction.
388 CCAssignFn *RetAssignFn = TLI.CCAssignFnForReturn(F.getCallingConv());
Tim Northover9a467182016-09-21 12:57:45 +0000389 if (OrigRet.Reg) {
390 SplitArgs.clear();
Tim Northoverb18ea162016-09-20 15:20:36 +0000391
392 SmallVector<uint64_t, 8> RegOffsets;
Tim Northover9a467182016-09-21 12:57:45 +0000393 SmallVector<unsigned, 8> SplitRegs;
Tim Northoveref1fc5a2017-08-21 21:56:11 +0000394 splitToValueTypes(OrigRet, SplitArgs, DL, MRI, F.getCallingConv(),
Tim Northoverc2c545b2017-03-06 23:50:28 +0000395 [&](unsigned Reg, uint64_t Offset) {
396 RegOffsets.push_back(Offset);
397 SplitRegs.push_back(Reg);
Tim Northoverb18ea162016-09-20 15:20:36 +0000398 });
399
Tim Northoverd9433542017-01-17 22:30:10 +0000400 CallReturnHandler Handler(MIRBuilder, MRI, MIB, RetAssignFn);
401 if (!handleAssignments(MIRBuilder, SplitArgs, Handler))
Tim Northover9a467182016-09-21 12:57:45 +0000402 return false;
Tim Northover406024a2016-08-10 21:44:01 +0000403
Tim Northoverb18ea162016-09-20 15:20:36 +0000404 if (!RegOffsets.empty())
Tim Northover9a467182016-09-21 12:57:45 +0000405 MIRBuilder.buildSequence(OrigRet.Reg, SplitRegs, RegOffsets);
Tim Northoverb18ea162016-09-20 15:20:36 +0000406 }
407
Serge Pavlovd526b132017-05-09 13:35:13 +0000408 CallSeqStart.addImm(Handler.StackSize).addImm(0);
Tim Northover509091f2017-01-17 22:43:34 +0000409 MIRBuilder.buildInstr(AArch64::ADJCALLSTACKUP)
410 .addImm(Handler.StackSize)
411 .addImm(0);
412
Tim Northover406024a2016-08-10 21:44:01 +0000413 return true;
414}