xref: /freebsd-src/contrib/llvm-project/llvm/lib/CodeGen/CallingConvLower.cpp (revision 06c3fb2749bda94cb5201f81ffdb8fa6c3161b2e)
10b57cec5SDimitry Andric //===-- CallingConvLower.cpp - Calling Conventions ------------------------===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric //
90b57cec5SDimitry Andric // This file implements the CCState class, used for lowering and implementing
100b57cec5SDimitry Andric // calling conventions.
110b57cec5SDimitry Andric //
120b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
130b57cec5SDimitry Andric 
140b57cec5SDimitry Andric #include "llvm/CodeGen/CallingConvLower.h"
150b57cec5SDimitry Andric #include "llvm/CodeGen/MachineFrameInfo.h"
16e8d8bef9SDimitry Andric #include "llvm/CodeGen/MachineFunction.h"
170b57cec5SDimitry Andric #include "llvm/CodeGen/TargetLowering.h"
180b57cec5SDimitry Andric #include "llvm/CodeGen/TargetRegisterInfo.h"
190b57cec5SDimitry Andric #include "llvm/CodeGen/TargetSubtargetInfo.h"
2081ad6265SDimitry Andric #include "llvm/MC/MCRegisterInfo.h"
210b57cec5SDimitry Andric #include "llvm/Support/Debug.h"
220b57cec5SDimitry Andric #include "llvm/Support/ErrorHandling.h"
230b57cec5SDimitry Andric #include "llvm/Support/SaveAndRestore.h"
240b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h"
250b57cec5SDimitry Andric 
260b57cec5SDimitry Andric using namespace llvm;
270b57cec5SDimitry Andric 
CCState(CallingConv::ID CC,bool IsVarArg,MachineFunction & MF,SmallVectorImpl<CCValAssign> & Locs,LLVMContext & Context,bool NegativeOffsets)28*06c3fb27SDimitry Andric CCState::CCState(CallingConv::ID CC, bool IsVarArg, MachineFunction &MF,
29*06c3fb27SDimitry Andric                  SmallVectorImpl<CCValAssign> &Locs, LLVMContext &Context,
30*06c3fb27SDimitry Andric                  bool NegativeOffsets)
31*06c3fb27SDimitry Andric     : CallingConv(CC), IsVarArg(IsVarArg), MF(MF),
32*06c3fb27SDimitry Andric       TRI(*MF.getSubtarget().getRegisterInfo()), Locs(Locs), Context(Context),
33*06c3fb27SDimitry Andric       NegativeOffsets(NegativeOffsets) {
34*06c3fb27SDimitry Andric 
350b57cec5SDimitry Andric   // No stack is used.
36*06c3fb27SDimitry Andric   StackSize = 0;
370b57cec5SDimitry Andric 
380b57cec5SDimitry Andric   clearByValRegsInfo();
390b57cec5SDimitry Andric   UsedRegs.resize((TRI.getNumRegs()+31)/32);
400b57cec5SDimitry Andric }
410b57cec5SDimitry Andric 
420b57cec5SDimitry Andric /// Allocate space on the stack large enough to pass an argument by value.
430b57cec5SDimitry Andric /// The size and alignment information of the argument is encoded in
440b57cec5SDimitry Andric /// its parameter attribute.
HandleByVal(unsigned ValNo,MVT ValVT,MVT LocVT,CCValAssign::LocInfo LocInfo,int MinSize,Align MinAlign,ISD::ArgFlagsTy ArgFlags)458bcb0991SDimitry Andric void CCState::HandleByVal(unsigned ValNo, MVT ValVT, MVT LocVT,
468bcb0991SDimitry Andric                           CCValAssign::LocInfo LocInfo, int MinSize,
475ffd83dbSDimitry Andric                           Align MinAlign, ISD::ArgFlagsTy ArgFlags) {
485ffd83dbSDimitry Andric   Align Alignment = ArgFlags.getNonZeroByValAlign();
490b57cec5SDimitry Andric   unsigned Size  = ArgFlags.getByValSize();
500b57cec5SDimitry Andric   if (MinSize > (int)Size)
510b57cec5SDimitry Andric     Size = MinSize;
528bcb0991SDimitry Andric   if (MinAlign > Alignment)
538bcb0991SDimitry Andric     Alignment = MinAlign;
548bcb0991SDimitry Andric   ensureMaxAlignment(Alignment);
555ffd83dbSDimitry Andric   MF.getSubtarget().getTargetLowering()->HandleByVal(this, Size, Alignment);
560b57cec5SDimitry Andric   Size = unsigned(alignTo(Size, MinAlign));
57*06c3fb27SDimitry Andric   uint64_t Offset = AllocateStack(Size, Alignment);
580b57cec5SDimitry Andric   addLoc(CCValAssign::getMem(ValNo, ValVT, Offset, LocVT, LocInfo));
590b57cec5SDimitry Andric }
600b57cec5SDimitry Andric 
610b57cec5SDimitry Andric /// Mark a register and all of its aliases as allocated.
MarkAllocated(MCPhysReg Reg)625ffd83dbSDimitry Andric void CCState::MarkAllocated(MCPhysReg Reg) {
630b57cec5SDimitry Andric   for (MCRegAliasIterator AI(Reg, &TRI, true); AI.isValid(); ++AI)
640b57cec5SDimitry Andric     UsedRegs[*AI / 32] |= 1 << (*AI & 31);
650b57cec5SDimitry Andric }
660b57cec5SDimitry Andric 
MarkUnallocated(MCPhysReg Reg)67e8d8bef9SDimitry Andric void CCState::MarkUnallocated(MCPhysReg Reg) {
68e8d8bef9SDimitry Andric   for (MCRegAliasIterator AI(Reg, &TRI, true); AI.isValid(); ++AI)
69e8d8bef9SDimitry Andric     UsedRegs[*AI / 32] &= ~(1 << (*AI & 31));
70e8d8bef9SDimitry Andric }
71e8d8bef9SDimitry Andric 
IsShadowAllocatedReg(MCRegister Reg) const725ffd83dbSDimitry Andric bool CCState::IsShadowAllocatedReg(MCRegister Reg) const {
730b57cec5SDimitry Andric   if (!isAllocated(Reg))
740b57cec5SDimitry Andric     return false;
750b57cec5SDimitry Andric 
7681ad6265SDimitry Andric   for (auto const &ValAssign : Locs)
7781ad6265SDimitry Andric     if (ValAssign.isRegLoc() && TRI.regsOverlap(ValAssign.getLocReg(), Reg))
780b57cec5SDimitry Andric       return false;
790b57cec5SDimitry Andric   return true;
800b57cec5SDimitry Andric }
810b57cec5SDimitry Andric 
820b57cec5SDimitry Andric /// Analyze an array of argument values,
830b57cec5SDimitry Andric /// incorporating info about the formals into this state.
840b57cec5SDimitry Andric void
AnalyzeFormalArguments(const SmallVectorImpl<ISD::InputArg> & Ins,CCAssignFn Fn)850b57cec5SDimitry Andric CCState::AnalyzeFormalArguments(const SmallVectorImpl<ISD::InputArg> &Ins,
860b57cec5SDimitry Andric                                 CCAssignFn Fn) {
870b57cec5SDimitry Andric   unsigned NumArgs = Ins.size();
880b57cec5SDimitry Andric 
890b57cec5SDimitry Andric   for (unsigned i = 0; i != NumArgs; ++i) {
900b57cec5SDimitry Andric     MVT ArgVT = Ins[i].VT;
910b57cec5SDimitry Andric     ISD::ArgFlagsTy ArgFlags = Ins[i].Flags;
928bcb0991SDimitry Andric     if (Fn(i, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, *this))
938bcb0991SDimitry Andric       report_fatal_error("unable to allocate function argument #" + Twine(i));
940b57cec5SDimitry Andric   }
950b57cec5SDimitry Andric }
960b57cec5SDimitry Andric 
970b57cec5SDimitry Andric /// Analyze the return values of a function, returning true if the return can
980b57cec5SDimitry Andric /// be performed without sret-demotion and false otherwise.
CheckReturn(const SmallVectorImpl<ISD::OutputArg> & Outs,CCAssignFn Fn)990b57cec5SDimitry Andric bool CCState::CheckReturn(const SmallVectorImpl<ISD::OutputArg> &Outs,
1000b57cec5SDimitry Andric                           CCAssignFn Fn) {
1010b57cec5SDimitry Andric   // Determine which register each value should be copied into.
1020b57cec5SDimitry Andric   for (unsigned i = 0, e = Outs.size(); i != e; ++i) {
1030b57cec5SDimitry Andric     MVT VT = Outs[i].VT;
1040b57cec5SDimitry Andric     ISD::ArgFlagsTy ArgFlags = Outs[i].Flags;
1050b57cec5SDimitry Andric     if (Fn(i, VT, VT, CCValAssign::Full, ArgFlags, *this))
1060b57cec5SDimitry Andric       return false;
1070b57cec5SDimitry Andric   }
1080b57cec5SDimitry Andric   return true;
1090b57cec5SDimitry Andric }
1100b57cec5SDimitry Andric 
1110b57cec5SDimitry Andric /// Analyze the returned values of a return,
1120b57cec5SDimitry Andric /// incorporating info about the result values into this state.
AnalyzeReturn(const SmallVectorImpl<ISD::OutputArg> & Outs,CCAssignFn Fn)1130b57cec5SDimitry Andric void CCState::AnalyzeReturn(const SmallVectorImpl<ISD::OutputArg> &Outs,
1140b57cec5SDimitry Andric                             CCAssignFn Fn) {
1150b57cec5SDimitry Andric   // Determine which register each value should be copied into.
1160b57cec5SDimitry Andric   for (unsigned i = 0, e = Outs.size(); i != e; ++i) {
1170b57cec5SDimitry Andric     MVT VT = Outs[i].VT;
1180b57cec5SDimitry Andric     ISD::ArgFlagsTy ArgFlags = Outs[i].Flags;
1198bcb0991SDimitry Andric     if (Fn(i, VT, VT, CCValAssign::Full, ArgFlags, *this))
1208bcb0991SDimitry Andric       report_fatal_error("unable to allocate function return #" + Twine(i));
1210b57cec5SDimitry Andric   }
1220b57cec5SDimitry Andric }
1230b57cec5SDimitry Andric 
1240b57cec5SDimitry Andric /// Analyze the outgoing arguments to a call,
1250b57cec5SDimitry Andric /// incorporating info about the passed values into this state.
AnalyzeCallOperands(const SmallVectorImpl<ISD::OutputArg> & Outs,CCAssignFn Fn)1260b57cec5SDimitry Andric void CCState::AnalyzeCallOperands(const SmallVectorImpl<ISD::OutputArg> &Outs,
1270b57cec5SDimitry Andric                                   CCAssignFn Fn) {
1280b57cec5SDimitry Andric   unsigned NumOps = Outs.size();
1290b57cec5SDimitry Andric   for (unsigned i = 0; i != NumOps; ++i) {
1300b57cec5SDimitry Andric     MVT ArgVT = Outs[i].VT;
1310b57cec5SDimitry Andric     ISD::ArgFlagsTy ArgFlags = Outs[i].Flags;
1320b57cec5SDimitry Andric     if (Fn(i, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, *this)) {
1330b57cec5SDimitry Andric #ifndef NDEBUG
1340b57cec5SDimitry Andric       dbgs() << "Call operand #" << i << " has unhandled type "
135*06c3fb27SDimitry Andric              << ArgVT << '\n';
1360b57cec5SDimitry Andric #endif
1370b57cec5SDimitry Andric       llvm_unreachable(nullptr);
1380b57cec5SDimitry Andric     }
1390b57cec5SDimitry Andric   }
1400b57cec5SDimitry Andric }
1410b57cec5SDimitry Andric 
1420b57cec5SDimitry Andric /// Same as above except it takes vectors of types and argument flags.
AnalyzeCallOperands(SmallVectorImpl<MVT> & ArgVTs,SmallVectorImpl<ISD::ArgFlagsTy> & Flags,CCAssignFn Fn)1430b57cec5SDimitry Andric void CCState::AnalyzeCallOperands(SmallVectorImpl<MVT> &ArgVTs,
1440b57cec5SDimitry Andric                                   SmallVectorImpl<ISD::ArgFlagsTy> &Flags,
1450b57cec5SDimitry Andric                                   CCAssignFn Fn) {
1460b57cec5SDimitry Andric   unsigned NumOps = ArgVTs.size();
1470b57cec5SDimitry Andric   for (unsigned i = 0; i != NumOps; ++i) {
1480b57cec5SDimitry Andric     MVT ArgVT = ArgVTs[i];
1490b57cec5SDimitry Andric     ISD::ArgFlagsTy ArgFlags = Flags[i];
1500b57cec5SDimitry Andric     if (Fn(i, ArgVT, ArgVT, CCValAssign::Full, ArgFlags, *this)) {
1510b57cec5SDimitry Andric #ifndef NDEBUG
1520b57cec5SDimitry Andric       dbgs() << "Call operand #" << i << " has unhandled type "
153*06c3fb27SDimitry Andric              << ArgVT << '\n';
1540b57cec5SDimitry Andric #endif
1550b57cec5SDimitry Andric       llvm_unreachable(nullptr);
1560b57cec5SDimitry Andric     }
1570b57cec5SDimitry Andric   }
1580b57cec5SDimitry Andric }
1590b57cec5SDimitry Andric 
1600b57cec5SDimitry Andric /// Analyze the return values of a call, incorporating info about the passed
1610b57cec5SDimitry Andric /// values into this state.
AnalyzeCallResult(const SmallVectorImpl<ISD::InputArg> & Ins,CCAssignFn Fn)1620b57cec5SDimitry Andric void CCState::AnalyzeCallResult(const SmallVectorImpl<ISD::InputArg> &Ins,
1630b57cec5SDimitry Andric                                 CCAssignFn Fn) {
1640b57cec5SDimitry Andric   for (unsigned i = 0, e = Ins.size(); i != e; ++i) {
1650b57cec5SDimitry Andric     MVT VT = Ins[i].VT;
1660b57cec5SDimitry Andric     ISD::ArgFlagsTy Flags = Ins[i].Flags;
1670b57cec5SDimitry Andric     if (Fn(i, VT, VT, CCValAssign::Full, Flags, *this)) {
1680b57cec5SDimitry Andric #ifndef NDEBUG
1690b57cec5SDimitry Andric       dbgs() << "Call result #" << i << " has unhandled type "
170*06c3fb27SDimitry Andric              << VT << '\n';
1710b57cec5SDimitry Andric #endif
1720b57cec5SDimitry Andric       llvm_unreachable(nullptr);
1730b57cec5SDimitry Andric     }
1740b57cec5SDimitry Andric   }
1750b57cec5SDimitry Andric }
1760b57cec5SDimitry Andric 
1770b57cec5SDimitry Andric /// Same as above except it's specialized for calls that produce a single value.
AnalyzeCallResult(MVT VT,CCAssignFn Fn)1780b57cec5SDimitry Andric void CCState::AnalyzeCallResult(MVT VT, CCAssignFn Fn) {
1790b57cec5SDimitry Andric   if (Fn(0, VT, VT, CCValAssign::Full, ISD::ArgFlagsTy(), *this)) {
1800b57cec5SDimitry Andric #ifndef NDEBUG
1810b57cec5SDimitry Andric     dbgs() << "Call result has unhandled type "
182*06c3fb27SDimitry Andric            << VT << '\n';
1830b57cec5SDimitry Andric #endif
1840b57cec5SDimitry Andric     llvm_unreachable(nullptr);
1850b57cec5SDimitry Andric   }
1860b57cec5SDimitry Andric }
1870b57cec5SDimitry Andric 
ensureMaxAlignment(Align Alignment)188e8d8bef9SDimitry Andric void CCState::ensureMaxAlignment(Align Alignment) {
189e8d8bef9SDimitry Andric   if (!AnalyzingMustTailForwardedRegs)
190e8d8bef9SDimitry Andric     MF.getFrameInfo().ensureMaxAlignment(Alignment);
191e8d8bef9SDimitry Andric }
192e8d8bef9SDimitry Andric 
isValueTypeInRegForCC(CallingConv::ID CC,MVT VT)1930b57cec5SDimitry Andric static bool isValueTypeInRegForCC(CallingConv::ID CC, MVT VT) {
1940b57cec5SDimitry Andric   if (VT.isVector())
1950b57cec5SDimitry Andric     return true; // Assume -msse-regparm might be in effect.
1960b57cec5SDimitry Andric   if (!VT.isInteger())
1970b57cec5SDimitry Andric     return false;
198e8d8bef9SDimitry Andric   return (CC == CallingConv::X86_VectorCall || CC == CallingConv::X86_FastCall);
1990b57cec5SDimitry Andric }
2000b57cec5SDimitry Andric 
getRemainingRegParmsForType(SmallVectorImpl<MCPhysReg> & Regs,MVT VT,CCAssignFn Fn)2010b57cec5SDimitry Andric void CCState::getRemainingRegParmsForType(SmallVectorImpl<MCPhysReg> &Regs,
2020b57cec5SDimitry Andric                                           MVT VT, CCAssignFn Fn) {
203*06c3fb27SDimitry Andric   uint64_t SavedStackSize = StackSize;
2048bcb0991SDimitry Andric   Align SavedMaxStackArgAlign = MaxStackArgAlign;
2050b57cec5SDimitry Andric   unsigned NumLocs = Locs.size();
2060b57cec5SDimitry Andric 
2070b57cec5SDimitry Andric   // Set the 'inreg' flag if it is used for this calling convention.
2080b57cec5SDimitry Andric   ISD::ArgFlagsTy Flags;
2090b57cec5SDimitry Andric   if (isValueTypeInRegForCC(CallingConv, VT))
2100b57cec5SDimitry Andric     Flags.setInReg();
2110b57cec5SDimitry Andric 
2120b57cec5SDimitry Andric   // Allocate something of this value type repeatedly until we get assigned a
2130b57cec5SDimitry Andric   // location in memory.
214e8d8bef9SDimitry Andric   bool HaveRegParm;
215e8d8bef9SDimitry Andric   do {
2160b57cec5SDimitry Andric     if (Fn(0, VT, VT, CCValAssign::Full, Flags, *this)) {
2170b57cec5SDimitry Andric #ifndef NDEBUG
218*06c3fb27SDimitry Andric       dbgs() << "Call has unhandled type " << VT
2190b57cec5SDimitry Andric              << " while computing remaining regparms\n";
2200b57cec5SDimitry Andric #endif
2210b57cec5SDimitry Andric       llvm_unreachable(nullptr);
2220b57cec5SDimitry Andric     }
2230b57cec5SDimitry Andric     HaveRegParm = Locs.back().isRegLoc();
224e8d8bef9SDimitry Andric   } while (HaveRegParm);
2250b57cec5SDimitry Andric 
2260b57cec5SDimitry Andric   // Copy all the registers from the value locations we added.
2270b57cec5SDimitry Andric   assert(NumLocs < Locs.size() && "CC assignment failed to add location");
2280b57cec5SDimitry Andric   for (unsigned I = NumLocs, E = Locs.size(); I != E; ++I)
2290b57cec5SDimitry Andric     if (Locs[I].isRegLoc())
2300b57cec5SDimitry Andric       Regs.push_back(MCPhysReg(Locs[I].getLocReg()));
2310b57cec5SDimitry Andric 
2320b57cec5SDimitry Andric   // Clear the assigned values and stack memory. We leave the registers marked
2330b57cec5SDimitry Andric   // as allocated so that future queries don't return the same registers, i.e.
2340b57cec5SDimitry Andric   // when i64 and f64 are both passed in GPRs.
235*06c3fb27SDimitry Andric   StackSize = SavedStackSize;
2360b57cec5SDimitry Andric   MaxStackArgAlign = SavedMaxStackArgAlign;
237bdd1243dSDimitry Andric   Locs.truncate(NumLocs);
2380b57cec5SDimitry Andric }
2390b57cec5SDimitry Andric 
analyzeMustTailForwardedRegisters(SmallVectorImpl<ForwardedRegister> & Forwards,ArrayRef<MVT> RegParmTypes,CCAssignFn Fn)2400b57cec5SDimitry Andric void CCState::analyzeMustTailForwardedRegisters(
2410b57cec5SDimitry Andric     SmallVectorImpl<ForwardedRegister> &Forwards, ArrayRef<MVT> RegParmTypes,
2420b57cec5SDimitry Andric     CCAssignFn Fn) {
2430b57cec5SDimitry Andric   // Oftentimes calling conventions will not user register parameters for
2440b57cec5SDimitry Andric   // variadic functions, so we need to assume we're not variadic so that we get
2450b57cec5SDimitry Andric   // all the registers that might be used in a non-variadic call.
246bdd1243dSDimitry Andric   SaveAndRestore SavedVarArg(IsVarArg, false);
247bdd1243dSDimitry Andric   SaveAndRestore SavedMustTail(AnalyzingMustTailForwardedRegs, true);
2480b57cec5SDimitry Andric 
2490b57cec5SDimitry Andric   for (MVT RegVT : RegParmTypes) {
2500b57cec5SDimitry Andric     SmallVector<MCPhysReg, 8> RemainingRegs;
2510b57cec5SDimitry Andric     getRemainingRegParmsForType(RemainingRegs, RegVT, Fn);
2520b57cec5SDimitry Andric     const TargetLowering *TL = MF.getSubtarget().getTargetLowering();
2530b57cec5SDimitry Andric     const TargetRegisterClass *RC = TL->getRegClassFor(RegVT);
2540b57cec5SDimitry Andric     for (MCPhysReg PReg : RemainingRegs) {
255e8d8bef9SDimitry Andric       Register VReg = MF.addLiveIn(PReg, RC);
2560b57cec5SDimitry Andric       Forwards.push_back(ForwardedRegister(VReg, PReg, RegVT));
2570b57cec5SDimitry Andric     }
2580b57cec5SDimitry Andric   }
2590b57cec5SDimitry Andric }
2600b57cec5SDimitry Andric 
resultsCompatible(CallingConv::ID CalleeCC,CallingConv::ID CallerCC,MachineFunction & MF,LLVMContext & C,const SmallVectorImpl<ISD::InputArg> & Ins,CCAssignFn CalleeFn,CCAssignFn CallerFn)2610b57cec5SDimitry Andric bool CCState::resultsCompatible(CallingConv::ID CalleeCC,
2620b57cec5SDimitry Andric                                 CallingConv::ID CallerCC, MachineFunction &MF,
2630b57cec5SDimitry Andric                                 LLVMContext &C,
2640b57cec5SDimitry Andric                                 const SmallVectorImpl<ISD::InputArg> &Ins,
2650b57cec5SDimitry Andric                                 CCAssignFn CalleeFn, CCAssignFn CallerFn) {
2660b57cec5SDimitry Andric   if (CalleeCC == CallerCC)
2670b57cec5SDimitry Andric     return true;
2680b57cec5SDimitry Andric   SmallVector<CCValAssign, 4> RVLocs1;
2690b57cec5SDimitry Andric   CCState CCInfo1(CalleeCC, false, MF, RVLocs1, C);
2700b57cec5SDimitry Andric   CCInfo1.AnalyzeCallResult(Ins, CalleeFn);
2710b57cec5SDimitry Andric 
2720b57cec5SDimitry Andric   SmallVector<CCValAssign, 4> RVLocs2;
2730b57cec5SDimitry Andric   CCState CCInfo2(CallerCC, false, MF, RVLocs2, C);
2740b57cec5SDimitry Andric   CCInfo2.AnalyzeCallResult(Ins, CallerFn);
2750b57cec5SDimitry Andric 
276bdd1243dSDimitry Andric   auto AreCompatible = [](const CCValAssign &Loc1, const CCValAssign &Loc2) {
277bdd1243dSDimitry Andric     assert(!Loc1.isPendingLoc() && !Loc2.isPendingLoc() &&
278bdd1243dSDimitry Andric            "The location must have been decided by now");
279bdd1243dSDimitry Andric     // Must fill the same part of their locations.
280bdd1243dSDimitry Andric     if (Loc1.getLocInfo() != Loc2.getLocInfo())
2810b57cec5SDimitry Andric       return false;
282bdd1243dSDimitry Andric     // Must both be in the same registers, or both in memory at the same offset.
283bdd1243dSDimitry Andric     if (Loc1.isRegLoc() && Loc2.isRegLoc())
284bdd1243dSDimitry Andric       return Loc1.getLocReg() == Loc2.getLocReg();
285bdd1243dSDimitry Andric     if (Loc1.isMemLoc() && Loc2.isMemLoc())
286bdd1243dSDimitry Andric       return Loc1.getLocMemOffset() == Loc2.getLocMemOffset();
287bdd1243dSDimitry Andric     llvm_unreachable("Unknown location kind");
288bdd1243dSDimitry Andric   };
2895ffd83dbSDimitry Andric 
290bdd1243dSDimitry Andric   return std::equal(RVLocs1.begin(), RVLocs1.end(), RVLocs2.begin(),
291bdd1243dSDimitry Andric                     RVLocs2.end(), AreCompatible);
2920b57cec5SDimitry Andric }
293