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