xref: /aosp_15_r20/external/llvm/lib/Target/AArch64/AArch64CallLowering.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //===-- llvm/lib/Target/AArch64/AArch64CallLowering.cpp - Call lowering ---===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker //                     The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker ///
10*9880d681SAndroid Build Coastguard Worker /// \file
11*9880d681SAndroid Build Coastguard Worker /// This file implements the lowering of LLVM calls to machine code calls for
12*9880d681SAndroid Build Coastguard Worker /// GlobalISel.
13*9880d681SAndroid Build Coastguard Worker ///
14*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
15*9880d681SAndroid Build Coastguard Worker 
16*9880d681SAndroid Build Coastguard Worker #include "AArch64CallLowering.h"
17*9880d681SAndroid Build Coastguard Worker #include "AArch64ISelLowering.h"
18*9880d681SAndroid Build Coastguard Worker 
19*9880d681SAndroid Build Coastguard Worker #include "llvm/CodeGen/GlobalISel/MachineIRBuilder.h"
20*9880d681SAndroid Build Coastguard Worker #include "llvm/CodeGen/MachineInstrBuilder.h"
21*9880d681SAndroid Build Coastguard Worker 
22*9880d681SAndroid Build Coastguard Worker using namespace llvm;
23*9880d681SAndroid Build Coastguard Worker 
24*9880d681SAndroid Build Coastguard Worker #ifndef LLVM_BUILD_GLOBAL_ISEL
25*9880d681SAndroid Build Coastguard Worker #error "This shouldn't be built without GISel"
26*9880d681SAndroid Build Coastguard Worker #endif
27*9880d681SAndroid Build Coastguard Worker 
AArch64CallLowering(const AArch64TargetLowering & TLI)28*9880d681SAndroid Build Coastguard Worker AArch64CallLowering::AArch64CallLowering(const AArch64TargetLowering &TLI)
29*9880d681SAndroid Build Coastguard Worker   : CallLowering(&TLI) {
30*9880d681SAndroid Build Coastguard Worker }
31*9880d681SAndroid Build Coastguard Worker 
lowerReturn(MachineIRBuilder & MIRBuilder,const Value * Val,unsigned VReg) const32*9880d681SAndroid Build Coastguard Worker bool AArch64CallLowering::lowerReturn(MachineIRBuilder &MIRBuilder,
33*9880d681SAndroid Build Coastguard Worker                                         const Value *Val, unsigned VReg) const {
34*9880d681SAndroid Build Coastguard Worker   MachineInstr *Return = MIRBuilder.buildInstr(AArch64::RET_ReallyLR);
35*9880d681SAndroid Build Coastguard Worker   assert(Return && "Unable to build a return instruction?!");
36*9880d681SAndroid Build Coastguard Worker 
37*9880d681SAndroid Build Coastguard Worker   assert(((Val && VReg) || (!Val && !VReg)) && "Return value without a vreg");
38*9880d681SAndroid Build Coastguard Worker   if (VReg) {
39*9880d681SAndroid Build Coastguard Worker     assert(Val->getType()->isIntegerTy() && "Type not supported yet");
40*9880d681SAndroid Build Coastguard Worker     unsigned Size = Val->getType()->getPrimitiveSizeInBits();
41*9880d681SAndroid Build Coastguard Worker     assert((Size == 64 || Size == 32) && "Size not supported yet");
42*9880d681SAndroid Build Coastguard Worker     unsigned ResReg = (Size == 32) ? AArch64::W0 : AArch64::X0;
43*9880d681SAndroid Build Coastguard Worker     // Set the insertion point to be right before Return.
44*9880d681SAndroid Build Coastguard Worker     MIRBuilder.setInstr(*Return, /* Before */ true);
45*9880d681SAndroid Build Coastguard Worker     MachineInstr *Copy =
46*9880d681SAndroid Build Coastguard Worker         MIRBuilder.buildInstr(TargetOpcode::COPY, ResReg, VReg);
47*9880d681SAndroid Build Coastguard Worker     (void)Copy;
48*9880d681SAndroid Build Coastguard Worker     assert(Copy->getNextNode() == Return &&
49*9880d681SAndroid Build Coastguard Worker            "The insertion did not happen where we expected");
50*9880d681SAndroid Build Coastguard Worker     MachineInstrBuilder(MIRBuilder.getMF(), Return)
51*9880d681SAndroid Build Coastguard Worker         .addReg(ResReg, RegState::Implicit);
52*9880d681SAndroid Build Coastguard Worker   }
53*9880d681SAndroid Build Coastguard Worker   return true;
54*9880d681SAndroid Build Coastguard Worker }
55*9880d681SAndroid Build Coastguard Worker 
lowerFormalArguments(MachineIRBuilder & MIRBuilder,const Function::ArgumentListType & Args,const SmallVectorImpl<unsigned> & VRegs) const56*9880d681SAndroid Build Coastguard Worker bool AArch64CallLowering::lowerFormalArguments(
57*9880d681SAndroid Build Coastguard Worker     MachineIRBuilder &MIRBuilder, const Function::ArgumentListType &Args,
58*9880d681SAndroid Build Coastguard Worker     const SmallVectorImpl<unsigned> &VRegs) const {
59*9880d681SAndroid Build Coastguard Worker   MachineFunction &MF = MIRBuilder.getMF();
60*9880d681SAndroid Build Coastguard Worker   const Function &F = *MF.getFunction();
61*9880d681SAndroid Build Coastguard Worker 
62*9880d681SAndroid Build Coastguard Worker   SmallVector<CCValAssign, 16> ArgLocs;
63*9880d681SAndroid Build Coastguard Worker   CCState CCInfo(F.getCallingConv(), F.isVarArg(), MF, ArgLocs, F.getContext());
64*9880d681SAndroid Build Coastguard Worker 
65*9880d681SAndroid Build Coastguard Worker   unsigned NumArgs = Args.size();
66*9880d681SAndroid Build Coastguard Worker   Function::const_arg_iterator CurOrigArg = Args.begin();
67*9880d681SAndroid Build Coastguard Worker   const AArch64TargetLowering &TLI = *getTLI<AArch64TargetLowering>();
68*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0; i != NumArgs; ++i, ++CurOrigArg) {
69*9880d681SAndroid Build Coastguard Worker     MVT ValVT = MVT::getVT(CurOrigArg->getType());
70*9880d681SAndroid Build Coastguard Worker     CCAssignFn *AssignFn =
71*9880d681SAndroid Build Coastguard Worker         TLI.CCAssignFnForCall(F.getCallingConv(), /*IsVarArg=*/false);
72*9880d681SAndroid Build Coastguard Worker     bool Res =
73*9880d681SAndroid Build Coastguard Worker         AssignFn(i, ValVT, ValVT, CCValAssign::Full, ISD::ArgFlagsTy(), CCInfo);
74*9880d681SAndroid Build Coastguard Worker     assert(!Res && "Call operand has unhandled type");
75*9880d681SAndroid Build Coastguard Worker     (void)Res;
76*9880d681SAndroid Build Coastguard Worker   }
77*9880d681SAndroid Build Coastguard Worker   assert(ArgLocs.size() == Args.size() &&
78*9880d681SAndroid Build Coastguard Worker          "We have a different number of location and args?!");
79*9880d681SAndroid Build Coastguard Worker   for (unsigned i = 0, e = ArgLocs.size(); i != e; ++i) {
80*9880d681SAndroid Build Coastguard Worker     CCValAssign &VA = ArgLocs[i];
81*9880d681SAndroid Build Coastguard Worker 
82*9880d681SAndroid Build Coastguard Worker     assert(VA.isRegLoc() && "Not yet implemented");
83*9880d681SAndroid Build Coastguard Worker     // Transform the arguments in physical registers into virtual ones.
84*9880d681SAndroid Build Coastguard Worker     MIRBuilder.getMBB().addLiveIn(VA.getLocReg());
85*9880d681SAndroid Build Coastguard Worker     MIRBuilder.buildInstr(TargetOpcode::COPY, VRegs[i], VA.getLocReg());
86*9880d681SAndroid Build Coastguard Worker 
87*9880d681SAndroid Build Coastguard Worker     switch (VA.getLocInfo()) {
88*9880d681SAndroid Build Coastguard Worker     default:
89*9880d681SAndroid Build Coastguard Worker       llvm_unreachable("Unknown loc info!");
90*9880d681SAndroid Build Coastguard Worker     case CCValAssign::Full:
91*9880d681SAndroid Build Coastguard Worker       break;
92*9880d681SAndroid Build Coastguard Worker     case CCValAssign::BCvt:
93*9880d681SAndroid Build Coastguard Worker       // We don't care about bitcast.
94*9880d681SAndroid Build Coastguard Worker       break;
95*9880d681SAndroid Build Coastguard Worker     case CCValAssign::AExt:
96*9880d681SAndroid Build Coastguard Worker     case CCValAssign::SExt:
97*9880d681SAndroid Build Coastguard Worker     case CCValAssign::ZExt:
98*9880d681SAndroid Build Coastguard Worker       // Zero/Sign extend the register.
99*9880d681SAndroid Build Coastguard Worker       assert(0 && "Not yet implemented");
100*9880d681SAndroid Build Coastguard Worker       break;
101*9880d681SAndroid Build Coastguard Worker     }
102*9880d681SAndroid Build Coastguard Worker   }
103*9880d681SAndroid Build Coastguard Worker   return true;
104*9880d681SAndroid Build Coastguard Worker }
105