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