106c3fb27SDimitry Andric //===- AArch64.cpp --------------------------------------------------------===//
206c3fb27SDimitry Andric //
306c3fb27SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
406c3fb27SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
506c3fb27SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
606c3fb27SDimitry Andric //
706c3fb27SDimitry Andric //===----------------------------------------------------------------------===//
806c3fb27SDimitry Andric 
906c3fb27SDimitry Andric #include "ABIInfoImpl.h"
1006c3fb27SDimitry Andric #include "TargetInfo.h"
1106c3fb27SDimitry Andric 
1206c3fb27SDimitry Andric using namespace clang;
1306c3fb27SDimitry Andric using namespace clang::CodeGen;
1406c3fb27SDimitry Andric 
1506c3fb27SDimitry Andric //===----------------------------------------------------------------------===//
1606c3fb27SDimitry Andric // AArch64 ABI Implementation
1706c3fb27SDimitry Andric //===----------------------------------------------------------------------===//
1806c3fb27SDimitry Andric 
1906c3fb27SDimitry Andric namespace {
2006c3fb27SDimitry Andric 
2106c3fb27SDimitry Andric class AArch64ABIInfo : public ABIInfo {
2206c3fb27SDimitry Andric   AArch64ABIKind Kind;
2306c3fb27SDimitry Andric 
2406c3fb27SDimitry Andric public:
AArch64ABIInfo(CodeGenTypes & CGT,AArch64ABIKind Kind)2506c3fb27SDimitry Andric   AArch64ABIInfo(CodeGenTypes &CGT, AArch64ABIKind Kind)
2606c3fb27SDimitry Andric       : ABIInfo(CGT), Kind(Kind) {}
2706c3fb27SDimitry Andric 
2806c3fb27SDimitry Andric private:
getABIKind() const2906c3fb27SDimitry Andric   AArch64ABIKind getABIKind() const { return Kind; }
isDarwinPCS() const3006c3fb27SDimitry Andric   bool isDarwinPCS() const { return Kind == AArch64ABIKind::DarwinPCS; }
3106c3fb27SDimitry Andric 
3206c3fb27SDimitry Andric   ABIArgInfo classifyReturnType(QualType RetTy, bool IsVariadic) const;
3306c3fb27SDimitry Andric   ABIArgInfo classifyArgumentType(QualType RetTy, bool IsVariadic,
3406c3fb27SDimitry Andric                                   unsigned CallingConvention) const;
3506c3fb27SDimitry Andric   ABIArgInfo coerceIllegalVector(QualType Ty) const;
3606c3fb27SDimitry Andric   bool isHomogeneousAggregateBaseType(QualType Ty) const override;
3706c3fb27SDimitry Andric   bool isHomogeneousAggregateSmallEnough(const Type *Ty,
3806c3fb27SDimitry Andric                                          uint64_t Members) const override;
3906c3fb27SDimitry Andric   bool isZeroLengthBitfieldPermittedInHomogeneousAggregate() const override;
4006c3fb27SDimitry Andric 
4106c3fb27SDimitry Andric   bool isIllegalVectorType(QualType Ty) const;
4206c3fb27SDimitry Andric 
computeInfo(CGFunctionInfo & FI) const4306c3fb27SDimitry Andric   void computeInfo(CGFunctionInfo &FI) const override {
4406c3fb27SDimitry Andric     if (!::classifyReturnType(getCXXABI(), FI, *this))
4506c3fb27SDimitry Andric       FI.getReturnInfo() =
4606c3fb27SDimitry Andric           classifyReturnType(FI.getReturnType(), FI.isVariadic());
4706c3fb27SDimitry Andric 
4806c3fb27SDimitry Andric     for (auto &it : FI.arguments())
4906c3fb27SDimitry Andric       it.info = classifyArgumentType(it.type, FI.isVariadic(),
5006c3fb27SDimitry Andric                                      FI.getCallingConvention());
5106c3fb27SDimitry Andric   }
5206c3fb27SDimitry Andric 
5306c3fb27SDimitry Andric   Address EmitDarwinVAArg(Address VAListAddr, QualType Ty,
5406c3fb27SDimitry Andric                           CodeGenFunction &CGF) const;
5506c3fb27SDimitry Andric 
5606c3fb27SDimitry Andric   Address EmitAAPCSVAArg(Address VAListAddr, QualType Ty,
5706c3fb27SDimitry Andric                          CodeGenFunction &CGF) const;
5806c3fb27SDimitry Andric 
EmitVAArg(CodeGenFunction & CGF,Address VAListAddr,QualType Ty) const5906c3fb27SDimitry Andric   Address EmitVAArg(CodeGenFunction &CGF, Address VAListAddr,
6006c3fb27SDimitry Andric                     QualType Ty) const override {
6106c3fb27SDimitry Andric     llvm::Type *BaseTy = CGF.ConvertType(Ty);
6206c3fb27SDimitry Andric     if (isa<llvm::ScalableVectorType>(BaseTy))
6306c3fb27SDimitry Andric       llvm::report_fatal_error("Passing SVE types to variadic functions is "
6406c3fb27SDimitry Andric                                "currently not supported");
6506c3fb27SDimitry Andric 
6606c3fb27SDimitry Andric     return Kind == AArch64ABIKind::Win64 ? EmitMSVAArg(CGF, VAListAddr, Ty)
6706c3fb27SDimitry Andric            : isDarwinPCS()               ? EmitDarwinVAArg(VAListAddr, Ty, CGF)
6806c3fb27SDimitry Andric                                          : EmitAAPCSVAArg(VAListAddr, Ty, CGF);
6906c3fb27SDimitry Andric   }
7006c3fb27SDimitry Andric 
7106c3fb27SDimitry Andric   Address EmitMSVAArg(CodeGenFunction &CGF, Address VAListAddr,
7206c3fb27SDimitry Andric                       QualType Ty) const override;
7306c3fb27SDimitry Andric 
allowBFloatArgsAndRet() const7406c3fb27SDimitry Andric   bool allowBFloatArgsAndRet() const override {
7506c3fb27SDimitry Andric     return getTarget().hasBFloat16Type();
7606c3fb27SDimitry Andric   }
7706c3fb27SDimitry Andric };
7806c3fb27SDimitry Andric 
7906c3fb27SDimitry Andric class AArch64SwiftABIInfo : public SwiftABIInfo {
8006c3fb27SDimitry Andric public:
AArch64SwiftABIInfo(CodeGenTypes & CGT)8106c3fb27SDimitry Andric   explicit AArch64SwiftABIInfo(CodeGenTypes &CGT)
8206c3fb27SDimitry Andric       : SwiftABIInfo(CGT, /*SwiftErrorInRegister=*/true) {}
8306c3fb27SDimitry Andric 
8406c3fb27SDimitry Andric   bool isLegalVectorType(CharUnits VectorSize, llvm::Type *EltTy,
8506c3fb27SDimitry Andric                          unsigned NumElts) const override;
8606c3fb27SDimitry Andric };
8706c3fb27SDimitry Andric 
8806c3fb27SDimitry Andric class AArch64TargetCodeGenInfo : public TargetCodeGenInfo {
8906c3fb27SDimitry Andric public:
AArch64TargetCodeGenInfo(CodeGenTypes & CGT,AArch64ABIKind Kind)9006c3fb27SDimitry Andric   AArch64TargetCodeGenInfo(CodeGenTypes &CGT, AArch64ABIKind Kind)
9106c3fb27SDimitry Andric       : TargetCodeGenInfo(std::make_unique<AArch64ABIInfo>(CGT, Kind)) {
9206c3fb27SDimitry Andric     SwiftInfo = std::make_unique<AArch64SwiftABIInfo>(CGT);
9306c3fb27SDimitry Andric   }
9406c3fb27SDimitry Andric 
getARCRetainAutoreleasedReturnValueMarker() const9506c3fb27SDimitry Andric   StringRef getARCRetainAutoreleasedReturnValueMarker() const override {
9606c3fb27SDimitry Andric     return "mov\tfp, fp\t\t// marker for objc_retainAutoreleaseReturnValue";
9706c3fb27SDimitry Andric   }
9806c3fb27SDimitry Andric 
getDwarfEHStackPointer(CodeGen::CodeGenModule & M) const9906c3fb27SDimitry Andric   int getDwarfEHStackPointer(CodeGen::CodeGenModule &M) const override {
10006c3fb27SDimitry Andric     return 31;
10106c3fb27SDimitry Andric   }
10206c3fb27SDimitry Andric 
doesReturnSlotInterfereWithArgs() const10306c3fb27SDimitry Andric   bool doesReturnSlotInterfereWithArgs() const override { return false; }
10406c3fb27SDimitry Andric 
setTargetAttributes(const Decl * D,llvm::GlobalValue * GV,CodeGen::CodeGenModule & CGM) const10506c3fb27SDimitry Andric   void setTargetAttributes(const Decl *D, llvm::GlobalValue *GV,
10606c3fb27SDimitry Andric                            CodeGen::CodeGenModule &CGM) const override {
10706c3fb27SDimitry Andric     const FunctionDecl *FD = dyn_cast_or_null<FunctionDecl>(D);
10806c3fb27SDimitry Andric     if (!FD)
10906c3fb27SDimitry Andric       return;
11006c3fb27SDimitry Andric 
11106c3fb27SDimitry Andric     const auto *TA = FD->getAttr<TargetAttr>();
11206c3fb27SDimitry Andric     if (TA == nullptr)
11306c3fb27SDimitry Andric       return;
11406c3fb27SDimitry Andric 
11506c3fb27SDimitry Andric     ParsedTargetAttr Attr =
11606c3fb27SDimitry Andric         CGM.getTarget().parseTargetAttr(TA->getFeaturesStr());
11706c3fb27SDimitry Andric     if (Attr.BranchProtection.empty())
11806c3fb27SDimitry Andric       return;
11906c3fb27SDimitry Andric 
12006c3fb27SDimitry Andric     TargetInfo::BranchProtectionInfo BPI;
12106c3fb27SDimitry Andric     StringRef Error;
12206c3fb27SDimitry Andric     (void)CGM.getTarget().validateBranchProtection(Attr.BranchProtection,
12306c3fb27SDimitry Andric                                                    Attr.CPU, BPI, Error);
12406c3fb27SDimitry Andric     assert(Error.empty());
12506c3fb27SDimitry Andric 
12606c3fb27SDimitry Andric     auto *Fn = cast<llvm::Function>(GV);
12706c3fb27SDimitry Andric     static const char *SignReturnAddrStr[] = {"none", "non-leaf", "all"};
12806c3fb27SDimitry Andric     Fn->addFnAttr("sign-return-address", SignReturnAddrStr[static_cast<int>(BPI.SignReturnAddr)]);
12906c3fb27SDimitry Andric 
13006c3fb27SDimitry Andric     if (BPI.SignReturnAddr != LangOptions::SignReturnAddressScopeKind::None) {
13106c3fb27SDimitry Andric       Fn->addFnAttr("sign-return-address-key",
13206c3fb27SDimitry Andric                     BPI.SignKey == LangOptions::SignReturnAddressKeyKind::AKey
13306c3fb27SDimitry Andric                         ? "a_key"
13406c3fb27SDimitry Andric                         : "b_key");
13506c3fb27SDimitry Andric     }
13606c3fb27SDimitry Andric 
13706c3fb27SDimitry Andric     Fn->addFnAttr("branch-target-enforcement",
13806c3fb27SDimitry Andric                   BPI.BranchTargetEnforcement ? "true" : "false");
139cb14a3feSDimitry Andric     Fn->addFnAttr("branch-protection-pauth-lr",
140cb14a3feSDimitry Andric                   BPI.BranchProtectionPAuthLR ? "true" : "false");
141*297eecfbSDimitry Andric     Fn->addFnAttr("guarded-control-stack",
142*297eecfbSDimitry Andric                   BPI.GuardedControlStack ? "true" : "false");
14306c3fb27SDimitry Andric   }
14406c3fb27SDimitry Andric 
isScalarizableAsmOperand(CodeGen::CodeGenFunction & CGF,llvm::Type * Ty) const14506c3fb27SDimitry Andric   bool isScalarizableAsmOperand(CodeGen::CodeGenFunction &CGF,
14606c3fb27SDimitry Andric                                 llvm::Type *Ty) const override {
14706c3fb27SDimitry Andric     if (CGF.getTarget().hasFeature("ls64")) {
14806c3fb27SDimitry Andric       auto *ST = dyn_cast<llvm::StructType>(Ty);
14906c3fb27SDimitry Andric       if (ST && ST->getNumElements() == 1) {
15006c3fb27SDimitry Andric         auto *AT = dyn_cast<llvm::ArrayType>(ST->getElementType(0));
15106c3fb27SDimitry Andric         if (AT && AT->getNumElements() == 8 &&
15206c3fb27SDimitry Andric             AT->getElementType()->isIntegerTy(64))
15306c3fb27SDimitry Andric           return true;
15406c3fb27SDimitry Andric       }
15506c3fb27SDimitry Andric     }
15606c3fb27SDimitry Andric     return TargetCodeGenInfo::isScalarizableAsmOperand(CGF, Ty);
15706c3fb27SDimitry Andric   }
15806c3fb27SDimitry Andric };
15906c3fb27SDimitry Andric 
16006c3fb27SDimitry Andric class WindowsAArch64TargetCodeGenInfo : public AArch64TargetCodeGenInfo {
16106c3fb27SDimitry Andric public:
WindowsAArch64TargetCodeGenInfo(CodeGenTypes & CGT,AArch64ABIKind K)16206c3fb27SDimitry Andric   WindowsAArch64TargetCodeGenInfo(CodeGenTypes &CGT, AArch64ABIKind K)
16306c3fb27SDimitry Andric       : AArch64TargetCodeGenInfo(CGT, K) {}
16406c3fb27SDimitry Andric 
16506c3fb27SDimitry Andric   void setTargetAttributes(const Decl *D, llvm::GlobalValue *GV,
16606c3fb27SDimitry Andric                            CodeGen::CodeGenModule &CGM) const override;
16706c3fb27SDimitry Andric 
getDependentLibraryOption(llvm::StringRef Lib,llvm::SmallString<24> & Opt) const16806c3fb27SDimitry Andric   void getDependentLibraryOption(llvm::StringRef Lib,
16906c3fb27SDimitry Andric                                  llvm::SmallString<24> &Opt) const override {
17006c3fb27SDimitry Andric     Opt = "/DEFAULTLIB:" + qualifyWindowsLibrary(Lib);
17106c3fb27SDimitry Andric   }
17206c3fb27SDimitry Andric 
getDetectMismatchOption(llvm::StringRef Name,llvm::StringRef Value,llvm::SmallString<32> & Opt) const17306c3fb27SDimitry Andric   void getDetectMismatchOption(llvm::StringRef Name, llvm::StringRef Value,
17406c3fb27SDimitry Andric                                llvm::SmallString<32> &Opt) const override {
17506c3fb27SDimitry Andric     Opt = "/FAILIFMISMATCH:\"" + Name.str() + "=" + Value.str() + "\"";
17606c3fb27SDimitry Andric   }
17706c3fb27SDimitry Andric };
17806c3fb27SDimitry Andric 
setTargetAttributes(const Decl * D,llvm::GlobalValue * GV,CodeGen::CodeGenModule & CGM) const17906c3fb27SDimitry Andric void WindowsAArch64TargetCodeGenInfo::setTargetAttributes(
18006c3fb27SDimitry Andric     const Decl *D, llvm::GlobalValue *GV, CodeGen::CodeGenModule &CGM) const {
18106c3fb27SDimitry Andric   AArch64TargetCodeGenInfo::setTargetAttributes(D, GV, CGM);
18206c3fb27SDimitry Andric   if (GV->isDeclaration())
18306c3fb27SDimitry Andric     return;
18406c3fb27SDimitry Andric   addStackProbeTargetAttributes(D, GV, CGM);
18506c3fb27SDimitry Andric }
18606c3fb27SDimitry Andric }
18706c3fb27SDimitry Andric 
coerceIllegalVector(QualType Ty) const18806c3fb27SDimitry Andric ABIArgInfo AArch64ABIInfo::coerceIllegalVector(QualType Ty) const {
18906c3fb27SDimitry Andric   assert(Ty->isVectorType() && "expected vector type!");
19006c3fb27SDimitry Andric 
19106c3fb27SDimitry Andric   const auto *VT = Ty->castAs<VectorType>();
1925f757f3fSDimitry Andric   if (VT->getVectorKind() == VectorKind::SveFixedLengthPredicate) {
19306c3fb27SDimitry Andric     assert(VT->getElementType()->isBuiltinType() && "expected builtin type!");
19406c3fb27SDimitry Andric     assert(VT->getElementType()->castAs<BuiltinType>()->getKind() ==
19506c3fb27SDimitry Andric                BuiltinType::UChar &&
19606c3fb27SDimitry Andric            "unexpected builtin type for SVE predicate!");
19706c3fb27SDimitry Andric     return ABIArgInfo::getDirect(llvm::ScalableVectorType::get(
19806c3fb27SDimitry Andric         llvm::Type::getInt1Ty(getVMContext()), 16));
19906c3fb27SDimitry Andric   }
20006c3fb27SDimitry Andric 
2015f757f3fSDimitry Andric   if (VT->getVectorKind() == VectorKind::SveFixedLengthData) {
20206c3fb27SDimitry Andric     assert(VT->getElementType()->isBuiltinType() && "expected builtin type!");
20306c3fb27SDimitry Andric 
20406c3fb27SDimitry Andric     const auto *BT = VT->getElementType()->castAs<BuiltinType>();
20506c3fb27SDimitry Andric     llvm::ScalableVectorType *ResType = nullptr;
20606c3fb27SDimitry Andric     switch (BT->getKind()) {
20706c3fb27SDimitry Andric     default:
20806c3fb27SDimitry Andric       llvm_unreachable("unexpected builtin type for SVE vector!");
20906c3fb27SDimitry Andric     case BuiltinType::SChar:
21006c3fb27SDimitry Andric     case BuiltinType::UChar:
21106c3fb27SDimitry Andric       ResType = llvm::ScalableVectorType::get(
21206c3fb27SDimitry Andric           llvm::Type::getInt8Ty(getVMContext()), 16);
21306c3fb27SDimitry Andric       break;
21406c3fb27SDimitry Andric     case BuiltinType::Short:
21506c3fb27SDimitry Andric     case BuiltinType::UShort:
21606c3fb27SDimitry Andric       ResType = llvm::ScalableVectorType::get(
21706c3fb27SDimitry Andric           llvm::Type::getInt16Ty(getVMContext()), 8);
21806c3fb27SDimitry Andric       break;
21906c3fb27SDimitry Andric     case BuiltinType::Int:
22006c3fb27SDimitry Andric     case BuiltinType::UInt:
22106c3fb27SDimitry Andric       ResType = llvm::ScalableVectorType::get(
22206c3fb27SDimitry Andric           llvm::Type::getInt32Ty(getVMContext()), 4);
22306c3fb27SDimitry Andric       break;
22406c3fb27SDimitry Andric     case BuiltinType::Long:
22506c3fb27SDimitry Andric     case BuiltinType::ULong:
22606c3fb27SDimitry Andric       ResType = llvm::ScalableVectorType::get(
22706c3fb27SDimitry Andric           llvm::Type::getInt64Ty(getVMContext()), 2);
22806c3fb27SDimitry Andric       break;
22906c3fb27SDimitry Andric     case BuiltinType::Half:
23006c3fb27SDimitry Andric       ResType = llvm::ScalableVectorType::get(
23106c3fb27SDimitry Andric           llvm::Type::getHalfTy(getVMContext()), 8);
23206c3fb27SDimitry Andric       break;
23306c3fb27SDimitry Andric     case BuiltinType::Float:
23406c3fb27SDimitry Andric       ResType = llvm::ScalableVectorType::get(
23506c3fb27SDimitry Andric           llvm::Type::getFloatTy(getVMContext()), 4);
23606c3fb27SDimitry Andric       break;
23706c3fb27SDimitry Andric     case BuiltinType::Double:
23806c3fb27SDimitry Andric       ResType = llvm::ScalableVectorType::get(
23906c3fb27SDimitry Andric           llvm::Type::getDoubleTy(getVMContext()), 2);
24006c3fb27SDimitry Andric       break;
24106c3fb27SDimitry Andric     case BuiltinType::BFloat16:
24206c3fb27SDimitry Andric       ResType = llvm::ScalableVectorType::get(
24306c3fb27SDimitry Andric           llvm::Type::getBFloatTy(getVMContext()), 8);
24406c3fb27SDimitry Andric       break;
24506c3fb27SDimitry Andric     }
24606c3fb27SDimitry Andric     return ABIArgInfo::getDirect(ResType);
24706c3fb27SDimitry Andric   }
24806c3fb27SDimitry Andric 
24906c3fb27SDimitry Andric   uint64_t Size = getContext().getTypeSize(Ty);
25006c3fb27SDimitry Andric   // Android promotes <2 x i8> to i16, not i32
25106c3fb27SDimitry Andric   if ((isAndroid() || isOHOSFamily()) && (Size <= 16)) {
25206c3fb27SDimitry Andric     llvm::Type *ResType = llvm::Type::getInt16Ty(getVMContext());
25306c3fb27SDimitry Andric     return ABIArgInfo::getDirect(ResType);
25406c3fb27SDimitry Andric   }
25506c3fb27SDimitry Andric   if (Size <= 32) {
25606c3fb27SDimitry Andric     llvm::Type *ResType = llvm::Type::getInt32Ty(getVMContext());
25706c3fb27SDimitry Andric     return ABIArgInfo::getDirect(ResType);
25806c3fb27SDimitry Andric   }
25906c3fb27SDimitry Andric   if (Size == 64) {
26006c3fb27SDimitry Andric     auto *ResType =
26106c3fb27SDimitry Andric         llvm::FixedVectorType::get(llvm::Type::getInt32Ty(getVMContext()), 2);
26206c3fb27SDimitry Andric     return ABIArgInfo::getDirect(ResType);
26306c3fb27SDimitry Andric   }
26406c3fb27SDimitry Andric   if (Size == 128) {
26506c3fb27SDimitry Andric     auto *ResType =
26606c3fb27SDimitry Andric         llvm::FixedVectorType::get(llvm::Type::getInt32Ty(getVMContext()), 4);
26706c3fb27SDimitry Andric     return ABIArgInfo::getDirect(ResType);
26806c3fb27SDimitry Andric   }
26906c3fb27SDimitry Andric   return getNaturalAlignIndirect(Ty, /*ByVal=*/false);
27006c3fb27SDimitry Andric }
27106c3fb27SDimitry Andric 
27206c3fb27SDimitry Andric ABIArgInfo
classifyArgumentType(QualType Ty,bool IsVariadic,unsigned CallingConvention) const27306c3fb27SDimitry Andric AArch64ABIInfo::classifyArgumentType(QualType Ty, bool IsVariadic,
27406c3fb27SDimitry Andric                                      unsigned CallingConvention) const {
27506c3fb27SDimitry Andric   Ty = useFirstFieldIfTransparentUnion(Ty);
27606c3fb27SDimitry Andric 
27706c3fb27SDimitry Andric   // Handle illegal vector types here.
27806c3fb27SDimitry Andric   if (isIllegalVectorType(Ty))
27906c3fb27SDimitry Andric     return coerceIllegalVector(Ty);
28006c3fb27SDimitry Andric 
28106c3fb27SDimitry Andric   if (!isAggregateTypeForABI(Ty)) {
28206c3fb27SDimitry Andric     // Treat an enum type as its underlying type.
28306c3fb27SDimitry Andric     if (const EnumType *EnumTy = Ty->getAs<EnumType>())
28406c3fb27SDimitry Andric       Ty = EnumTy->getDecl()->getIntegerType();
28506c3fb27SDimitry Andric 
28606c3fb27SDimitry Andric     if (const auto *EIT = Ty->getAs<BitIntType>())
28706c3fb27SDimitry Andric       if (EIT->getNumBits() > 128)
28806c3fb27SDimitry Andric         return getNaturalAlignIndirect(Ty);
28906c3fb27SDimitry Andric 
29006c3fb27SDimitry Andric     return (isPromotableIntegerTypeForABI(Ty) && isDarwinPCS()
29106c3fb27SDimitry Andric                 ? ABIArgInfo::getExtend(Ty)
29206c3fb27SDimitry Andric                 : ABIArgInfo::getDirect());
29306c3fb27SDimitry Andric   }
29406c3fb27SDimitry Andric 
29506c3fb27SDimitry Andric   // Structures with either a non-trivial destructor or a non-trivial
29606c3fb27SDimitry Andric   // copy constructor are always indirect.
29706c3fb27SDimitry Andric   if (CGCXXABI::RecordArgABI RAA = getRecordArgABI(Ty, getCXXABI())) {
29806c3fb27SDimitry Andric     return getNaturalAlignIndirect(Ty, /*ByVal=*/RAA ==
29906c3fb27SDimitry Andric                                      CGCXXABI::RAA_DirectInMemory);
30006c3fb27SDimitry Andric   }
30106c3fb27SDimitry Andric 
30206c3fb27SDimitry Andric   // Empty records are always ignored on Darwin, but actually passed in C++ mode
30306c3fb27SDimitry Andric   // elsewhere for GNU compatibility.
30406c3fb27SDimitry Andric   uint64_t Size = getContext().getTypeSize(Ty);
30506c3fb27SDimitry Andric   bool IsEmpty = isEmptyRecord(getContext(), Ty, true);
30606c3fb27SDimitry Andric   if (IsEmpty || Size == 0) {
30706c3fb27SDimitry Andric     if (!getContext().getLangOpts().CPlusPlus || isDarwinPCS())
30806c3fb27SDimitry Andric       return ABIArgInfo::getIgnore();
30906c3fb27SDimitry Andric 
31006c3fb27SDimitry Andric     // GNU C mode. The only argument that gets ignored is an empty one with size
31106c3fb27SDimitry Andric     // 0.
31206c3fb27SDimitry Andric     if (IsEmpty && Size == 0)
31306c3fb27SDimitry Andric       return ABIArgInfo::getIgnore();
31406c3fb27SDimitry Andric     return ABIArgInfo::getDirect(llvm::Type::getInt8Ty(getVMContext()));
31506c3fb27SDimitry Andric   }
31606c3fb27SDimitry Andric 
31706c3fb27SDimitry Andric   // Homogeneous Floating-point Aggregates (HFAs) need to be expanded.
31806c3fb27SDimitry Andric   const Type *Base = nullptr;
31906c3fb27SDimitry Andric   uint64_t Members = 0;
32006c3fb27SDimitry Andric   bool IsWin64 = Kind == AArch64ABIKind::Win64 ||
32106c3fb27SDimitry Andric                  CallingConvention == llvm::CallingConv::Win64;
32206c3fb27SDimitry Andric   bool IsWinVariadic = IsWin64 && IsVariadic;
32306c3fb27SDimitry Andric   // In variadic functions on Windows, all composite types are treated alike,
32406c3fb27SDimitry Andric   // no special handling of HFAs/HVAs.
32506c3fb27SDimitry Andric   if (!IsWinVariadic && isHomogeneousAggregate(Ty, Base, Members)) {
32606c3fb27SDimitry Andric     if (Kind != AArch64ABIKind::AAPCS)
32706c3fb27SDimitry Andric       return ABIArgInfo::getDirect(
32806c3fb27SDimitry Andric           llvm::ArrayType::get(CGT.ConvertType(QualType(Base, 0)), Members));
32906c3fb27SDimitry Andric 
3305f757f3fSDimitry Andric     // For HFAs/HVAs, cap the argument alignment to 16, otherwise
3315f757f3fSDimitry Andric     // set it to 8 according to the AAPCS64 document.
33206c3fb27SDimitry Andric     unsigned Align =
33306c3fb27SDimitry Andric         getContext().getTypeUnadjustedAlignInChars(Ty).getQuantity();
3345f757f3fSDimitry Andric     Align = (Align >= 16) ? 16 : 8;
33506c3fb27SDimitry Andric     return ABIArgInfo::getDirect(
33606c3fb27SDimitry Andric         llvm::ArrayType::get(CGT.ConvertType(QualType(Base, 0)), Members), 0,
33706c3fb27SDimitry Andric         nullptr, true, Align);
33806c3fb27SDimitry Andric   }
33906c3fb27SDimitry Andric 
34006c3fb27SDimitry Andric   // Aggregates <= 16 bytes are passed directly in registers or on the stack.
34106c3fb27SDimitry Andric   if (Size <= 128) {
34206c3fb27SDimitry Andric     // On RenderScript, coerce Aggregates <= 16 bytes to an integer array of
34306c3fb27SDimitry Andric     // same size and alignment.
34406c3fb27SDimitry Andric     if (getTarget().isRenderScriptTarget()) {
34506c3fb27SDimitry Andric       return coerceToIntArray(Ty, getContext(), getVMContext());
34606c3fb27SDimitry Andric     }
34706c3fb27SDimitry Andric     unsigned Alignment;
34806c3fb27SDimitry Andric     if (Kind == AArch64ABIKind::AAPCS) {
34906c3fb27SDimitry Andric       Alignment = getContext().getTypeUnadjustedAlign(Ty);
35006c3fb27SDimitry Andric       Alignment = Alignment < 128 ? 64 : 128;
35106c3fb27SDimitry Andric     } else {
35206c3fb27SDimitry Andric       Alignment =
35306c3fb27SDimitry Andric           std::max(getContext().getTypeAlign(Ty),
35406c3fb27SDimitry Andric                    (unsigned)getTarget().getPointerWidth(LangAS::Default));
35506c3fb27SDimitry Andric     }
35606c3fb27SDimitry Andric     Size = llvm::alignTo(Size, Alignment);
35706c3fb27SDimitry Andric 
35806c3fb27SDimitry Andric     // We use a pair of i64 for 16-byte aggregate with 8-byte alignment.
35906c3fb27SDimitry Andric     // For aggregates with 16-byte alignment, we use i128.
36006c3fb27SDimitry Andric     llvm::Type *BaseTy = llvm::Type::getIntNTy(getVMContext(), Alignment);
36106c3fb27SDimitry Andric     return ABIArgInfo::getDirect(
36206c3fb27SDimitry Andric         Size == Alignment ? BaseTy
36306c3fb27SDimitry Andric                           : llvm::ArrayType::get(BaseTy, Size / Alignment));
36406c3fb27SDimitry Andric   }
36506c3fb27SDimitry Andric 
36606c3fb27SDimitry Andric   return getNaturalAlignIndirect(Ty, /*ByVal=*/false);
36706c3fb27SDimitry Andric }
36806c3fb27SDimitry Andric 
classifyReturnType(QualType RetTy,bool IsVariadic) const36906c3fb27SDimitry Andric ABIArgInfo AArch64ABIInfo::classifyReturnType(QualType RetTy,
37006c3fb27SDimitry Andric                                               bool IsVariadic) const {
37106c3fb27SDimitry Andric   if (RetTy->isVoidType())
37206c3fb27SDimitry Andric     return ABIArgInfo::getIgnore();
37306c3fb27SDimitry Andric 
37406c3fb27SDimitry Andric   if (const auto *VT = RetTy->getAs<VectorType>()) {
3755f757f3fSDimitry Andric     if (VT->getVectorKind() == VectorKind::SveFixedLengthData ||
3765f757f3fSDimitry Andric         VT->getVectorKind() == VectorKind::SveFixedLengthPredicate)
37706c3fb27SDimitry Andric       return coerceIllegalVector(RetTy);
37806c3fb27SDimitry Andric   }
37906c3fb27SDimitry Andric 
38006c3fb27SDimitry Andric   // Large vector types should be returned via memory.
38106c3fb27SDimitry Andric   if (RetTy->isVectorType() && getContext().getTypeSize(RetTy) > 128)
38206c3fb27SDimitry Andric     return getNaturalAlignIndirect(RetTy);
38306c3fb27SDimitry Andric 
38406c3fb27SDimitry Andric   if (!isAggregateTypeForABI(RetTy)) {
38506c3fb27SDimitry Andric     // Treat an enum type as its underlying type.
38606c3fb27SDimitry Andric     if (const EnumType *EnumTy = RetTy->getAs<EnumType>())
38706c3fb27SDimitry Andric       RetTy = EnumTy->getDecl()->getIntegerType();
38806c3fb27SDimitry Andric 
38906c3fb27SDimitry Andric     if (const auto *EIT = RetTy->getAs<BitIntType>())
39006c3fb27SDimitry Andric       if (EIT->getNumBits() > 128)
39106c3fb27SDimitry Andric         return getNaturalAlignIndirect(RetTy);
39206c3fb27SDimitry Andric 
39306c3fb27SDimitry Andric     return (isPromotableIntegerTypeForABI(RetTy) && isDarwinPCS()
39406c3fb27SDimitry Andric                 ? ABIArgInfo::getExtend(RetTy)
39506c3fb27SDimitry Andric                 : ABIArgInfo::getDirect());
39606c3fb27SDimitry Andric   }
39706c3fb27SDimitry Andric 
39806c3fb27SDimitry Andric   uint64_t Size = getContext().getTypeSize(RetTy);
39906c3fb27SDimitry Andric   if (isEmptyRecord(getContext(), RetTy, true) || Size == 0)
40006c3fb27SDimitry Andric     return ABIArgInfo::getIgnore();
40106c3fb27SDimitry Andric 
40206c3fb27SDimitry Andric   const Type *Base = nullptr;
40306c3fb27SDimitry Andric   uint64_t Members = 0;
40406c3fb27SDimitry Andric   if (isHomogeneousAggregate(RetTy, Base, Members) &&
40506c3fb27SDimitry Andric       !(getTarget().getTriple().getArch() == llvm::Triple::aarch64_32 &&
40606c3fb27SDimitry Andric         IsVariadic))
40706c3fb27SDimitry Andric     // Homogeneous Floating-point Aggregates (HFAs) are returned directly.
40806c3fb27SDimitry Andric     return ABIArgInfo::getDirect();
40906c3fb27SDimitry Andric 
41006c3fb27SDimitry Andric   // Aggregates <= 16 bytes are returned directly in registers or on the stack.
41106c3fb27SDimitry Andric   if (Size <= 128) {
41206c3fb27SDimitry Andric     // On RenderScript, coerce Aggregates <= 16 bytes to an integer array of
41306c3fb27SDimitry Andric     // same size and alignment.
41406c3fb27SDimitry Andric     if (getTarget().isRenderScriptTarget()) {
41506c3fb27SDimitry Andric       return coerceToIntArray(RetTy, getContext(), getVMContext());
41606c3fb27SDimitry Andric     }
41706c3fb27SDimitry Andric 
41806c3fb27SDimitry Andric     if (Size <= 64 && getDataLayout().isLittleEndian()) {
41906c3fb27SDimitry Andric       // Composite types are returned in lower bits of a 64-bit register for LE,
42006c3fb27SDimitry Andric       // and in higher bits for BE. However, integer types are always returned
42106c3fb27SDimitry Andric       // in lower bits for both LE and BE, and they are not rounded up to
42206c3fb27SDimitry Andric       // 64-bits. We can skip rounding up of composite types for LE, but not for
42306c3fb27SDimitry Andric       // BE, otherwise composite types will be indistinguishable from integer
42406c3fb27SDimitry Andric       // types.
42506c3fb27SDimitry Andric       return ABIArgInfo::getDirect(
42606c3fb27SDimitry Andric           llvm::IntegerType::get(getVMContext(), Size));
42706c3fb27SDimitry Andric     }
42806c3fb27SDimitry Andric 
42906c3fb27SDimitry Andric     unsigned Alignment = getContext().getTypeAlign(RetTy);
43006c3fb27SDimitry Andric     Size = llvm::alignTo(Size, 64); // round up to multiple of 8 bytes
43106c3fb27SDimitry Andric 
43206c3fb27SDimitry Andric     // We use a pair of i64 for 16-byte aggregate with 8-byte alignment.
43306c3fb27SDimitry Andric     // For aggregates with 16-byte alignment, we use i128.
43406c3fb27SDimitry Andric     if (Alignment < 128 && Size == 128) {
43506c3fb27SDimitry Andric       llvm::Type *BaseTy = llvm::Type::getInt64Ty(getVMContext());
43606c3fb27SDimitry Andric       return ABIArgInfo::getDirect(llvm::ArrayType::get(BaseTy, Size / 64));
43706c3fb27SDimitry Andric     }
43806c3fb27SDimitry Andric     return ABIArgInfo::getDirect(llvm::IntegerType::get(getVMContext(), Size));
43906c3fb27SDimitry Andric   }
44006c3fb27SDimitry Andric 
44106c3fb27SDimitry Andric   return getNaturalAlignIndirect(RetTy);
44206c3fb27SDimitry Andric }
44306c3fb27SDimitry Andric 
44406c3fb27SDimitry Andric /// isIllegalVectorType - check whether the vector type is legal for AArch64.
isIllegalVectorType(QualType Ty) const44506c3fb27SDimitry Andric bool AArch64ABIInfo::isIllegalVectorType(QualType Ty) const {
44606c3fb27SDimitry Andric   if (const VectorType *VT = Ty->getAs<VectorType>()) {
44706c3fb27SDimitry Andric     // Check whether VT is a fixed-length SVE vector. These types are
44806c3fb27SDimitry Andric     // represented as scalable vectors in function args/return and must be
44906c3fb27SDimitry Andric     // coerced from fixed vectors.
4505f757f3fSDimitry Andric     if (VT->getVectorKind() == VectorKind::SveFixedLengthData ||
4515f757f3fSDimitry Andric         VT->getVectorKind() == VectorKind::SveFixedLengthPredicate)
45206c3fb27SDimitry Andric       return true;
45306c3fb27SDimitry Andric 
45406c3fb27SDimitry Andric     // Check whether VT is legal.
45506c3fb27SDimitry Andric     unsigned NumElements = VT->getNumElements();
45606c3fb27SDimitry Andric     uint64_t Size = getContext().getTypeSize(VT);
45706c3fb27SDimitry Andric     // NumElements should be power of 2.
45806c3fb27SDimitry Andric     if (!llvm::isPowerOf2_32(NumElements))
45906c3fb27SDimitry Andric       return true;
46006c3fb27SDimitry Andric 
46106c3fb27SDimitry Andric     // arm64_32 has to be compatible with the ARM logic here, which allows huge
46206c3fb27SDimitry Andric     // vectors for some reason.
46306c3fb27SDimitry Andric     llvm::Triple Triple = getTarget().getTriple();
46406c3fb27SDimitry Andric     if (Triple.getArch() == llvm::Triple::aarch64_32 &&
46506c3fb27SDimitry Andric         Triple.isOSBinFormatMachO())
46606c3fb27SDimitry Andric       return Size <= 32;
46706c3fb27SDimitry Andric 
46806c3fb27SDimitry Andric     return Size != 64 && (Size != 128 || NumElements == 1);
46906c3fb27SDimitry Andric   }
47006c3fb27SDimitry Andric   return false;
47106c3fb27SDimitry Andric }
47206c3fb27SDimitry Andric 
isLegalVectorType(CharUnits VectorSize,llvm::Type * EltTy,unsigned NumElts) const47306c3fb27SDimitry Andric bool AArch64SwiftABIInfo::isLegalVectorType(CharUnits VectorSize,
47406c3fb27SDimitry Andric                                             llvm::Type *EltTy,
47506c3fb27SDimitry Andric                                             unsigned NumElts) const {
47606c3fb27SDimitry Andric   if (!llvm::isPowerOf2_32(NumElts))
47706c3fb27SDimitry Andric     return false;
47806c3fb27SDimitry Andric   if (VectorSize.getQuantity() != 8 &&
47906c3fb27SDimitry Andric       (VectorSize.getQuantity() != 16 || NumElts == 1))
48006c3fb27SDimitry Andric     return false;
48106c3fb27SDimitry Andric   return true;
48206c3fb27SDimitry Andric }
48306c3fb27SDimitry Andric 
isHomogeneousAggregateBaseType(QualType Ty) const48406c3fb27SDimitry Andric bool AArch64ABIInfo::isHomogeneousAggregateBaseType(QualType Ty) const {
48506c3fb27SDimitry Andric   // Homogeneous aggregates for AAPCS64 must have base types of a floating
48606c3fb27SDimitry Andric   // point type or a short-vector type. This is the same as the 32-bit ABI,
48706c3fb27SDimitry Andric   // but with the difference that any floating-point type is allowed,
48806c3fb27SDimitry Andric   // including __fp16.
48906c3fb27SDimitry Andric   if (const BuiltinType *BT = Ty->getAs<BuiltinType>()) {
49006c3fb27SDimitry Andric     if (BT->isFloatingPoint())
49106c3fb27SDimitry Andric       return true;
49206c3fb27SDimitry Andric   } else if (const VectorType *VT = Ty->getAs<VectorType>()) {
49306c3fb27SDimitry Andric     unsigned VecSize = getContext().getTypeSize(VT);
49406c3fb27SDimitry Andric     if (VecSize == 64 || VecSize == 128)
49506c3fb27SDimitry Andric       return true;
49606c3fb27SDimitry Andric   }
49706c3fb27SDimitry Andric   return false;
49806c3fb27SDimitry Andric }
49906c3fb27SDimitry Andric 
isHomogeneousAggregateSmallEnough(const Type * Base,uint64_t Members) const50006c3fb27SDimitry Andric bool AArch64ABIInfo::isHomogeneousAggregateSmallEnough(const Type *Base,
50106c3fb27SDimitry Andric                                                        uint64_t Members) const {
50206c3fb27SDimitry Andric   return Members <= 4;
50306c3fb27SDimitry Andric }
50406c3fb27SDimitry Andric 
isZeroLengthBitfieldPermittedInHomogeneousAggregate() const50506c3fb27SDimitry Andric bool AArch64ABIInfo::isZeroLengthBitfieldPermittedInHomogeneousAggregate()
50606c3fb27SDimitry Andric     const {
50706c3fb27SDimitry Andric   // AAPCS64 says that the rule for whether something is a homogeneous
50806c3fb27SDimitry Andric   // aggregate is applied to the output of the data layout decision. So
50906c3fb27SDimitry Andric   // anything that doesn't affect the data layout also does not affect
51006c3fb27SDimitry Andric   // homogeneity. In particular, zero-length bitfields don't stop a struct
51106c3fb27SDimitry Andric   // being homogeneous.
51206c3fb27SDimitry Andric   return true;
51306c3fb27SDimitry Andric }
51406c3fb27SDimitry Andric 
EmitAAPCSVAArg(Address VAListAddr,QualType Ty,CodeGenFunction & CGF) const51506c3fb27SDimitry Andric Address AArch64ABIInfo::EmitAAPCSVAArg(Address VAListAddr, QualType Ty,
51606c3fb27SDimitry Andric                                        CodeGenFunction &CGF) const {
51706c3fb27SDimitry Andric   ABIArgInfo AI = classifyArgumentType(Ty, /*IsVariadic=*/true,
51806c3fb27SDimitry Andric                                        CGF.CurFnInfo->getCallingConvention());
51906c3fb27SDimitry Andric   // Empty records are ignored for parameter passing purposes.
52006c3fb27SDimitry Andric   if (AI.isIgnore()) {
52106c3fb27SDimitry Andric     uint64_t PointerSize = getTarget().getPointerWidth(LangAS::Default) / 8;
52206c3fb27SDimitry Andric     CharUnits SlotSize = CharUnits::fromQuantity(PointerSize);
52306c3fb27SDimitry Andric     VAListAddr = VAListAddr.withElementType(CGF.Int8PtrTy);
52406c3fb27SDimitry Andric     auto *Load = CGF.Builder.CreateLoad(VAListAddr);
52506c3fb27SDimitry Andric     return Address(Load, CGF.ConvertTypeForMem(Ty), SlotSize);
52606c3fb27SDimitry Andric   }
52706c3fb27SDimitry Andric 
52806c3fb27SDimitry Andric   bool IsIndirect = AI.isIndirect();
52906c3fb27SDimitry Andric 
53006c3fb27SDimitry Andric   llvm::Type *BaseTy = CGF.ConvertType(Ty);
53106c3fb27SDimitry Andric   if (IsIndirect)
53206c3fb27SDimitry Andric     BaseTy = llvm::PointerType::getUnqual(BaseTy);
53306c3fb27SDimitry Andric   else if (AI.getCoerceToType())
53406c3fb27SDimitry Andric     BaseTy = AI.getCoerceToType();
53506c3fb27SDimitry Andric 
53606c3fb27SDimitry Andric   unsigned NumRegs = 1;
53706c3fb27SDimitry Andric   if (llvm::ArrayType *ArrTy = dyn_cast<llvm::ArrayType>(BaseTy)) {
53806c3fb27SDimitry Andric     BaseTy = ArrTy->getElementType();
53906c3fb27SDimitry Andric     NumRegs = ArrTy->getNumElements();
54006c3fb27SDimitry Andric   }
54106c3fb27SDimitry Andric   bool IsFPR = BaseTy->isFloatingPointTy() || BaseTy->isVectorTy();
54206c3fb27SDimitry Andric 
54306c3fb27SDimitry Andric   // The AArch64 va_list type and handling is specified in the Procedure Call
54406c3fb27SDimitry Andric   // Standard, section B.4:
54506c3fb27SDimitry Andric   //
54606c3fb27SDimitry Andric   // struct {
54706c3fb27SDimitry Andric   //   void *__stack;
54806c3fb27SDimitry Andric   //   void *__gr_top;
54906c3fb27SDimitry Andric   //   void *__vr_top;
55006c3fb27SDimitry Andric   //   int __gr_offs;
55106c3fb27SDimitry Andric   //   int __vr_offs;
55206c3fb27SDimitry Andric   // };
55306c3fb27SDimitry Andric 
55406c3fb27SDimitry Andric   llvm::BasicBlock *MaybeRegBlock = CGF.createBasicBlock("vaarg.maybe_reg");
55506c3fb27SDimitry Andric   llvm::BasicBlock *InRegBlock = CGF.createBasicBlock("vaarg.in_reg");
55606c3fb27SDimitry Andric   llvm::BasicBlock *OnStackBlock = CGF.createBasicBlock("vaarg.on_stack");
55706c3fb27SDimitry Andric   llvm::BasicBlock *ContBlock = CGF.createBasicBlock("vaarg.end");
55806c3fb27SDimitry Andric 
55906c3fb27SDimitry Andric   CharUnits TySize = getContext().getTypeSizeInChars(Ty);
56006c3fb27SDimitry Andric   CharUnits TyAlign = getContext().getTypeUnadjustedAlignInChars(Ty);
56106c3fb27SDimitry Andric 
56206c3fb27SDimitry Andric   Address reg_offs_p = Address::invalid();
56306c3fb27SDimitry Andric   llvm::Value *reg_offs = nullptr;
56406c3fb27SDimitry Andric   int reg_top_index;
56506c3fb27SDimitry Andric   int RegSize = IsIndirect ? 8 : TySize.getQuantity();
56606c3fb27SDimitry Andric   if (!IsFPR) {
56706c3fb27SDimitry Andric     // 3 is the field number of __gr_offs
56806c3fb27SDimitry Andric     reg_offs_p = CGF.Builder.CreateStructGEP(VAListAddr, 3, "gr_offs_p");
56906c3fb27SDimitry Andric     reg_offs = CGF.Builder.CreateLoad(reg_offs_p, "gr_offs");
57006c3fb27SDimitry Andric     reg_top_index = 1; // field number for __gr_top
57106c3fb27SDimitry Andric     RegSize = llvm::alignTo(RegSize, 8);
57206c3fb27SDimitry Andric   } else {
57306c3fb27SDimitry Andric     // 4 is the field number of __vr_offs.
57406c3fb27SDimitry Andric     reg_offs_p = CGF.Builder.CreateStructGEP(VAListAddr, 4, "vr_offs_p");
57506c3fb27SDimitry Andric     reg_offs = CGF.Builder.CreateLoad(reg_offs_p, "vr_offs");
57606c3fb27SDimitry Andric     reg_top_index = 2; // field number for __vr_top
57706c3fb27SDimitry Andric     RegSize = 16 * NumRegs;
57806c3fb27SDimitry Andric   }
57906c3fb27SDimitry Andric 
58006c3fb27SDimitry Andric   //=======================================
58106c3fb27SDimitry Andric   // Find out where argument was passed
58206c3fb27SDimitry Andric   //=======================================
58306c3fb27SDimitry Andric 
58406c3fb27SDimitry Andric   // If reg_offs >= 0 we're already using the stack for this type of
58506c3fb27SDimitry Andric   // argument. We don't want to keep updating reg_offs (in case it overflows,
58606c3fb27SDimitry Andric   // though anyone passing 2GB of arguments, each at most 16 bytes, deserves
58706c3fb27SDimitry Andric   // whatever they get).
58806c3fb27SDimitry Andric   llvm::Value *UsingStack = nullptr;
58906c3fb27SDimitry Andric   UsingStack = CGF.Builder.CreateICmpSGE(
59006c3fb27SDimitry Andric       reg_offs, llvm::ConstantInt::get(CGF.Int32Ty, 0));
59106c3fb27SDimitry Andric 
59206c3fb27SDimitry Andric   CGF.Builder.CreateCondBr(UsingStack, OnStackBlock, MaybeRegBlock);
59306c3fb27SDimitry Andric 
59406c3fb27SDimitry Andric   // Otherwise, at least some kind of argument could go in these registers, the
59506c3fb27SDimitry Andric   // question is whether this particular type is too big.
59606c3fb27SDimitry Andric   CGF.EmitBlock(MaybeRegBlock);
59706c3fb27SDimitry Andric 
59806c3fb27SDimitry Andric   // Integer arguments may need to correct register alignment (for example a
59906c3fb27SDimitry Andric   // "struct { __int128 a; };" gets passed in x_2N, x_{2N+1}). In this case we
60006c3fb27SDimitry Andric   // align __gr_offs to calculate the potential address.
60106c3fb27SDimitry Andric   if (!IsFPR && !IsIndirect && TyAlign.getQuantity() > 8) {
60206c3fb27SDimitry Andric     int Align = TyAlign.getQuantity();
60306c3fb27SDimitry Andric 
60406c3fb27SDimitry Andric     reg_offs = CGF.Builder.CreateAdd(
60506c3fb27SDimitry Andric         reg_offs, llvm::ConstantInt::get(CGF.Int32Ty, Align - 1),
60606c3fb27SDimitry Andric         "align_regoffs");
60706c3fb27SDimitry Andric     reg_offs = CGF.Builder.CreateAnd(
60806c3fb27SDimitry Andric         reg_offs, llvm::ConstantInt::get(CGF.Int32Ty, -Align),
60906c3fb27SDimitry Andric         "aligned_regoffs");
61006c3fb27SDimitry Andric   }
61106c3fb27SDimitry Andric 
61206c3fb27SDimitry Andric   // Update the gr_offs/vr_offs pointer for next call to va_arg on this va_list.
61306c3fb27SDimitry Andric   // The fact that this is done unconditionally reflects the fact that
61406c3fb27SDimitry Andric   // allocating an argument to the stack also uses up all the remaining
61506c3fb27SDimitry Andric   // registers of the appropriate kind.
61606c3fb27SDimitry Andric   llvm::Value *NewOffset = nullptr;
61706c3fb27SDimitry Andric   NewOffset = CGF.Builder.CreateAdd(
61806c3fb27SDimitry Andric       reg_offs, llvm::ConstantInt::get(CGF.Int32Ty, RegSize), "new_reg_offs");
61906c3fb27SDimitry Andric   CGF.Builder.CreateStore(NewOffset, reg_offs_p);
62006c3fb27SDimitry Andric 
62106c3fb27SDimitry Andric   // Now we're in a position to decide whether this argument really was in
62206c3fb27SDimitry Andric   // registers or not.
62306c3fb27SDimitry Andric   llvm::Value *InRegs = nullptr;
62406c3fb27SDimitry Andric   InRegs = CGF.Builder.CreateICmpSLE(
62506c3fb27SDimitry Andric       NewOffset, llvm::ConstantInt::get(CGF.Int32Ty, 0), "inreg");
62606c3fb27SDimitry Andric 
62706c3fb27SDimitry Andric   CGF.Builder.CreateCondBr(InRegs, InRegBlock, OnStackBlock);
62806c3fb27SDimitry Andric 
62906c3fb27SDimitry Andric   //=======================================
63006c3fb27SDimitry Andric   // Argument was in registers
63106c3fb27SDimitry Andric   //=======================================
63206c3fb27SDimitry Andric 
63306c3fb27SDimitry Andric   // Now we emit the code for if the argument was originally passed in
63406c3fb27SDimitry Andric   // registers. First start the appropriate block:
63506c3fb27SDimitry Andric   CGF.EmitBlock(InRegBlock);
63606c3fb27SDimitry Andric 
63706c3fb27SDimitry Andric   llvm::Value *reg_top = nullptr;
63806c3fb27SDimitry Andric   Address reg_top_p =
63906c3fb27SDimitry Andric       CGF.Builder.CreateStructGEP(VAListAddr, reg_top_index, "reg_top_p");
64006c3fb27SDimitry Andric   reg_top = CGF.Builder.CreateLoad(reg_top_p, "reg_top");
64106c3fb27SDimitry Andric   Address BaseAddr(CGF.Builder.CreateInBoundsGEP(CGF.Int8Ty, reg_top, reg_offs),
64206c3fb27SDimitry Andric                    CGF.Int8Ty, CharUnits::fromQuantity(IsFPR ? 16 : 8));
64306c3fb27SDimitry Andric   Address RegAddr = Address::invalid();
64406c3fb27SDimitry Andric   llvm::Type *MemTy = CGF.ConvertTypeForMem(Ty), *ElementTy = MemTy;
64506c3fb27SDimitry Andric 
64606c3fb27SDimitry Andric   if (IsIndirect) {
64706c3fb27SDimitry Andric     // If it's been passed indirectly (actually a struct), whatever we find from
64806c3fb27SDimitry Andric     // stored registers or on the stack will actually be a struct **.
64906c3fb27SDimitry Andric     MemTy = llvm::PointerType::getUnqual(MemTy);
65006c3fb27SDimitry Andric   }
65106c3fb27SDimitry Andric 
65206c3fb27SDimitry Andric   const Type *Base = nullptr;
65306c3fb27SDimitry Andric   uint64_t NumMembers = 0;
65406c3fb27SDimitry Andric   bool IsHFA = isHomogeneousAggregate(Ty, Base, NumMembers);
65506c3fb27SDimitry Andric   if (IsHFA && NumMembers > 1) {
65606c3fb27SDimitry Andric     // Homogeneous aggregates passed in registers will have their elements split
65706c3fb27SDimitry Andric     // and stored 16-bytes apart regardless of size (they're notionally in qN,
65806c3fb27SDimitry Andric     // qN+1, ...). We reload and store into a temporary local variable
65906c3fb27SDimitry Andric     // contiguously.
66006c3fb27SDimitry Andric     assert(!IsIndirect && "Homogeneous aggregates should be passed directly");
66106c3fb27SDimitry Andric     auto BaseTyInfo = getContext().getTypeInfoInChars(QualType(Base, 0));
66206c3fb27SDimitry Andric     llvm::Type *BaseTy = CGF.ConvertType(QualType(Base, 0));
66306c3fb27SDimitry Andric     llvm::Type *HFATy = llvm::ArrayType::get(BaseTy, NumMembers);
66406c3fb27SDimitry Andric     Address Tmp = CGF.CreateTempAlloca(HFATy,
66506c3fb27SDimitry Andric                                        std::max(TyAlign, BaseTyInfo.Align));
66606c3fb27SDimitry Andric 
66706c3fb27SDimitry Andric     // On big-endian platforms, the value will be right-aligned in its slot.
66806c3fb27SDimitry Andric     int Offset = 0;
66906c3fb27SDimitry Andric     if (CGF.CGM.getDataLayout().isBigEndian() &&
67006c3fb27SDimitry Andric         BaseTyInfo.Width.getQuantity() < 16)
67106c3fb27SDimitry Andric       Offset = 16 - BaseTyInfo.Width.getQuantity();
67206c3fb27SDimitry Andric 
67306c3fb27SDimitry Andric     for (unsigned i = 0; i < NumMembers; ++i) {
67406c3fb27SDimitry Andric       CharUnits BaseOffset = CharUnits::fromQuantity(16 * i + Offset);
67506c3fb27SDimitry Andric       Address LoadAddr =
67606c3fb27SDimitry Andric         CGF.Builder.CreateConstInBoundsByteGEP(BaseAddr, BaseOffset);
67706c3fb27SDimitry Andric       LoadAddr = LoadAddr.withElementType(BaseTy);
67806c3fb27SDimitry Andric 
67906c3fb27SDimitry Andric       Address StoreAddr = CGF.Builder.CreateConstArrayGEP(Tmp, i);
68006c3fb27SDimitry Andric 
68106c3fb27SDimitry Andric       llvm::Value *Elem = CGF.Builder.CreateLoad(LoadAddr);
68206c3fb27SDimitry Andric       CGF.Builder.CreateStore(Elem, StoreAddr);
68306c3fb27SDimitry Andric     }
68406c3fb27SDimitry Andric 
68506c3fb27SDimitry Andric     RegAddr = Tmp.withElementType(MemTy);
68606c3fb27SDimitry Andric   } else {
68706c3fb27SDimitry Andric     // Otherwise the object is contiguous in memory.
68806c3fb27SDimitry Andric 
68906c3fb27SDimitry Andric     // It might be right-aligned in its slot.
69006c3fb27SDimitry Andric     CharUnits SlotSize = BaseAddr.getAlignment();
69106c3fb27SDimitry Andric     if (CGF.CGM.getDataLayout().isBigEndian() && !IsIndirect &&
69206c3fb27SDimitry Andric         (IsHFA || !isAggregateTypeForABI(Ty)) &&
69306c3fb27SDimitry Andric         TySize < SlotSize) {
69406c3fb27SDimitry Andric       CharUnits Offset = SlotSize - TySize;
69506c3fb27SDimitry Andric       BaseAddr = CGF.Builder.CreateConstInBoundsByteGEP(BaseAddr, Offset);
69606c3fb27SDimitry Andric     }
69706c3fb27SDimitry Andric 
69806c3fb27SDimitry Andric     RegAddr = BaseAddr.withElementType(MemTy);
69906c3fb27SDimitry Andric   }
70006c3fb27SDimitry Andric 
70106c3fb27SDimitry Andric   CGF.EmitBranch(ContBlock);
70206c3fb27SDimitry Andric 
70306c3fb27SDimitry Andric   //=======================================
70406c3fb27SDimitry Andric   // Argument was on the stack
70506c3fb27SDimitry Andric   //=======================================
70606c3fb27SDimitry Andric   CGF.EmitBlock(OnStackBlock);
70706c3fb27SDimitry Andric 
70806c3fb27SDimitry Andric   Address stack_p = CGF.Builder.CreateStructGEP(VAListAddr, 0, "stack_p");
70906c3fb27SDimitry Andric   llvm::Value *OnStackPtr = CGF.Builder.CreateLoad(stack_p, "stack");
71006c3fb27SDimitry Andric 
71106c3fb27SDimitry Andric   // Again, stack arguments may need realignment. In this case both integer and
71206c3fb27SDimitry Andric   // floating-point ones might be affected.
71306c3fb27SDimitry Andric   if (!IsIndirect && TyAlign.getQuantity() > 8) {
71406c3fb27SDimitry Andric     int Align = TyAlign.getQuantity();
71506c3fb27SDimitry Andric 
71606c3fb27SDimitry Andric     OnStackPtr = CGF.Builder.CreatePtrToInt(OnStackPtr, CGF.Int64Ty);
71706c3fb27SDimitry Andric 
71806c3fb27SDimitry Andric     OnStackPtr = CGF.Builder.CreateAdd(
71906c3fb27SDimitry Andric         OnStackPtr, llvm::ConstantInt::get(CGF.Int64Ty, Align - 1),
72006c3fb27SDimitry Andric         "align_stack");
72106c3fb27SDimitry Andric     OnStackPtr = CGF.Builder.CreateAnd(
72206c3fb27SDimitry Andric         OnStackPtr, llvm::ConstantInt::get(CGF.Int64Ty, -Align),
72306c3fb27SDimitry Andric         "align_stack");
72406c3fb27SDimitry Andric 
72506c3fb27SDimitry Andric     OnStackPtr = CGF.Builder.CreateIntToPtr(OnStackPtr, CGF.Int8PtrTy);
72606c3fb27SDimitry Andric   }
72706c3fb27SDimitry Andric   Address OnStackAddr = Address(OnStackPtr, CGF.Int8Ty,
72806c3fb27SDimitry Andric                                 std::max(CharUnits::fromQuantity(8), TyAlign));
72906c3fb27SDimitry Andric 
73006c3fb27SDimitry Andric   // All stack slots are multiples of 8 bytes.
73106c3fb27SDimitry Andric   CharUnits StackSlotSize = CharUnits::fromQuantity(8);
73206c3fb27SDimitry Andric   CharUnits StackSize;
73306c3fb27SDimitry Andric   if (IsIndirect)
73406c3fb27SDimitry Andric     StackSize = StackSlotSize;
73506c3fb27SDimitry Andric   else
73606c3fb27SDimitry Andric     StackSize = TySize.alignTo(StackSlotSize);
73706c3fb27SDimitry Andric 
73806c3fb27SDimitry Andric   llvm::Value *StackSizeC = CGF.Builder.getSize(StackSize);
73906c3fb27SDimitry Andric   llvm::Value *NewStack = CGF.Builder.CreateInBoundsGEP(
74006c3fb27SDimitry Andric       CGF.Int8Ty, OnStackPtr, StackSizeC, "new_stack");
74106c3fb27SDimitry Andric 
74206c3fb27SDimitry Andric   // Write the new value of __stack for the next call to va_arg
74306c3fb27SDimitry Andric   CGF.Builder.CreateStore(NewStack, stack_p);
74406c3fb27SDimitry Andric 
74506c3fb27SDimitry Andric   if (CGF.CGM.getDataLayout().isBigEndian() && !isAggregateTypeForABI(Ty) &&
74606c3fb27SDimitry Andric       TySize < StackSlotSize) {
74706c3fb27SDimitry Andric     CharUnits Offset = StackSlotSize - TySize;
74806c3fb27SDimitry Andric     OnStackAddr = CGF.Builder.CreateConstInBoundsByteGEP(OnStackAddr, Offset);
74906c3fb27SDimitry Andric   }
75006c3fb27SDimitry Andric 
75106c3fb27SDimitry Andric   OnStackAddr = OnStackAddr.withElementType(MemTy);
75206c3fb27SDimitry Andric 
75306c3fb27SDimitry Andric   CGF.EmitBranch(ContBlock);
75406c3fb27SDimitry Andric 
75506c3fb27SDimitry Andric   //=======================================
75606c3fb27SDimitry Andric   // Tidy up
75706c3fb27SDimitry Andric   //=======================================
75806c3fb27SDimitry Andric   CGF.EmitBlock(ContBlock);
75906c3fb27SDimitry Andric 
76006c3fb27SDimitry Andric   Address ResAddr = emitMergePHI(CGF, RegAddr, InRegBlock, OnStackAddr,
76106c3fb27SDimitry Andric                                  OnStackBlock, "vaargs.addr");
76206c3fb27SDimitry Andric 
76306c3fb27SDimitry Andric   if (IsIndirect)
76406c3fb27SDimitry Andric     return Address(CGF.Builder.CreateLoad(ResAddr, "vaarg.addr"), ElementTy,
76506c3fb27SDimitry Andric                    TyAlign);
76606c3fb27SDimitry Andric 
76706c3fb27SDimitry Andric   return ResAddr;
76806c3fb27SDimitry Andric }
76906c3fb27SDimitry Andric 
EmitDarwinVAArg(Address VAListAddr,QualType Ty,CodeGenFunction & CGF) const77006c3fb27SDimitry Andric Address AArch64ABIInfo::EmitDarwinVAArg(Address VAListAddr, QualType Ty,
77106c3fb27SDimitry Andric                                         CodeGenFunction &CGF) const {
77206c3fb27SDimitry Andric   // The backend's lowering doesn't support va_arg for aggregates or
77306c3fb27SDimitry Andric   // illegal vector types.  Lower VAArg here for these cases and use
77406c3fb27SDimitry Andric   // the LLVM va_arg instruction for everything else.
77506c3fb27SDimitry Andric   if (!isAggregateTypeForABI(Ty) && !isIllegalVectorType(Ty))
77606c3fb27SDimitry Andric     return EmitVAArgInstr(CGF, VAListAddr, Ty, ABIArgInfo::getDirect());
77706c3fb27SDimitry Andric 
77806c3fb27SDimitry Andric   uint64_t PointerSize = getTarget().getPointerWidth(LangAS::Default) / 8;
77906c3fb27SDimitry Andric   CharUnits SlotSize = CharUnits::fromQuantity(PointerSize);
78006c3fb27SDimitry Andric 
78106c3fb27SDimitry Andric   // Empty records are ignored for parameter passing purposes.
78206c3fb27SDimitry Andric   if (isEmptyRecord(getContext(), Ty, true))
78306c3fb27SDimitry Andric     return Address(CGF.Builder.CreateLoad(VAListAddr, "ap.cur"),
78406c3fb27SDimitry Andric                    CGF.ConvertTypeForMem(Ty), SlotSize);
78506c3fb27SDimitry Andric 
78606c3fb27SDimitry Andric   // The size of the actual thing passed, which might end up just
78706c3fb27SDimitry Andric   // being a pointer for indirect types.
78806c3fb27SDimitry Andric   auto TyInfo = getContext().getTypeInfoInChars(Ty);
78906c3fb27SDimitry Andric 
79006c3fb27SDimitry Andric   // Arguments bigger than 16 bytes which aren't homogeneous
79106c3fb27SDimitry Andric   // aggregates should be passed indirectly.
79206c3fb27SDimitry Andric   bool IsIndirect = false;
79306c3fb27SDimitry Andric   if (TyInfo.Width.getQuantity() > 16) {
79406c3fb27SDimitry Andric     const Type *Base = nullptr;
79506c3fb27SDimitry Andric     uint64_t Members = 0;
79606c3fb27SDimitry Andric     IsIndirect = !isHomogeneousAggregate(Ty, Base, Members);
79706c3fb27SDimitry Andric   }
79806c3fb27SDimitry Andric 
79906c3fb27SDimitry Andric   return emitVoidPtrVAArg(CGF, VAListAddr, Ty, IsIndirect,
80006c3fb27SDimitry Andric                           TyInfo, SlotSize, /*AllowHigherAlign*/ true);
80106c3fb27SDimitry Andric }
80206c3fb27SDimitry Andric 
EmitMSVAArg(CodeGenFunction & CGF,Address VAListAddr,QualType Ty) const80306c3fb27SDimitry Andric Address AArch64ABIInfo::EmitMSVAArg(CodeGenFunction &CGF, Address VAListAddr,
80406c3fb27SDimitry Andric                                     QualType Ty) const {
80506c3fb27SDimitry Andric   bool IsIndirect = false;
80606c3fb27SDimitry Andric 
80706c3fb27SDimitry Andric   // Composites larger than 16 bytes are passed by reference.
80806c3fb27SDimitry Andric   if (isAggregateTypeForABI(Ty) && getContext().getTypeSize(Ty) > 128)
80906c3fb27SDimitry Andric     IsIndirect = true;
81006c3fb27SDimitry Andric 
81106c3fb27SDimitry Andric   return emitVoidPtrVAArg(CGF, VAListAddr, Ty, IsIndirect,
81206c3fb27SDimitry Andric                           CGF.getContext().getTypeInfoInChars(Ty),
81306c3fb27SDimitry Andric                           CharUnits::fromQuantity(8),
81406c3fb27SDimitry Andric                           /*allowHigherAlign*/ false);
81506c3fb27SDimitry Andric }
81606c3fb27SDimitry Andric 
81706c3fb27SDimitry Andric std::unique_ptr<TargetCodeGenInfo>
createAArch64TargetCodeGenInfo(CodeGenModule & CGM,AArch64ABIKind Kind)81806c3fb27SDimitry Andric CodeGen::createAArch64TargetCodeGenInfo(CodeGenModule &CGM,
81906c3fb27SDimitry Andric                                         AArch64ABIKind Kind) {
82006c3fb27SDimitry Andric   return std::make_unique<AArch64TargetCodeGenInfo>(CGM.getTypes(), Kind);
82106c3fb27SDimitry Andric }
82206c3fb27SDimitry Andric 
82306c3fb27SDimitry Andric std::unique_ptr<TargetCodeGenInfo>
createWindowsAArch64TargetCodeGenInfo(CodeGenModule & CGM,AArch64ABIKind K)82406c3fb27SDimitry Andric CodeGen::createWindowsAArch64TargetCodeGenInfo(CodeGenModule &CGM,
82506c3fb27SDimitry Andric                                                AArch64ABIKind K) {
82606c3fb27SDimitry Andric   return std::make_unique<WindowsAArch64TargetCodeGenInfo>(CGM.getTypes(), K);
82706c3fb27SDimitry Andric }
828