39#include "llvm/ADT/ArrayRef.h"
40#include "llvm/ADT/ScopeExit.h"
41#include "llvm/Frontend/OpenMP/OMPIRBuilder.h"
42#include "llvm/IR/DataLayout.h"
43#include "llvm/IR/Dominators.h"
44#include "llvm/IR/FPEnv.h"
45#include "llvm/IR/Instruction.h"
46#include "llvm/IR/IntrinsicInst.h"
47#include "llvm/IR/Intrinsics.h"
48#include "llvm/IR/IntrinsicsPowerPC.h"
49#include "llvm/IR/MDBuilder.h"
50#include "llvm/Support/CRC.h"
51#include "llvm/Support/SipHash.h"
52#include "llvm/Support/xxhash.h"
53#include "llvm/Transforms/Scalar/LowerExpectIntrinsic.h"
54#include "llvm/Transforms/Utils/PromoteMemToReg.h"
64 if (CGOpts.DisableLifetimeMarkers)
68 if (CGOpts.SanitizeAddressUseAfterScope ||
75 return CGOpts.OptimizationLevel != 0;
78CodeGenFunction::CodeGenFunction(
CodeGenModule &cgm,
bool suppressNewContext)
82 DebugInfo(
CGM.getModuleDebugInfo()),
84 ShouldEmitLifetimeMarkers(
86 if (!suppressNewContext)
87 CGM.getCXXABI().getMangleContext().startNewFunction();
96 "missed to deactivate a cleanup");
99 CGM.getOpenMPRuntime().functionFinished(*
this);
106 if (
CGM.getLangOpts().OpenMPIRBuilder &&
CurFn)
107 CGM.getOpenMPRuntime().getOMPBuilder().finalize(
CurFn);
112llvm::fp::ExceptionBehavior
120 llvm_unreachable(
"Unsupported FP Exception Behavior");
125 llvm::FastMathFlags FMF;
126 FMF.setAllowReassoc(FPFeatures.getAllowFPReassociate());
127 FMF.setNoNaNs(FPFeatures.getNoHonorNaNs());
128 FMF.setNoInfs(FPFeatures.getNoHonorInfs());
129 FMF.setNoSignedZeros(FPFeatures.getNoSignedZero());
130 FMF.setAllowReciprocal(FPFeatures.getAllowReciprocal());
131 FMF.setApproxFunc(FPFeatures.getAllowApproxFunc());
145 ConstructorHelper(FPFeatures);
148void CodeGenFunction::CGFPOptionsRAII::ConstructorHelper(
FPOptions FPFeatures) {
149 OldFPFeatures = CGF.CurFPFeatures;
150 CGF.CurFPFeatures = FPFeatures;
152 OldExcept = CGF.Builder.getDefaultConstrainedExcept();
153 OldRounding = CGF.Builder.getDefaultConstrainedRounding();
155 if (OldFPFeatures == FPFeatures)
158 FMFGuard.emplace(CGF.Builder);
161 CGF.Builder.setDefaultConstrainedRounding(NewRoundingBehavior);
162 auto NewExceptionBehavior =
164 CGF.Builder.setDefaultConstrainedExcept(NewExceptionBehavior);
166 CGF.SetFastMathFlags(FPFeatures);
168 assert((CGF.CurFuncDecl ==
nullptr || CGF.Builder.getIsFPConstrained() ||
171 (NewExceptionBehavior == llvm::fp::ebIgnore &&
172 NewRoundingBehavior == llvm::RoundingMode::NearestTiesToEven)) &&
173 "FPConstrained should be enabled on entire function");
175 auto mergeFnAttrValue = [&](StringRef Name,
bool Value) {
177 CGF.CurFn->getFnAttribute(Name).getValueAsBool();
178 auto NewValue = OldValue &
Value;
179 if (OldValue != NewValue)
180 CGF.CurFn->addFnAttr(Name, llvm::toStringRef(NewValue));
182 mergeFnAttrValue(
"no-signed-zeros-fp-math", FPFeatures.getNoSignedZero());
186 CGF.CurFPFeatures = OldFPFeatures;
187 CGF.Builder.setDefaultConstrainedExcept(OldExcept);
188 CGF.Builder.setDefaultConstrainedRounding(OldRounding);
202 nullptr, IsKnownNonNull)
210 return ::makeNaturalAlignAddrLValue(
V, T,
false,
217 return ::makeNaturalAlignAddrLValue(
V, T,
true,
223 return ::makeNaturalAlignAddrLValue(
V, T,
false,
229 return ::makeNaturalAlignAddrLValue(
V, T,
true,
234 return CGM.getTypes().ConvertTypeForMem(T);
238 return CGM.getTypes().ConvertType(T);
242 llvm::Type *LLVMTy) {
243 return CGM.getTypes().convertTypeForLoadStore(ASTTy, LLVMTy);
249 switch (
type->getTypeClass()) {
250#define TYPE(name, parent)
251#define ABSTRACT_TYPE(name, parent)
252#define NON_CANONICAL_TYPE(name, parent) case Type::name:
253#define DEPENDENT_TYPE(name, parent) case Type::name:
254#define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(name, parent) case Type::name:
255#include "clang/AST/TypeNodes.inc"
256 llvm_unreachable(
"non-canonical or dependent type in IR-generation");
259 case Type::DeducedTemplateSpecialization:
260 llvm_unreachable(
"undeduced type in IR-generation");
265 case Type::BlockPointer:
266 case Type::LValueReference:
267 case Type::RValueReference:
268 case Type::MemberPointer:
270 case Type::ExtVector:
271 case Type::ConstantMatrix:
272 case Type::FunctionProto:
273 case Type::FunctionNoProto:
275 case Type::ObjCObjectPointer:
278 case Type::HLSLAttributedResource:
279 case Type::HLSLInlineSpirv:
280 case Type::OverflowBehavior:
288 case Type::ConstantArray:
289 case Type::IncompleteArray:
290 case Type::VariableArray:
292 case Type::ObjCObject:
293 case Type::ObjCInterface:
294 case Type::ArrayParameter:
302 llvm_unreachable(
"unknown type kind!");
309 llvm::BasicBlock *CurBB =
Builder.GetInsertBlock();
312 assert(!CurBB->getTerminator() &&
"Unexpected terminated block.");
316 if (CurBB->empty() ||
ReturnBlock.getBlock()->use_empty()) {
322 return llvm::DebugLoc();
330 dyn_cast<llvm::UncondBrInst>(*
ReturnBlock.getBlock()->user_begin());
331 if (BI && BI->getSuccessor(0) ==
ReturnBlock.getBlock()) {
334 llvm::DebugLoc Loc = BI->getDebugLoc();
335 Builder.SetInsertPoint(BI->getParent());
336 BI->eraseFromParent();
348 return llvm::DebugLoc();
353 if (!BB->use_empty()) {
361 assert(BreakContinueStack.empty() &&
362 "mismatched push/pop in break/continue stack!");
364 "mismatched push/pop of cleanups in EHStack!");
366 "mismatched activate/deactivate of cleanups!");
368 if (
CGM.shouldEmitConvergenceTokens()) {
371 "mismatched push/pop in convergence stack!");
374 bool OnlySimpleReturnStmts = NumSimpleReturnExprs > 0
375 && NumSimpleReturnExprs == NumReturnExprs
390 if (OnlySimpleReturnStmts)
391 DI->EmitLocation(
Builder, LastStopPoint);
393 DI->EmitLocation(
Builder, EndLoc);
401 bool HasOnlyNoopCleanups =
403 bool EmitRetDbgLoc = !HasCleanups || HasOnlyNoopCleanups;
405 std::optional<ApplyDebugLocation> OAL;
410 if (OnlySimpleReturnStmts)
411 DI->EmitLocation(
Builder, EndLoc);
425 if (
CGM.getCodeGenOpts().InstrumentFunctions)
426 CurFn->addFnAttr(
"instrument-function-exit",
"__cyg_profile_func_exit");
427 if (
CGM.getCodeGenOpts().InstrumentFunctionsAfterInlining)
428 CurFn->addFnAttr(
"instrument-function-exit-inlined",
429 "__cyg_profile_func_exit");
438 uint64_t RetKeyInstructionsAtomGroup = Loc ? Loc->getAtomGroup() : 0;
441 RetKeyInstructionsAtomGroup);
445 "did not remove all scopes from cleanup stack!");
449 if (IndirectBranch) {
456 if (!EscapedLocals.empty()) {
460 EscapeArgs.resize(EscapedLocals.size());
461 for (
auto &Pair : EscapedLocals)
462 EscapeArgs[Pair.second] = Pair.first;
463 llvm::Function *FrameEscapeFn = llvm::Intrinsic::getOrInsertDeclaration(
464 &
CGM.getModule(), llvm::Intrinsic::localescape);
471 Ptr->eraseFromParent();
475 if (PostAllocaInsertPt) {
476 llvm::Instruction *PostPtr = PostAllocaInsertPt;
477 PostAllocaInsertPt =
nullptr;
478 PostPtr->eraseFromParent();
483 if (IndirectBranch) {
485 if (PN->getNumIncomingValues() == 0) {
486 PN->replaceAllUsesWith(llvm::PoisonValue::get(PN->getType()));
487 PN->eraseFromParent();
496 for (
const auto &FuncletAndParent : TerminateFunclets)
499 if (
CGM.getCodeGenOpts().EmitDeclMetadata)
502 for (
const auto &R : DeferredReplacements) {
503 if (llvm::Value *Old = R.first) {
504 Old->replaceAllUsesWith(R.second);
508 DeferredReplacements.clear();
517 llvm::DominatorTree DT(*
CurFn);
518 llvm::PromoteMemToReg(
524 for (llvm::Argument &A :
CurFn->args())
525 if (
auto *VT = dyn_cast<llvm::VectorType>(A.getType()))
527 std::max((uint64_t)LargestVectorWidth,
528 VT->getPrimitiveSizeInBits().getKnownMinValue());
531 if (
auto *VT = dyn_cast<llvm::VectorType>(
CurFn->getReturnType()))
533 std::max((uint64_t)LargestVectorWidth,
534 VT->getPrimitiveSizeInBits().getKnownMinValue());
536 if (
CurFnInfo->getMaxVectorWidth() > LargestVectorWidth)
537 LargestVectorWidth =
CurFnInfo->getMaxVectorWidth();
547 CurFn->addFnAttr(
"min-legal-vector-width",
548 llvm::utostr(LargestVectorWidth));
557 dyn_cast<llvm::AllocaInst>(
ReturnValue.emitRawPointer(*
this));
558 if (RetAlloca && RetAlloca->use_empty()) {
559 RetAlloca->eraseFromParent();
568 if (!
CGM.getCodeGenOpts().InstrumentFunctions &&
569 !
CGM.getCodeGenOpts().InstrumentFunctionsAfterInlining &&
570 !
CGM.getCodeGenOpts().InstrumentFunctionEntryBare)
580 return CurFuncDecl->hasAttr<DisableSanitizerInstrumentationAttr>();
586 return CGM.getCodeGenOpts().XRayInstrumentFunctions;
592 return CGM.getCodeGenOpts().XRayInstrumentFunctions &&
593 (
CGM.getCodeGenOpts().XRayAlwaysEmitCustomEvents ||
594 CGM.getCodeGenOpts().XRayInstrumentationBundle.Mask ==
599 return CGM.getCodeGenOpts().XRayInstrumentFunctions &&
600 (
CGM.getCodeGenOpts().XRayAlwaysEmitTypedEvents ||
601 CGM.getCodeGenOpts().XRayInstrumentationBundle.Mask ==
612 llvm::raw_string_ostream Out(Mangled);
613 CGM.getCXXABI().getMangleContext().mangleCanonicalTypeName(Ty, Out,
false);
614 return llvm::ConstantInt::get(
615 CGM.Int32Ty,
static_cast<uint32_t
>(llvm::xxh3_64bits(Mangled)));
618void CodeGenFunction::EmitKernelMetadata(
const FunctionDecl *FD,
619 llvm::Function *Fn) {
620 if (!FD->
hasAttr<DeviceKernelAttr>() && !FD->
hasAttr<CUDAGlobalAttr>())
632 if (
const VecTypeHintAttr *A = FD->
getAttr<VecTypeHintAttr>()) {
633 QualType HintQTy = A->getTypeHint();
635 bool IsSignedInteger =
637 (HintEltQTy && HintEltQTy->
getElementType()->isSignedIntegerType());
638 llvm::Metadata *AttrMDArgs[] = {
639 llvm::ConstantAsMetadata::get(llvm::PoisonValue::get(
641 llvm::ConstantAsMetadata::get(llvm::ConstantInt::get(
642 llvm::IntegerType::get(Context, 32),
643 llvm::APInt(32, (uint64_t)(IsSignedInteger ? 1 : 0))))};
644 Fn->setMetadata(
"vec_type_hint", llvm::MDNode::get(Context, AttrMDArgs));
647 if (
const WorkGroupSizeHintAttr *A = FD->
getAttr<WorkGroupSizeHintAttr>()) {
648 auto Eval = [&](
Expr *E) {
649 return E->EvaluateKnownConstInt(FD->
getASTContext()).getExtValue();
651 llvm::Metadata *AttrMDArgs[] = {
652 llvm::ConstantAsMetadata::get(
Builder.getInt32(Eval(A->getXDim()))),
653 llvm::ConstantAsMetadata::get(
Builder.getInt32(Eval(A->getYDim()))),
654 llvm::ConstantAsMetadata::get(
Builder.getInt32(Eval(A->getZDim())))};
655 Fn->setMetadata(
"work_group_size_hint", llvm::MDNode::get(Context, AttrMDArgs));
658 if (
const ReqdWorkGroupSizeAttr *A = FD->
getAttr<ReqdWorkGroupSizeAttr>()) {
659 auto Eval = [&](Expr *E) {
660 return E->EvaluateKnownConstInt(FD->
getASTContext()).getExtValue();
662 llvm::Metadata *AttrMDArgs[] = {
663 llvm::ConstantAsMetadata::get(
Builder.getInt32(Eval(A->getXDim()))),
664 llvm::ConstantAsMetadata::get(
Builder.getInt32(Eval(A->getYDim()))),
665 llvm::ConstantAsMetadata::get(
Builder.getInt32(Eval(A->getZDim())))};
666 Fn->setMetadata(
"reqd_work_group_size", llvm::MDNode::get(Context, AttrMDArgs));
669 if (
const OpenCLIntelReqdSubGroupSizeAttr *A =
670 FD->
getAttr<OpenCLIntelReqdSubGroupSizeAttr>()) {
671 llvm::Metadata *AttrMDArgs[] = {
672 llvm::ConstantAsMetadata::get(
Builder.getInt32(A->getSubGroupSize()))};
673 Fn->setMetadata(
"intel_reqd_sub_group_size",
674 llvm::MDNode::get(Context, AttrMDArgs));
680 const Stmt *Body =
nullptr;
681 if (
auto *FD = dyn_cast_or_null<FunctionDecl>(F))
683 else if (
auto *OMD = dyn_cast_or_null<ObjCMethodDecl>(F))
684 Body = OMD->getBody();
686 if (
auto *CS = dyn_cast_or_null<CompoundStmt>(Body)) {
687 auto LastStmt = CS->body_rbegin();
688 if (LastStmt != CS->body_rend())
695 if (
SanOpts.has(SanitizerKind::Thread)) {
696 Fn->addFnAttr(
"sanitize_thread_no_checking_at_run_time");
697 Fn->removeFnAttr(llvm::Attribute::SanitizeThread);
702bool CodeGenFunction::requiresReturnValueCheck()
const {
703 return requiresReturnValueNullabilityCheck() ||
709 auto *MD = dyn_cast_or_null<CXXMethodDecl>(D);
710 if (!MD || !MD->getDeclName().getAsIdentifierInfo() ||
711 !MD->getDeclName().getAsIdentifierInfo()->isStr(
"allocate") ||
712 (MD->getNumParams() != 1 && MD->getNumParams() != 2))
718 if (MD->getNumParams() == 2) {
719 auto *PT = MD->parameters()[1]->getType()->getAs<
PointerType>();
720 if (!PT || !PT->isVoidPointerType() ||
721 !PT->getPointeeType().isConstQualified())
728bool CodeGenFunction::isInAllocaArgument(
CGCXXABI &ABI, QualType Ty) {
733bool CodeGenFunction::hasInAllocaArg(
const CXXMethodDecl *MD) {
736 llvm::any_of(MD->
parameters(), [&](ParmVarDecl *P) {
737 return isInAllocaArgument(CGM.getCXXABI(), P->getType());
744 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FD))
757 "Do not use a CodeGenFunction object for more than one function");
761 DidCallStackSave =
false;
763 const FunctionDecl *FD = dyn_cast_or_null<FunctionDecl>(D);
770 assert(
CurFn->isDeclaration() &&
"Function already has body?");
775#define SANITIZER(NAME, ID) \
776 if (SanOpts.empty()) \
778 if (SanOpts.has(SanitizerKind::ID)) \
779 if (CGM.isInNoSanitizeList(SanitizerKind::ID, Fn, Loc)) \
780 SanOpts.set(SanitizerKind::ID, false);
782#include "clang/Basic/Sanitizers.def"
787 const bool SanitizeBounds =
SanOpts.hasOneOf(SanitizerKind::Bounds);
789 bool NoSanitizeCoverage =
false;
792 no_sanitize_mask |=
Attr->getMask();
794 if (
Attr->hasCoverage())
795 NoSanitizeCoverage =
true;
799 SanOpts.Mask &= ~no_sanitize_mask;
800 if (no_sanitize_mask & SanitizerKind::Address)
801 SanOpts.set(SanitizerKind::KernelAddress,
false);
802 if (no_sanitize_mask & SanitizerKind::KernelAddress)
803 SanOpts.set(SanitizerKind::Address,
false);
804 if (no_sanitize_mask & SanitizerKind::HWAddress)
805 SanOpts.set(SanitizerKind::KernelHWAddress,
false);
806 if (no_sanitize_mask & SanitizerKind::KernelHWAddress)
807 SanOpts.set(SanitizerKind::HWAddress,
false);
809 if (SanitizeBounds && !
SanOpts.hasOneOf(SanitizerKind::Bounds))
810 Fn->addFnAttr(llvm::Attribute::NoSanitizeBounds);
812 if (NoSanitizeCoverage &&
CGM.getCodeGenOpts().hasSanitizeCoverage())
813 Fn->addFnAttr(llvm::Attribute::NoSanitizeCoverage);
816 if (
CGM.getCodeGenOpts().hasSanitizeBinaryMetadata()) {
817 if (no_sanitize_mask & SanitizerKind::Thread)
818 Fn->addFnAttr(
"no_sanitize_thread");
823 CurFn->addFnAttr(llvm::Attribute::DisableSanitizerInstrumentation);
826 if (
SanOpts.hasOneOf(SanitizerKind::Address | SanitizerKind::KernelAddress))
827 Fn->addFnAttr(llvm::Attribute::SanitizeAddress);
828 if (
SanOpts.hasOneOf(SanitizerKind::HWAddress |
829 SanitizerKind::KernelHWAddress))
830 Fn->addFnAttr(llvm::Attribute::SanitizeHWAddress);
831 if (
SanOpts.has(SanitizerKind::MemtagStack))
832 Fn->addFnAttr(llvm::Attribute::SanitizeMemTag);
833 if (
SanOpts.has(SanitizerKind::Thread))
834 Fn->addFnAttr(llvm::Attribute::SanitizeThread);
835 if (
SanOpts.has(SanitizerKind::Type))
836 Fn->addFnAttr(llvm::Attribute::SanitizeType);
837 if (
SanOpts.has(SanitizerKind::NumericalStability))
838 Fn->addFnAttr(llvm::Attribute::SanitizeNumericalStability);
839 if (
SanOpts.hasOneOf(SanitizerKind::Memory | SanitizerKind::KernelMemory))
840 Fn->addFnAttr(llvm::Attribute::SanitizeMemory);
841 if (
SanOpts.has(SanitizerKind::AllocToken))
842 Fn->addFnAttr(llvm::Attribute::SanitizeAllocToken);
844 if (
SanOpts.has(SanitizerKind::SafeStack))
845 Fn->addFnAttr(llvm::Attribute::SafeStack);
846 if (
SanOpts.has(SanitizerKind::ShadowCallStack))
847 Fn->addFnAttr(llvm::Attribute::ShadowCallStack);
849 if (
SanOpts.has(SanitizerKind::Realtime))
853 Fn->addFnAttr(llvm::Attribute::SanitizeRealtime);
855 Fn->addFnAttr(llvm::Attribute::SanitizeRealtimeBlocking);
859 if (
SanOpts.hasOneOf(SanitizerKind::Fuzzer | SanitizerKind::FuzzerNoLink))
860 Fn->addFnAttr(llvm::Attribute::OptForFuzzing);
864 if (
SanOpts.has(SanitizerKind::Thread)) {
865 if (
const auto *OMD = dyn_cast_or_null<ObjCMethodDecl>(D)) {
866 const IdentifierInfo *II = OMD->getSelector().getIdentifierInfoForSlot(0);
869 (OMD->getSelector().isUnarySelector() && II->
isStr(
".cxx_destruct"))) {
878 if (D &&
SanOpts.has(SanitizerKind::CFIUnrelatedCast)) {
886 if (D &&
SanOpts.has(SanitizerKind::Null))
892 bool AlwaysXRayAttr =
false;
893 if (
const auto *XRayAttr = D ? D->
getAttr<XRayInstrumentAttr>() :
nullptr) {
894 if (
CGM.getCodeGenOpts().XRayInstrumentationBundle.has(
896 CGM.getCodeGenOpts().XRayInstrumentationBundle.has(
899 Fn->addFnAttr(
"function-instrument",
"xray-always");
900 AlwaysXRayAttr =
true;
902 if (XRayAttr->neverXRayInstrument())
903 Fn->addFnAttr(
"function-instrument",
"xray-never");
904 if (
const auto *LogArgs = D->
getAttr<XRayLogArgsAttr>())
906 Fn->addFnAttr(
"xray-log-args",
907 llvm::utostr(LogArgs->getArgumentCount()));
912 "xray-instruction-threshold",
913 llvm::itostr(
CGM.getCodeGenOpts().XRayInstructionThreshold));
917 if (
CGM.getCodeGenOpts().XRayIgnoreLoops)
918 Fn->addFnAttr(
"xray-ignore-loops");
920 if (!
CGM.getCodeGenOpts().XRayInstrumentationBundle.has(
922 Fn->addFnAttr(
"xray-skip-exit");
924 if (!
CGM.getCodeGenOpts().XRayInstrumentationBundle.has(
926 Fn->addFnAttr(
"xray-skip-entry");
928 auto FuncGroups =
CGM.getCodeGenOpts().XRayTotalFunctionGroups;
929 if (FuncGroups > 1) {
931 CurFn->getName().bytes_end());
932 auto Group = crc32(FuncName) % FuncGroups;
933 if (Group !=
CGM.getCodeGenOpts().XRaySelectedFunctionGroup &&
935 Fn->addFnAttr(
"function-instrument",
"xray-never");
939 if (
CGM.getCodeGenOpts().getProfileInstr() !=
940 llvm::driver::ProfileInstrKind::ProfileNone) {
941 switch (
CGM.isFunctionBlockedFromProfileInstr(Fn, Loc)) {
943 Fn->addFnAttr(llvm::Attribute::SkipProfile);
946 Fn->addFnAttr(llvm::Attribute::NoProfile);
953 unsigned Count, Offset;
955 if (
const auto *
Attr =
956 D ? D->
getAttr<PatchableFunctionEntryAttr>() :
nullptr) {
957 Count =
Attr->getCount();
958 Offset =
Attr->getOffset();
959 Section =
Attr->getSection();
961 Count =
CGM.getCodeGenOpts().PatchableFunctionEntryCount;
962 Offset =
CGM.getCodeGenOpts().PatchableFunctionEntryOffset;
965 Section =
CGM.getCodeGenOpts().PatchableFunctionEntrySection;
966 if (Count && Offset <= Count) {
967 Fn->addFnAttr(
"patchable-function-entry", std::to_string(Count - Offset));
969 Fn->addFnAttr(
"patchable-function-prefix", std::to_string(Offset));
970 if (!Section.empty())
971 Fn->addFnAttr(
"patchable-function-entry-section", Section);
977 if (
CGM.getCodeGenOpts().HotPatch &&
980 llvm::Triple::CODE16)
981 Fn->addFnAttr(
"patchable-function",
"prologue-short-redirect");
984 if (
CGM.getCodeGenOpts().NoUseJumpTables)
985 Fn->addFnAttr(
"no-jump-tables",
"true");
988 if (
CGM.getCodeGenOpts().NoInlineLineTables)
989 Fn->addFnAttr(
"no-inline-line-tables");
992 if (
CGM.getCodeGenOpts().ProfileSampleAccurate)
993 Fn->addFnAttr(
"profile-sample-accurate");
995 if (!
CGM.getCodeGenOpts().SampleProfileFile.empty())
996 Fn->addFnAttr(
"use-sample-profile");
998 if (D && D->
hasAttr<CFICanonicalJumpTableAttr>())
999 Fn->addFnAttr(
"cfi-canonical-jump-table");
1001 if (D && D->
hasAttr<NoProfileFunctionAttr>())
1002 Fn->addFnAttr(llvm::Attribute::NoProfile);
1004 if (D && D->
hasAttr<HybridPatchableAttr>())
1005 Fn->addFnAttr(llvm::Attribute::HybridPatchable);
1009 if (
auto *A = D->
getAttr<FunctionReturnThunksAttr>()) {
1010 switch (A->getThunkType()) {
1011 case FunctionReturnThunksAttr::Kind::Keep:
1013 case FunctionReturnThunksAttr::Kind::Extern:
1014 Fn->addFnAttr(llvm::Attribute::FnRetThunkExtern);
1017 }
else if (
CGM.getCodeGenOpts().FunctionReturnThunks)
1018 Fn->addFnAttr(llvm::Attribute::FnRetThunkExtern);
1027 EmitKernelMetadata(FD, Fn);
1030 if (FD && FD->
hasAttr<ClspvLibclcBuiltinAttr>()) {
1031 Fn->setMetadata(
"clspv_libclc_builtin",
1037 if (FD &&
SanOpts.has(SanitizerKind::Function) &&
1040 llvm::LLVMContext &Ctx = Fn->getContext();
1041 llvm::MDBuilder MDB(Ctx);
1043 llvm::LLVMContext::MD_func_sanitize,
1044 MDB.createRTTIPointerPrologue(
1051 if (
SanOpts.has(SanitizerKind::NullabilityReturn)) {
1052 auto Nullability =
FnRetTy->getNullability();
1055 if (!(
SanOpts.has(SanitizerKind::ReturnsNonnullAttribute) &&
1057 RetValNullabilityPrecondition =
1080 Fn->addFnAttr(llvm::Attribute::NoRecurse);
1083 llvm::fp::ExceptionBehavior FPExceptionBehavior =
1085 Builder.setDefaultConstrainedRounding(RM);
1086 Builder.setDefaultConstrainedExcept(FPExceptionBehavior);
1088 (!FD && (FPExceptionBehavior != llvm::fp::ebIgnore ||
1089 RM != llvm::RoundingMode::NearestTiesToEven))) {
1090 Builder.setIsFPConstrained(
true);
1091 Fn->addFnAttr(llvm::Attribute::StrictFP);
1097 CGM.getCodeGenOpts().StackAlignment))
1098 Fn->addFnAttr(
"stackrealign");
1102 Fn->removeFnAttr(
"zero-call-used-regs");
1105 llvm::StringMap<bool> FeatureMap;
1110 if (T->getAArch64SMEAttributes() &
1117 std::optional<std::pair<unsigned, unsigned>> VScaleRange =
1121 CurFn->addFnAttr(llvm::Attribute::getWithVScaleRangeArgs(
1130 llvm::Value *Poison = llvm::PoisonValue::get(
Int32Ty);
1135 Builder.SetInsertPoint(EntryBB);
1139 if (requiresReturnValueCheck()) {
1150 DI->emitFunctionStart(GD, Loc, StartLoc,
1151 DI->getFunctionType(FD, RetTy, Args),
CurFn,
1156 if (
CGM.getCodeGenOpts().InstrumentFunctions)
1157 CurFn->addFnAttr(
"instrument-function-entry",
"__cyg_profile_func_enter");
1158 if (
CGM.getCodeGenOpts().InstrumentFunctionsAfterInlining)
1159 CurFn->addFnAttr(
"instrument-function-entry-inlined",
1160 "__cyg_profile_func_enter");
1161 if (
CGM.getCodeGenOpts().InstrumentFunctionEntryBare)
1162 CurFn->addFnAttr(
"instrument-function-entry-inlined",
1163 "__cyg_profile_func_enter_bare");
1170 if (
CGM.getCodeGenOpts().InstrumentForProfiling) {
1174 if (
CGM.getCodeGenOpts().CallFEntry)
1175 Fn->addFnAttr(
"fentry-call",
"true");
1177 Fn->addFnAttr(
"instrument-function-entry-inlined",
1180 if (
CGM.getCodeGenOpts().MNopMCount) {
1181 if (!
CGM.getCodeGenOpts().CallFEntry)
1182 CGM.getDiags().Report(diag::err_opt_not_valid_without_opt)
1183 <<
"-mnop-mcount" <<
"-mfentry";
1184 Fn->addFnAttr(
"mnop-mcount");
1187 if (
CGM.getCodeGenOpts().RecordMCount) {
1188 if (!
CGM.getCodeGenOpts().CallFEntry)
1189 CGM.getDiags().Report(diag::err_opt_not_valid_without_opt)
1190 <<
"-mrecord-mcount" <<
"-mfentry";
1191 Fn->addFnAttr(
"mrecord-mcount");
1196 if (
CGM.getCodeGenOpts().PackedStack) {
1198 llvm::Triple::systemz)
1199 CGM.getDiags().Report(diag::err_opt_not_valid_on_target)
1200 <<
"-mpacked-stack";
1201 Fn->addFnAttr(
"packed-stack");
1204 if (
CGM.getCodeGenOpts().WarnStackSize !=
UINT_MAX &&
1205 !
CGM.getDiags().isIgnored(diag::warn_fe_backend_frame_larger_than, Loc))
1206 Fn->addFnAttr(
"warn-stack-size",
1207 std::to_string(
CGM.getCodeGenOpts().WarnStackSize));
1219 auto AI =
CurFn->arg_begin();
1220 if (
CurFnInfo->getReturnInfo().isSRetAfterThis())
1223 &*AI, RetTy,
CurFnInfo->getReturnInfo().getIndirectAlign(),
false,
1225 if (!
CurFnInfo->getReturnInfo().getIndirectByVal()) {
1234 unsigned Idx =
CurFnInfo->getReturnInfo().getInAllocaFieldIndex();
1235 llvm::Function::arg_iterator EI =
CurFn->arg_end();
1267 if (FD->
hasAttr<HLSLShaderAttr>()) {
1268 CGM.getHLSLRuntime().emitEntryFunction(FD, Fn);
1274 if (
const CXXMethodDecl *MD = dyn_cast_if_present<CXXMethodDecl>(D);
1279 CGM.getCXXABI().EmitInstanceFunctionProlog(*
this);
1295 CXXThisValue = ThisFieldLValue.
getPointer(*
this);
1304 if (FD->hasCapturedVLAType()) {
1307 auto VAT = FD->getCapturedVLAType();
1308 VLASizeMap[VAT->getSizeExpr()] = ExprArg;
1315 CXXThisValue = CXXABIThisValue;
1319 if (CXXABIThisValue) {
1321 SkippedChecks.
set(SanitizerKind::ObjectSize,
true);
1328 SkippedChecks.
set(SanitizerKind::Null,
true);
1332 Loc, CXXABIThisValue, ThisTy, CXXABIThisAlignment, SkippedChecks);
1339 if (!FD || !FD->
hasAttr<NakedAttr>()) {
1340 for (
const VarDecl *VD : Args) {
1345 if (
const ParmVarDecl *PVD = dyn_cast<ParmVarDecl>(VD))
1346 Ty = PVD->getOriginalType();
1356 DI->EmitLocation(
Builder, StartLoc);
1360 if (
const auto *VecWidth =
CurFuncDecl->getAttr<MinVectorWidthAttr>())
1361 LargestVectorWidth = VecWidth->getVectorWidth();
1363 if (
CGM.shouldEmitConvergenceTokens())
1370 if (
const CompoundStmt *S = dyn_cast<CompoundStmt>(Body))
1382 llvm::BasicBlock *SkipCountBB =
nullptr;
1404 if (F->isInterposable())
return;
1406 for (llvm::BasicBlock &BB : *F)
1407 for (llvm::Instruction &I : BB)
1411 F->setDoesNotThrow();
1421 if (
CGM.getCXXABI().HasThisReturn(GD))
1423 else if (
CGM.getCXXABI().hasMostDerivedReturn(GD))
1424 ResTy =
CGM.getContext().VoidPtrTy;
1425 CGM.getCXXABI().buildThisParam(*
this, Args);
1431 bool PassedParams =
true;
1433 if (
auto Inherited = CD->getInheritedConstructor())
1439 Args.push_back(Param);
1440 if (!Param->hasAttr<PassObjectSizeAttr>())
1444 getContext(), Param->getDeclContext(), Param->getLocation(),
1452 CGM.getCXXABI().addImplicitStructorParams(*
this, ResTy, Args);
1459 assert(Fn &&
"generating code for null Function");
1466 CGM.getTargetCodeGenInfo().checkFunctionABI(
CGM, FD);
1472 std::string FDInlineName = (Fn->getName() +
".inline").str();
1473 llvm::Module *M = Fn->getParent();
1474 llvm::Function *Clone = M->getFunction(FDInlineName);
1476 Clone = llvm::Function::Create(Fn->getFunctionType(),
1477 llvm::GlobalValue::InternalLinkage,
1478 Fn->getAddressSpace(), FDInlineName, M);
1479 Clone->addFnAttr(llvm::Attribute::AlwaysInline);
1481 Fn->setLinkage(llvm::GlobalValue::ExternalLinkage);
1491 if (LLVM_UNLIKELY(PD->isInlineBuiltinDeclaration())) {
1492 std::string FDInlineName = (Fn->getName() +
".inline").str();
1493 llvm::Module *M = Fn->getParent();
1494 if (llvm::Function *Clone = M->getFunction(FDInlineName)) {
1495 Clone->replaceAllUsesWith(Fn);
1496 Clone->eraseFromParent();
1504 if (FD->
hasAttr<NoDebugAttr>()) {
1507 Fn->setSubprogram(
nullptr);
1509 DebugInfo =
nullptr;
1512 llvm::scope_exit Cleanup([
this] {
1514 DI->completeFunction();
1521 BodyRange = Body->getSourceRange();
1524 CurEHLocation = BodyRange.getEnd();
1536 if (SpecDecl->hasBody(SpecDecl))
1537 Loc = SpecDecl->getLocation();
1544 ShouldEmitLifetimeMarkers =
true;
1548 if (ShouldEmitLifetimeMarkers)
1553 StartFunction(GD, ResTy, Fn, FnInfo, Args, Loc, BodyRange.getBegin());
1556 if (Body && isa_and_nonnull<CoroutineBodyStmt>(Body))
1564 CurFn->addFnAttr(llvm::Attribute::MustProgress);
1567 PGO->assignRegionCounters(GD,
CurFn);
1574 FD->
hasAttr<CUDAGlobalAttr>())
1575 CGM.getCUDARuntime().emitDeviceStub(*
this, Args);
1598 }
else if (DeviceKernelAttr::isOpenCLSpelling(
1599 FD->
getAttr<DeviceKernelAttr>()) &&
1602 for (
unsigned i = 0; i < Args.size(); ++i) {
1604 QualType ArgQualType = Args[i]->getType();
1606 CallArgs.
add(ArgRValue, ArgQualType);
1610 CGM.getTargetCodeGenInfo().setOCLKernelStubCallingConvention(FT);
1612 CallArgs, FT,
false);
1613 llvm::FunctionType *FTy =
CGM.getTypes().GetFunctionType(FnInfo);
1614 llvm::Constant *GDStubFunctionPointer =
1615 CGM.getRawFunctionPointer(GDStub, FTy);
1622 llvm_unreachable(
"no definition for emitted function");
1632 bool ShouldEmitUnreachable =
1633 CGM.getCodeGenOpts().StrictReturn ||
1635 if (
SanOpts.has(SanitizerKind::Return)) {
1636 auto CheckOrdinal = SanitizerKind::SO_Return;
1637 auto CheckHandler = SanitizerHandler::MissingReturn;
1639 llvm::Value *IsFalse =
Builder.getFalse();
1640 EmitCheck(std::make_pair(IsFalse, CheckOrdinal), CheckHandler,
1642 }
else if (ShouldEmitUnreachable) {
1643 if (
CGM.getCodeGenOpts().OptimizationLevel == 0)
1646 if (
SanOpts.has(SanitizerKind::Return) || ShouldEmitUnreachable) {
1648 Builder.ClearInsertionPoint();
1655 PGO->verifyCounterMap();
1658 StringRef Identifier =
1659 CurCodeDecl->getAttr<PersonalityAttr>()->getRoutine()->getName();
1660 llvm::FunctionCallee PersonalityRoutine =
1661 CGM.CreateRuntimeFunction(llvm::FunctionType::get(
CGM.Int32Ty,
true),
1662 Identifier, {},
true);
1668 if (!
CurFn->doesNotThrow())
1677 if (!S)
return false;
1694 IgnoreCaseStmts =
true;
1709 if (!S)
return false;
1729 if (!S)
return false;
1760 if (!AllowLabels &&
CGM.getCodeGenOpts().hasProfileClangInstr() &&
1761 CGM.getCodeGenOpts().MCDCCoverage)
1764 llvm::APSInt ResultInt;
1768 ResultBool = ResultInt.getBoolValue();
1776 llvm::APSInt &ResultInt,
1784 llvm::APSInt Int =
Result.Val.getInt();
1788 PGO->markStmtMaybeUsed(
Cond);
1789 ResultInt = std::move(Int);
1818 llvm::BasicBlock *FalseBlock, uint64_t TrueCount ,
1821 bool InstrumentRegions =
CGM.getCodeGenOpts().hasProfileClangInstr();
1825 const Stmt *CntrStmt = (CntrIdx ? CntrIdx :
Cond);
1827 llvm::BasicBlock *ThenBlock =
nullptr;
1828 llvm::BasicBlock *ElseBlock =
nullptr;
1829 llvm::BasicBlock *NextBlock =
nullptr;
1834 llvm::BasicBlock *SkipIncrBlock =
1836 llvm::BasicBlock *SkipNextBlock =
nullptr;
1850 if (LOp == BO_LAnd) {
1851 SkipNextBlock = FalseBlock;
1852 ThenBlock = CounterIncrBlock;
1853 ElseBlock = (SkipIncrBlock ? SkipIncrBlock : SkipNextBlock);
1854 NextBlock = TrueBlock;
1869 else if (LOp == BO_LOr) {
1870 SkipNextBlock = TrueBlock;
1871 ThenBlock = (SkipIncrBlock ? SkipIncrBlock : SkipNextBlock);
1872 ElseBlock = CounterIncrBlock;
1873 NextBlock = FalseBlock;
1875 llvm_unreachable(
"Expected Opcode must be that of a Logical Operator");
1881 if (SkipIncrBlock) {
1905 const Expr *
Cond, llvm::BasicBlock *TrueBlock, llvm::BasicBlock *FalseBlock,
1907 const VarDecl *ConditionalDecl) {
1914 if (CondBOp->getOpcode() == BO_LAnd) {
1917 bool ConstantBool =
false;
1923 FalseBlock, TrueCount, LH);
1933 FalseBlock, TrueCount, LH, CondBOp);
1940 llvm::BasicBlock *LHSFalse =
1968 FalseBlock, TrueCount, LH);
1973 if (CondBOp->getOpcode() == BO_LOr) {
1976 bool ConstantBool =
false;
1982 FalseBlock, TrueCount, LH);
1992 FalseBlock, TrueCount, LH, CondBOp);
1997 llvm::BasicBlock *LHSTrue =
2005 uint64_t RHSCount = TrueCount - LHSCount;
2041 bool MCDCCondition =
CGM.getCodeGenOpts().hasProfileClangInstr() &&
2042 CGM.getCodeGenOpts().MCDCCoverage &&
2044 if (CondUOp->getOpcode() == UO_LNot && !MCDCCondition) {
2071 uint64_t LHSScaledTrueCount = 0;
2075 LHSScaledTrueCount = TrueCount * LHSRatio;
2084 LHSScaledTrueCount, LH, CondOp);
2092 TrueCount - LHSScaledTrueCount, LH, CondOp);
2127 MCDCBaseExpr = ConditionalOp;
2134 llvm::MDNode *Weights =
nullptr;
2135 llvm::MDNode *Unpredictable =
nullptr;
2140 auto *
Call = dyn_cast<CallExpr>(
Cond->IgnoreImpCasts());
2141 if (
Call &&
CGM.getCodeGenOpts().OptimizationLevel != 0) {
2142 auto *FD = dyn_cast_or_null<FunctionDecl>(
Call->getCalleeDecl());
2143 if (FD && FD->
getBuiltinID() == Builtin::BI__builtin_unpredictable) {
2145 Unpredictable = MDHelper.createUnpredictable();
2151 llvm::Value *NewCondV = emitCondLikelihoodViaExpectIntrinsic(CondV, LH);
2152 if (CondV != NewCondV)
2157 Weights = createProfileWeights(TrueCount, CurrentCount - TrueCount);
2160 llvm::Instruction *BrInst =
Builder.CreateCondBr(CondV, TrueBlock, FalseBlock,
2161 Weights, Unpredictable);
2165 case HLSLControlFlowHintAttr::Microsoft_branch:
2166 case HLSLControlFlowHintAttr::Microsoft_flatten: {
2167 llvm::MDBuilder MDHelper(
CGM.getLLVMContext());
2169 llvm::ConstantInt *BranchHintConstant =
2171 HLSLControlFlowHintAttr::Spelling::Microsoft_branch
2172 ? llvm::ConstantInt::get(
CGM.Int32Ty, 1)
2173 : llvm::ConstantInt::get(
CGM.Int32Ty, 2);
2176 {MDHelper.createString(
"hlsl.controlflow.hint"),
2177 MDHelper.createConstant(BranchHintConstant)});
2178 BrInst->setMetadata(
"hlsl.controlflow.hint",
2179 llvm::MDNode::get(
CGM.getLLVMContext(), Vals));
2183 case HLSLControlFlowHintAttr::SpellingNotCalculated:
2191 llvm::Value *Arg =
nullptr;
2192 if ((ICEArguments & (1 << Idx)) == 0) {
2197 std::optional<llvm::APSInt>
Result =
2199 assert(
Result &&
"Expected argument to be a constant");
2208 CGM.ErrorUnsupported(S,
Type);
2220 llvm::Value *sizeInChars) {
2224 llvm::Value *baseSizeInChars
2228 llvm::Value *end = Builder.CreateInBoundsGEP(begin.
getElementType(),
2230 sizeInChars,
"vla.end");
2232 llvm::BasicBlock *originBB = CGF.
Builder.GetInsertBlock();
2240 llvm::PHINode *cur = Builder.CreatePHI(begin.
getType(), 2,
"vla.cur");
2247 Builder.CreateMemCpy(
Address(cur, CGF.
Int8Ty, curAlign), src, baseSizeInChars,
2252 Builder.CreateInBoundsGEP(CGF.
Int8Ty, cur, baseSizeInChars,
"vla.next");
2255 llvm::Value *done = Builder.CreateICmpEQ(next, end,
"vla-init.isdone");
2256 Builder.CreateCondBr(done, contBB, loopBB);
2257 cur->addIncoming(next, loopBB);
2275 if (
CGM.getContext().arePFPFieldsTriviallyCopyable(Field->getParent())) {
2276 uint64_t FieldSignature =
2277 llvm::getPointerAuthStableSipHash(
CGM.getPFPFieldName(Field));
2278 Disc = llvm::ConstantInt::get(
CGM.Int64Ty, FieldSignature);
2282 llvm::GlobalValue *DS =
CGM.getPFPDeactivationSymbol(Field);
2283 llvm::OperandBundleDef DSBundle(
"deactivation-symbol", DS);
2286 Builder.CreateCall(
CGM.getIntrinsic(llvm::Intrinsic::protected_field_ptr,
2305 llvm::Value *SizeVal;
2312 dyn_cast_or_null<VariableArrayType>(
2315 SizeVal = VlaSize.NumElts;
2317 if (!eltSize.
isOne())
2318 SizeVal =
Builder.CreateNUWMul(SizeVal,
CGM.getSize(eltSize));
2324 SizeVal =
CGM.getSize(size);
2332 if (!
CGM.getTypes().isZeroInitializable(Ty)) {
2336 llvm::Constant *NullConstant =
CGM.EmitNullConstant(Ty);
2338 llvm::GlobalVariable *NullVariable =
2339 new llvm::GlobalVariable(
CGM.getModule(), NullConstant->getType(),
2341 llvm::GlobalVariable::PrivateLinkage,
2342 NullConstant, Twine());
2344 NullVariable->setAlignment(NullAlign.
getAsAlign());
2350 Builder.CreateMemCpy(DestPtr, SrcPtr, SizeVal,
false);
2356 Builder.CreateMemSet(DestPtr,
Builder.getInt8(0), SizeVal,
false);
2369 if (!IndirectBranch)
2375 IndirectBranch->addDestination(BB);
2376 return llvm::BlockAddress::get(
CurFn->getType(), BB);
2381 if (IndirectBranch)
return IndirectBranch->getParent();
2386 llvm::Value *DestVal = TmpBuilder.CreatePHI(
Int8PtrTy, 0,
2387 "indirect.goto.dest");
2390 IndirectBranch = TmpBuilder.CreateIndirectBr(DestVal);
2391 return IndirectBranch->getParent();
2403 llvm::Value *numVLAElements =
nullptr;
2415 baseType = elementType;
2416 return numVLAElements;
2430 llvm::ConstantInt *zero =
Builder.getInt32(0);
2431 gepIndices.push_back(zero);
2433 uint64_t countFromCLAs = 1;
2436 llvm::ArrayType *llvmArrayType =
2438 while (llvmArrayType) {
2441 llvmArrayType->getNumElements());
2443 gepIndices.push_back(zero);
2444 countFromCLAs *= llvmArrayType->getNumElements();
2448 dyn_cast<llvm::ArrayType>(llvmArrayType->getElementType());
2451 "LLVM and Clang types are out-of-synch");
2470 gepIndices,
"array.begin"),
2476 llvm::Value *numElements
2477 = llvm::ConstantInt::get(
SizeTy, countFromCLAs);
2481 numElements =
Builder.CreateNUWMul(numVLAElements, numElements);
2488 assert(vla &&
"type was not a variable array type!");
2495 llvm::Value *numElements =
nullptr;
2499 elementType =
type->getElementType();
2500 llvm::Value *vlaSize = VLASizeMap[
type->getSizeExpr()];
2501 assert(vlaSize &&
"no size for VLA!");
2502 assert(vlaSize->getType() ==
SizeTy);
2505 numElements = vlaSize;
2509 numElements =
Builder.CreateNUWMul(numElements, vlaSize);
2511 }
while ((
type =
getContext().getAsVariableArrayType(elementType)));
2513 return { numElements, elementType };
2519 assert(vla &&
"type was not a variable array type!");
2525 llvm::Value *VlaSize = VLASizeMap[Vla->
getSizeExpr()];
2526 assert(VlaSize &&
"no size for VLA!");
2527 assert(VlaSize->getType() ==
SizeTy);
2532 assert(
type->isVariablyModifiedType() &&
2533 "Must pass variably modified type to EmitVLASizes!");
2540 assert(
type->isVariablyModifiedType());
2542 const Type *ty =
type.getTypePtr();
2545#define TYPE(Class, Base)
2546#define ABSTRACT_TYPE(Class, Base)
2547#define NON_CANONICAL_TYPE(Class, Base)
2548#define DEPENDENT_TYPE(Class, Base) case Type::Class:
2549#define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(Class, Base)
2550#include "clang/AST/TypeNodes.inc"
2551 llvm_unreachable(
"unexpected dependent type!");
2557 case Type::ExtVector:
2558 case Type::ConstantMatrix:
2562 case Type::TemplateSpecialization:
2563 case Type::ObjCTypeParam:
2564 case Type::ObjCObject:
2565 case Type::ObjCInterface:
2566 case Type::ObjCObjectPointer:
2568 case Type::HLSLInlineSpirv:
2569 case Type::PredefinedSugar:
2570 llvm_unreachable(
"type class is never variably-modified!");
2572 case Type::Adjusted:
2584 case Type::BlockPointer:
2588 case Type::LValueReference:
2589 case Type::RValueReference:
2593 case Type::MemberPointer:
2597 case Type::ArrayParameter:
2598 case Type::ConstantArray:
2599 case Type::IncompleteArray:
2604 case Type::VariableArray: {
2613 llvm::Value *&entry = VLASizeMap[sizeExpr];
2621 if (
SanOpts.has(SanitizerKind::VLABound)) {
2622 auto CheckOrdinal = SanitizerKind::SO_VLABound;
2623 auto CheckHandler = SanitizerHandler::VLABoundNotPositive;
2625 llvm::Value *
Zero = llvm::Constant::getNullValue(size->getType());
2627 llvm::Value *CheckCondition =
2631 llvm::Constant *StaticArgs[] = {
2634 EmitCheck(std::make_pair(CheckCondition, CheckOrdinal),
2635 CheckHandler, StaticArgs, size);
2648 case Type::FunctionProto:
2649 case Type::FunctionNoProto:
2655 case Type::UnaryTransform:
2656 case Type::Attributed:
2657 case Type::BTFTagAttributed:
2658 case Type::OverflowBehavior:
2659 case Type::HLSLAttributedResource:
2660 case Type::SubstTemplateTypeParm:
2661 case Type::MacroQualified:
2662 case Type::CountAttributed:
2668 case Type::Decltype:
2670 case Type::DeducedTemplateSpecialization:
2671 case Type::PackIndexing:
2675 case Type::TypeOfExpr:
2688 }
while (
type->isVariablyModifiedType());
2692 if (
getContext().getBuiltinVaListType()->isArrayType())
2703 assert(
Init.hasValue() &&
"Invalid DeclRefExpr initializer!");
2705 if (
CGM.getCodeGenOpts().hasReducedDebugInfo())
2721 llvm::Instruction *inst =
new llvm::BitCastInst(value, value->getType(),
"",
2725 protection.Inst = inst;
2730 if (!protection.Inst)
return;
2733 protection.Inst->eraseFromParent();
2739 llvm::Value *Alignment,
2740 llvm::Value *OffsetValue) {
2741 if (Alignment->getType() !=
IntPtrTy)
2744 if (OffsetValue && OffsetValue->getType() !=
IntPtrTy)
2747 llvm::Value *TheCheck =
nullptr;
2748 if (
SanOpts.has(SanitizerKind::Alignment)) {
2749 llvm::Value *PtrIntValue =
2753 bool IsOffsetZero =
false;
2754 if (
const auto *CI = dyn_cast<llvm::ConstantInt>(OffsetValue))
2755 IsOffsetZero = CI->isZero();
2758 PtrIntValue =
Builder.CreateSub(PtrIntValue, OffsetValue,
"offsetptr");
2761 llvm::Value *
Zero = llvm::ConstantInt::get(
IntPtrTy, 0);
2764 llvm::Value *MaskedPtr =
Builder.CreateAnd(PtrIntValue, Mask,
"maskedptr");
2765 TheCheck =
Builder.CreateICmpEQ(MaskedPtr,
Zero,
"maskcond");
2767 llvm::Instruction *Assumption =
Builder.CreateAlignmentAssumption(
2768 CGM.getDataLayout(), PtrValue, Alignment, OffsetValue);
2770 if (!
SanOpts.has(SanitizerKind::Alignment))
2773 OffsetValue, TheCheck, Assumption);
2779 llvm::Value *Alignment,
2780 llvm::Value *OffsetValue) {
2789 llvm::Value *AnnotatedVal,
2790 StringRef AnnotationStr,
2792 const AnnotateAttr *
Attr) {
2795 CGM.EmitAnnotationString(AnnotationStr),
2796 CGM.EmitAnnotationUnit(Location),
2797 CGM.EmitAnnotationLineNo(Location),
2800 Args.push_back(
CGM.EmitAnnotationArgs(
Attr));
2801 return Builder.CreateCall(AnnotationFn, Args);
2805 assert(D->
hasAttr<AnnotateAttr>() &&
"no annotate attribute");
2808 {V->getType(), CGM.ConstGlobalsPtrTy}),
2814 assert(D->
hasAttr<AnnotateAttr>() &&
"no annotate attribute");
2815 llvm::Value *
V =
Addr.emitRawPointer(*
this);
2816 llvm::Type *VTy =
V->getType();
2817 auto *PTy = dyn_cast<llvm::PointerType>(VTy);
2818 unsigned AS = PTy ? PTy->getAddressSpace() : 0;
2819 llvm::PointerType *IntrinTy =
2820 llvm::PointerType::get(
CGM.getLLVMContext(), AS);
2821 llvm::Function *F =
CGM.getIntrinsic(llvm::Intrinsic::ptr_annotation,
2822 {IntrinTy,
CGM.ConstGlobalsPtrTy});
2828 if (VTy != IntrinTy)
2841 assert(!CGF->IsSanitizerScope);
2842 CGF->IsSanitizerScope =
true;
2846 CGF->IsSanitizerScope =
false;
2850 const llvm::Twine &Name,
2851 llvm::BasicBlock::iterator InsertPt)
const {
2854 I->setNoSanitizeMetadata();
2858 llvm::Instruction *I,
const llvm::Twine &Name,
2859 llvm::BasicBlock::iterator InsertPt)
const {
2860 llvm::IRBuilderDefaultInserter::InsertHelper(I, Name, InsertPt);
2862 CGF->InsertHelper(I, Name, InsertPt);
2871 if (
CGM.getContext().getTargetInfo().getTriple().isX86()) {
2873 if (BuiltinID == X86::BI__builtin_ia32_cmpps ||
2874 BuiltinID == X86::BI__builtin_ia32_cmpss ||
2875 BuiltinID == X86::BI__builtin_ia32_cmppd ||
2876 BuiltinID == X86::BI__builtin_ia32_cmpsd) {
2878 llvm::StringMap<bool> TargetFetureMap;
2879 CGM.getContext().getFunctionFeatureMap(TargetFetureMap, FD);
2881 *(E->
getArg(2)->getIntegerConstantExpr(
CGM.getContext()));
2882 if (
Result.getSExtValue() > 7 && !TargetFetureMap.lookup(
"avx"))
2883 CGM.getDiags().Report(E->
getBeginLoc(), diag::err_builtin_needs_feature)
2904 bool IsAlwaysInline = TargetDecl->
hasAttr<AlwaysInlineAttr>();
2905 bool IsFlatten = FD && FD->
hasAttr<FlattenAttr>();
2911 std::string MissingFeature;
2912 llvm::StringMap<bool> CallerFeatureMap;
2913 CGM.getContext().getFunctionFeatureMap(CallerFeatureMap, FD);
2920 StringRef FeatureList(
CGM.getContext().BuiltinInfo.getRequiredFeatures(BuiltinID));
2922 FeatureList, CallerFeatureMap) && !IsHipStdPar) {
2923 CGM.getDiags().Report(Loc, diag::err_builtin_needs_feature)
2928 TargetDecl->
hasAttr<TargetAttr>()) {
2931 const TargetAttr *TD = TargetDecl->
getAttr<TargetAttr>();
2933 CGM.getContext().filterFunctionTargetAttrs(TD);
2936 llvm::StringMap<bool> CalleeFeatureMap;
2937 CGM.getContext().getFunctionFeatureMap(CalleeFeatureMap, TargetDecl);
2940 if (F[0] ==
'+' && CalleeFeatureMap.lookup(F.substr(1)))
2941 ReqFeatures.push_back(StringRef(F).substr(1));
2944 for (
const auto &F : CalleeFeatureMap) {
2947 ReqFeatures.push_back(F.getKey());
2949 if (!llvm::all_of(ReqFeatures,
2951 if (!CallerFeatureMap.lookup(
Feature)) {
2952 MissingFeature =
Feature.str();
2959 CGM.getDiags().Report(Loc, diag::err_function_needs_feature)
2962 CGM.getDiags().Report(Loc, diag::err_flatten_function_needs_feature)
2967 llvm::StringMap<bool> CalleeFeatureMap;
2968 CGM.getContext().getFunctionFeatureMap(CalleeFeatureMap, TargetDecl);
2970 for (
const auto &F : CalleeFeatureMap) {
2972 (!CallerFeatureMap.lookup(F.getKey()) ||
2973 !CallerFeatureMap.find(F.getKey())->getValue()) &&
2976 CGM.getDiags().Report(Loc, diag::err_function_needs_feature)
2979 CGM.getDiags().Report(Loc, diag::err_flatten_function_needs_feature)
2987 if (!
CGM.getCodeGenOpts().SanitizeStats)
2990 llvm::IRBuilder<> IRB(
Builder.GetInsertBlock(),
Builder.GetInsertPoint());
2991 IRB.SetCurrentDebugLocation(
Builder.getCurrentDebugLocation());
2992 CGM.getSanStats().create(IRB, SSK);
3004 Salt = Info.CFISalt;
3006 Bundles.emplace_back(
"kcfi",
CGM.CreateKCFITypeId(FP->
desugar(), Salt));
3010CodeGenFunction::FormAArch64ResolverCondition(
const FMVResolverOption &RO) {
3011 return RO.Features.empty() ?
nullptr : EmitAArch64CpuSupports(RO.Features);
3015CodeGenFunction::FormX86ResolverCondition(
const FMVResolverOption &RO) {
3018 if (RO.Architecture) {
3019 StringRef
Arch = *RO.Architecture;
3022 if (
Arch.starts_with(
"x86-64"))
3028 if (!RO.Features.empty()) {
3029 llvm::Value *FeatureCond = EmitX86CpuSupports(RO.Features);
3037 llvm::Function *Resolver,
3039 llvm::Function *FuncToReturn,
3040 bool SupportsIFunc) {
3041 if (SupportsIFunc) {
3042 Builder.CreateRet(FuncToReturn);
3047 llvm::make_pointer_range(Resolver->args()));
3049 llvm::CallInst *Result = Builder.CreateCall(FuncToReturn, Args);
3050 Result->setTailCallKind(llvm::CallInst::TCK_MustTail);
3052 if (Resolver->getReturnType()->isVoidTy())
3053 Builder.CreateRetVoid();
3055 Builder.CreateRet(Result);
3061 llvm::Triple::ArchType ArchType =
3065 case llvm::Triple::x86:
3066 case llvm::Triple::x86_64:
3069 case llvm::Triple::aarch64:
3072 case llvm::Triple::riscv32:
3073 case llvm::Triple::riscv64:
3074 case llvm::Triple::riscv32be:
3075 case llvm::Triple::riscv64be:
3078 case llvm::Triple::ppc:
3079 case llvm::Triple::ppc64:
3087 "Only implemented for x86, AArch64, RISC-V, and PowerPC AIX");
3115 Builder.SetInsertPoint(CurBlock);
3117 if (!RO.Architecture && RO.Features.empty()) {
3120 assert(&RO == Options.end() - 1 &&
3121 "Default or Generic case must be last");
3122 Builder.CreateRet(RO.Function);
3131 assert(RO.Features.size() == 1 &&
3132 "for now one feature requirement per version");
3134 assert(RO.Features[0].starts_with(
"cpu="));
3135 StringRef CPU = RO.Features[0].split(
"=").second.trim();
3136 StringRef
Feature = llvm::StringSwitch<StringRef>(CPU)
3137 .Case(
"pwr7",
"arch_2_06")
3138 .Case(
"pwr8",
"arch_2_07")
3139 .Case(
"pwr9",
"arch_3_00")
3140 .Case(
"pwr10",
"arch_3_1")
3141 .Case(
"pwr11",
"arch_3_1")
3145 Builtin::BI__builtin_cpu_supports,
Builder.getInt1Ty(),
Feature);
3151 Builder.SetInsertPoint(ThenBlock);
3152 Builder.CreateRet(RO.Function);
3155 llvm_unreachable(
"Default case missing");
3162 llvm::Triple::OSType::Linux) {
3163 CGM.getDiags().Report(diag::err_os_unsupport_riscv_fmv);
3168 Builder.SetInsertPoint(CurBlock);
3172 bool HasDefault =
false;
3173 unsigned DefaultIndex = 0;
3176 for (
unsigned Index = 0; Index < Options.size(); Index++) {
3178 if (Options[Index].Features.empty()) {
3180 DefaultIndex = Index;
3184 Builder.SetInsertPoint(CurBlock);
3210 for (StringRef Feat : Options[Index].Features) {
3211 std::vector<std::string> FeatStr =
3214 assert(FeatStr.size() == 1 &&
"Feature string not delimited");
3216 std::string &CurrFeat = FeatStr.front();
3217 if (CurrFeat[0] ==
'+')
3218 TargetAttrFeats.push_back(CurrFeat.substr(1));
3221 if (TargetAttrFeats.empty())
3224 for (std::string &Feat : TargetAttrFeats)
3225 CurrTargetAttrFeats.push_back(Feat);
3227 Builder.SetInsertPoint(CurBlock);
3230 llvm::BasicBlock *RetBlock =
createBasicBlock(
"resolver_return", Resolver);
3233 Options[Index].
Function, SupportsIFunc);
3236 Builder.SetInsertPoint(CurBlock);
3237 Builder.CreateCondBr(FeatsCondition, RetBlock, ElseBlock);
3239 CurBlock = ElseBlock;
3244 Builder.SetInsertPoint(CurBlock);
3251 Builder.SetInsertPoint(CurBlock);
3252 llvm::CallInst *TrapCall =
EmitTrapCall(llvm::Intrinsic::trap);
3253 TrapCall->setDoesNotReturn();
3254 TrapCall->setDoesNotThrow();
3256 Builder.ClearInsertionPoint();
3261 assert(!Options.empty() &&
"No multiversion resolver options found");
3262 assert(Options.back().Features.size() == 0 &&
"Default case must be last");
3264 assert(SupportsIFunc &&
3265 "Multiversion resolver requires target IFUNC support");
3266 bool AArch64CpuInitialized =
false;
3270 Builder.SetInsertPoint(CurBlock);
3271 llvm::Value *
Condition = FormAArch64ResolverCondition(RO);
3280 if (!AArch64CpuInitialized) {
3281 Builder.SetInsertPoint(CurBlock, CurBlock->begin());
3282 EmitAArch64CpuInit();
3283 AArch64CpuInitialized =
true;
3284 Builder.SetInsertPoint(CurBlock);
3288 if (RO.Function ==
nullptr)
3291 llvm::BasicBlock *RetBlock =
createBasicBlock(
"resolver_return", Resolver);
3300 Builder.SetInsertPoint(CurBlock);
3301 llvm::CallInst *TrapCall =
EmitTrapCall(llvm::Intrinsic::trap);
3302 TrapCall->setDoesNotReturn();
3303 TrapCall->setDoesNotThrow();
3305 Builder.ClearInsertionPoint();
3315 Builder.SetInsertPoint(CurBlock);
3319 Builder.SetInsertPoint(CurBlock);
3320 llvm::Value *
Condition = FormX86ResolverCondition(RO);
3324 assert(&RO == Options.end() - 1 &&
3325 "Default or Generic case must be last");
3331 llvm::BasicBlock *RetBlock =
createBasicBlock(
"resolver_return", Resolver);
3340 Builder.SetInsertPoint(CurBlock);
3341 llvm::CallInst *TrapCall =
EmitTrapCall(llvm::Intrinsic::trap);
3342 TrapCall->setDoesNotReturn();
3343 TrapCall->setDoesNotThrow();
3345 Builder.ClearInsertionPoint();
3357 llvm::Value *OffsetValue, llvm::Value *TheCheck,
3358 llvm::Instruction *Assumption) {
3359 assert(isa_and_nonnull<llvm::CallInst>(Assumption) &&
3361 llvm::Intrinsic::getOrInsertDeclaration(
3362 Builder.GetInsertBlock()->getParent()->getParent(),
3363 llvm::Intrinsic::assume) &&
3364 "Assumption should be a call to llvm.assume().");
3365 assert(&(
Builder.GetInsertBlock()->back()) == Assumption &&
3366 "Assumption should be the last instruction of the basic block, "
3367 "since the basic block is still being generated.");
3369 if (!
SanOpts.has(SanitizerKind::Alignment))
3379 Assumption->removeFromParent();
3382 auto CheckOrdinal = SanitizerKind::SO_Alignment;
3383 auto CheckHandler = SanitizerHandler::AlignmentAssumption;
3387 OffsetValue =
Builder.getInt1(
false);
3392 llvm::Value *DynamicData[] = {Ptr, Alignment, OffsetValue};
3393 EmitCheck({std::make_pair(TheCheck, CheckOrdinal)}, CheckHandler,
3394 StaticData, DynamicData);
3405 return DI->SourceLocToDebugLoc(Location);
3407 return llvm::DebugLoc();
3411CodeGenFunction::emitCondLikelihoodViaExpectIntrinsic(llvm::Value *
Cond,
3422 llvm::Type *CondTy =
Cond->getType();
3423 assert(CondTy->isIntegerTy(1) &&
"expecting condition to be a boolean");
3424 llvm::Function *FnExpect =
3426 llvm::Value *ExpectedValueOfCond =
3428 return Builder.CreateCall(FnExpect, {
Cond, ExpectedValueOfCond},
3429 Cond->getName() +
".expval");
3431 llvm_unreachable(
"Unknown Likelihood");
3435 unsigned NumElementsDst,
3436 const llvm::Twine &Name) {
3438 unsigned NumElementsSrc = SrcTy->getNumElements();
3439 if (NumElementsSrc == NumElementsDst)
3442 std::vector<int> ShuffleMask(NumElementsDst, -1);
3443 for (
unsigned MaskIdx = 0;
3444 MaskIdx < std::min<>(NumElementsDst, NumElementsSrc); ++MaskIdx)
3445 ShuffleMask[MaskIdx] = MaskIdx;
3447 return Builder.CreateShuffleVector(SrcVec, ShuffleMask, Name);
3460 Discriminator =
Builder.getSize(0);
3462 llvm::Value *Args[] = {Key, Discriminator};
3463 Bundles.emplace_back(
"ptrauth", Args);
3469 unsigned IntrinsicID) {
3476 if (!Discriminator) {
3481 auto OrigType =
Pointer->getType();
3499 llvm::Intrinsic::ptrauth_sign);
3505 auto StripIntrinsic = CGF.
CGM.
getIntrinsic(llvm::Intrinsic::ptrauth_strip);
3509 auto OrigType =
Pointer->getType();
3526 llvm::Intrinsic::ptrauth_auth);
3530 llvm::Instruction *KeyInstruction, llvm::Value *Backup) {
3532 DI->addInstToCurrentSourceAtom(KeyInstruction, Backup);
3536 llvm::Instruction *KeyInstruction, llvm::Value *Backup, uint64_t Atom) {
3538 DI->addInstToSpecificSourceAtom(KeyInstruction, Backup, Atom);
3542 llvm::Value *Backup) {
3545 DI->addInstToCurrentSourceAtom(KeyInstruction, Backup);
3552 if (
getContext().arePFPFieldsTriviallyCopyable(Field.Field->getParent()))
3556 Builder.CreateStore(
Builder.CreateLoad(SrcFieldPtr), DestFieldPtr);
static void findPFPFields(const ASTContext &Ctx, QualType Ty, CharUnits Offset, std::vector< PFPField > &Fields, bool IncludeVBases)
Defines the clang::ASTContext interface.
This file provides some common utility functions for processing Lambda related AST Constructs.
Defines enum values for all the target-independent builtin functions.
static llvm::Value * EmitPointerAuthCommon(CodeGenFunction &CGF, const CGPointerAuthInfo &PointerAuth, llvm::Value *Pointer, unsigned IntrinsicID)
static void CreateMultiVersionResolverReturn(CodeGenModule &CGM, llvm::Function *Resolver, CGBuilderTy &Builder, llvm::Function *FuncToReturn, bool SupportsIFunc)
static llvm::Value * EmitStrip(CodeGenFunction &CGF, const CGPointerAuthInfo &PointerAuth, llvm::Value *Pointer)
static void emitNonZeroVLAInit(CodeGenFunction &CGF, QualType baseType, Address dest, Address src, llvm::Value *sizeInChars)
emitNonZeroVLAInit - Emit the "zero" initialization of a variable-length array whose elements have a ...
static void EmitIfUsed(CodeGenFunction &CGF, llvm::BasicBlock *BB)
static LValue makeNaturalAlignAddrLValue(llvm::Value *V, QualType T, bool ForPointeeType, bool MightBeSigned, CodeGenFunction &CGF, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
static void TryMarkNoThrow(llvm::Function *F)
Tries to mark the given function nounwind based on the non-existence of any throwing calls within it.
static llvm::Constant * getPrologueSignature(CodeGenModule &CGM, const FunctionDecl *FD)
Return the UBSan prologue signature for FD if one is available.
static bool endsWithReturn(const Decl *F)
Determine whether the function F ends with a return stmt.
static bool shouldEmitLifetimeMarkers(const CodeGenOptions &CGOpts, const LangOptions &LangOpts)
shouldEmitLifetimeMarkers - Decide whether we need emit the life-time markers.
static bool matchesStlAllocatorFn(const Decl *D, const ASTContext &Ctx)
Defines the C++ Decl subclasses, other than those for templates (found in DeclTemplate....
static StringRef getTriple(const Command &Job)
Defines the Objective-C statement AST node classes.
Enumerates target-specific builtins in their own namespaces within namespace clang.
APValue - This class implements a discriminated union of [uninitialized] [APSInt] [APFloat],...
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
QualType getFunctionTypeWithExceptionSpec(QualType Orig, const FunctionProtoType::ExceptionSpecInfo &ESI) const
Get a function type and produce the equivalent function type with the specified exception specificati...
QualType getBaseElementType(const ArrayType *VAT) const
Return the innermost element type of an array type.
bool hasAnyFunctionEffects() const
const ArrayType * getAsArrayType(QualType T) const
Type Query functions.
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
static bool hasSameType(QualType T1, QualType T2)
Determine whether the given types T1 and T2 are equivalent.
const VariableArrayType * getAsVariableArrayType(QualType T) const
QualType getSizeType() const
Return the unique type for "size_t" (C99 7.17), defined in <stddef.h>.
const TargetInfo & getTargetInfo() const
void getFunctionFeatureMap(llvm::StringMap< bool > &FeatureMap, const FunctionDecl *) const
Represents an array type, per C99 6.7.5.2 - Array Declarators.
QualType getElementType() const
Attr - This represents one attribute.
A builtin binary operation expression such as "x + y" or "x <= y".
static bool isLogicalOp(Opcode Opc)
BinaryOperatorKind Opcode
Represents a C++ constructor within a class.
Represents a static or instance method of a struct/union/class.
bool isImplicitObjectMemberFunction() const
[C++2b][dcl.fct]/p7 An implicit object member function is a non-static member function without an exp...
const CXXRecordDecl * getParent() const
Return the parent of this method declaration, which is the class in which this method is defined.
QualType getThisType() const
Return the type of the this pointer.
bool isLambda() const
Determine whether this class describes a lambda function object.
void getCaptureFields(llvm::DenseMap< const ValueDecl *, FieldDecl * > &Captures, FieldDecl *&ThisCapture) const
For a closure type, retrieve the mapping from captured variables and this to the non-static data memb...
bool isCapturelessLambda() const
bool isEmpty() const
Determine whether this is an empty class in the sense of (C++11 [meta.unary.prop]).
A C++ throw-expression (C++ [except.throw]).
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
Expr * getArg(unsigned Arg)
getArg - Return the specified argument.
SourceLocation getBeginLoc() const
CharUnits - This is an opaque type for sizes expressed in character units.
bool isZero() const
isZero - Test whether the quantity equals zero.
llvm::Align getAsAlign() const
getAsAlign - Returns Quantity as a valid llvm::Align, Beware llvm::Align assumes power of two 8-bit b...
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
CharUnits alignmentOfArrayElement(CharUnits elementSize) const
Given that this is the alignment of the first element of an array, return the minimum alignment of an...
bool isOne() const
isOne - Test whether the quantity equals one.
CodeGenOptions - Track various options which control how the code is optimized and passed to the back...
@ InAlloca
InAlloca - Pass the argument directly using the LLVM inalloca attribute.
@ Indirect
Indirect - Pass the argument indirectly via a hidden pointer with the specified alignment (0 indicate...
Like RawAddress, an abstract representation of an aligned address, but the pointer contained in this ...
llvm::Value * getBasePointer() const
llvm::Value * emitRawPointer(CodeGenFunction &CGF) const
Return the pointer contained in this class after authenticating it and adding offset to it if necessa...
CharUnits getAlignment() const
llvm::Type * getElementType() const
Return the type of the values stored in this address.
Address withElementType(llvm::Type *ElemTy) const
Return address with different element type, but same pointer and alignment.
llvm::PointerType * getType() const
Return the type of the pointer value.
A scoped helper to set the current source atom group for CGDebugInfo::addInstToCurrentSourceAtom.
A scoped helper to set the current debug location to the specified location or preferred location of ...
static ApplyDebugLocation CreateDefaultArtificial(CodeGenFunction &CGF, SourceLocation TemporaryLocation)
Apply TemporaryLocation if it is valid.
void InsertHelper(llvm::Instruction *I, const llvm::Twine &Name, llvm::BasicBlock::iterator InsertPt) const override
This forwards to CodeGenFunction::InsertHelper.
llvm::ConstantInt * getSize(CharUnits N)
@ RAA_DirectInMemory
Pass it on the stack using its defined layout.
virtual RecordArgABI getRecordArgABI(const CXXRecordDecl *RD) const =0
Returns how an argument of the given record type should be passed.
Abstract information about a function or function prototype.
const FunctionProtoType * getCalleeFunctionProtoType() const
All available information about a concrete callee.
static CGCallee forDirect(llvm::Constant *functionPtr, const CGCalleeInfo &abstractInfo=CGCalleeInfo())
This class gathers all debug information during compilation and is responsible for emitting to llvm g...
CGFunctionInfo - Class to encapsulate the information about a function definition.
bool isDelegateCall() const
llvm::Value * getDiscriminator() const
CallArgList - Type for representing both the value and type of arguments in a call.
void add(RValue rvalue, QualType type)
virtual ~CGCapturedStmtInfo()
CGFPOptionsRAII(CodeGenFunction &CGF, FPOptions FPFeatures)
An object to manage conditionally-evaluated expressions.
void begin(CodeGenFunction &CGF)
void end(CodeGenFunction &CGF)
An object which temporarily prevents a value from being destroyed by aggressive peephole optimization...
SanitizerScope(CodeGenFunction *CGF)
CodeGenFunction - This class organizes the per-function state that is used while generating LLVM code...
void EmitRISCVMultiVersionResolver(llvm::Function *Resolver, ArrayRef< FMVResolverOption > Options)
GlobalDecl CurGD
CurGD - The GlobalDecl for the current function being compiled.
void EmitBranchOnBoolExpr(const Expr *Cond, llvm::BasicBlock *TrueBlock, llvm::BasicBlock *FalseBlock, uint64_t TrueCount, Stmt::Likelihood LH=Stmt::LH_None, const Expr *ConditionalOp=nullptr, const VarDecl *ConditionalDecl=nullptr)
EmitBranchOnBoolExpr - Emit a branch on a boolean condition (e.g.
void setCurrentProfileCount(uint64_t Count)
Set the profiler's current count.
llvm::Value * emitBoolVecConversion(llvm::Value *SrcVec, unsigned NumElementsDst, const llvm::Twine &Name="")
void EmitAArch64MultiVersionResolver(llvm::Function *Resolver, ArrayRef< FMVResolverOption > Options)
JumpDest getJumpDestInCurrentScope(llvm::BasicBlock *Target)
The given basic block lies in the current EH scope, but may be a target of a potentially scope-crossi...
llvm::Value * EmitScalarOrConstFoldImmArg(unsigned ICEArguments, unsigned Idx, const CallExpr *E)
SanitizerSet SanOpts
Sanitizers enabled for this function.
@ UseSkipPath
Skip (false)
void EmitNullInitialization(Address DestPtr, QualType Ty)
EmitNullInitialization - Generate code to set a value of the given type to null, If the type contains...
void checkTargetFeatures(const CallExpr *E, const FunctionDecl *TargetDecl)
static bool ContainsLabel(const Stmt *S, bool IgnoreCaseStmts=false)
ContainsLabel - Return true if the statement contains a label in it.
void EmitPPCAIXMultiVersionResolver(llvm::Function *Resolver, ArrayRef< FMVResolverOption > Options)
define internal ptr @foo.resolver() { entry: is_version_1 = __builtin_cpu_supports(version_1) br i1 %...
bool ShouldSkipSanitizerInstrumentation()
ShouldSkipSanitizerInstrumentation - Return true if the current function should not be instrumented w...
llvm::Value * EmitPPCBuiltinCpu(unsigned BuiltinID, llvm::Type *ReturnType, StringRef CPUStr)
llvm::BlockAddress * GetAddrOfLabel(const LabelDecl *L)
llvm::Value * EmitRISCVCpuSupports(const CallExpr *E)
llvm::Value * EmitRISCVCpuInit()
static bool hasScalarEvaluationKind(QualType T)
llvm::Type * ConvertType(QualType T)
void GenerateCode(GlobalDecl GD, llvm::Function *Fn, const CGFunctionInfo &FnInfo)
void EmitSanitizerStatReport(llvm::SanitizerStatKind SSK)
void addInstToNewSourceAtom(llvm::Instruction *KeyInstruction, llvm::Value *Backup)
Add KeyInstruction and an optional Backup instruction to a new atom group (See ApplyAtomGroup for mor...
FieldDecl * LambdaThisCaptureField
PeepholeProtection protectFromPeepholes(RValue rvalue)
protectFromPeepholes - Protect a value that we're intending to store to the side, but which will prob...
void EmitLambdaStaticInvokeBody(const CXXMethodDecl *MD)
bool CurFuncIsThunk
In C++, whether we are code generating a thunk.
LValue MakeNaturalAlignPointeeAddrLValue(llvm::Value *V, QualType T)
Given a value of type T* that may not be to a complete object, construct an l-value with the natural ...
JumpDest getJumpDestForLabel(const LabelDecl *S)
getBasicBlockForLabel - Return the LLVM basicblock that the specified label maps to.
void EmitCXXThrowExpr(const CXXThrowExpr *E, bool KeepInsertionPoint=true)
SmallVector< llvm::ConvergenceControlInst *, 4 > ConvergenceTokenStack
Stack to track the controlled convergence tokens.
void unprotectFromPeepholes(PeepholeProtection protection)
RValue convertTempToRValue(Address addr, QualType type, SourceLocation Loc)
Given the address of a temporary variable, produce an r-value of its type.
llvm::Constant * EmitCheckSourceLocation(SourceLocation Loc)
Emit a description of a source location in a format suitable for passing to a runtime sanitizer handl...
llvm::SmallVector< DeferredDeactivateCleanup > DeferredDeactivationCleanupStack
llvm::BasicBlock * createBasicBlock(const Twine &name="", llvm::Function *parent=nullptr, llvm::BasicBlock *before=nullptr)
createBasicBlock - Create an LLVM basic block.
void addInstToCurrentSourceAtom(llvm::Instruction *KeyInstruction, llvm::Value *Backup)
See CGDebugInfo::addInstToCurrentSourceAtom.
const LangOptions & getLangOpts() const
void addInstToSpecificSourceAtom(llvm::Instruction *KeyInstruction, llvm::Value *Backup, uint64_t Atom)
See CGDebugInfo::addInstToSpecificSourceAtom.
LValue MakeNaturalAlignAddrLValue(llvm::Value *V, QualType T, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
void EmitVarAnnotations(const VarDecl *D, llvm::Value *V)
Emit local annotations for the local variable V, declared by D.
llvm::BasicBlock * EHResumeBlock
EHResumeBlock - Unified block containing a call to llvm.eh.resume.
Address EmitFieldAnnotations(const FieldDecl *D, Address V)
Emit field annotations for the given field & value.
void EmitConstructorBody(FunctionArgList &Args)
EmitConstructorBody - Emits the body of the current constructor.
void EmitKCFIOperandBundle(const CGCallee &Callee, SmallVectorImpl< llvm::OperandBundleDef > &Bundles)
void EmitDeclRefExprDbgValue(const DeclRefExpr *E, const APValue &Init)
Address makeNaturalAddressForPointer(llvm::Value *Ptr, QualType T, CharUnits Alignment=CharUnits::Zero(), bool ForPointeeType=false, LValueBaseInfo *BaseInfo=nullptr, TBAAAccessInfo *TBAAInfo=nullptr, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
Construct an address with the natural alignment of T.
LValue MakeNaturalAlignPointeeRawAddrLValue(llvm::Value *V, QualType T)
Same as MakeNaturalAlignPointeeAddrLValue except that the pointer is known to be unsigned.
@ TCK_ConstructorCall
Checking the 'this' pointer for a constructor call.
@ TCK_MemberCall
Checking the 'this' pointer for a call to a non-static member function.
const Decl * CurCodeDecl
CurCodeDecl - This is the inner-most code context, which includes blocks.
bool hasSkipCounter(const Stmt *S) const
llvm::AssertingVH< llvm::Instruction > AllocaInsertPt
AllocaInsertPoint - This is an instruction in the entry block before which we prefer to insert alloca...
void EmitFunctionBody(const Stmt *Body)
JumpDest ReturnBlock
ReturnBlock - Unified return block.
llvm::DebugLoc SourceLocToDebugLoc(SourceLocation Location)
Converts Location to a DebugLoc, if debug information is enabled.
llvm::Constant * EmitCheckTypeDescriptor(QualType T)
Emit a description of a type in a format suitable for passing to a runtime sanitizer handler.
llvm::DebugLoc EmitReturnBlock()
Emit the unified return block, trying to avoid its emission when possible.
RawAddress CreateDefaultAlignTempAlloca(llvm::Type *Ty, const Twine &Name="tmp")
CreateDefaultAlignedTempAlloca - This creates an alloca with the default ABI alignment of the given L...
const TargetInfo & getTarget() const
llvm::Value * EmitAnnotationCall(llvm::Function *AnnotationFn, llvm::Value *AnnotatedVal, StringRef AnnotationStr, SourceLocation Location, const AnnotateAttr *Attr)
Emit an annotation call (intrinsic).
Address EmitCompoundStmtWithoutScope(const CompoundStmt &S, bool GetLast=false, AggValueSlot AVS=AggValueSlot::ignored())
void PopCleanupBlocks(EHScopeStack::stable_iterator OldCleanupStackSize, std::initializer_list< llvm::Value ** > ValuesToReload={})
Takes the old cleanup stack size and emits the cleanup blocks that have been added.
void maybeCreateMCDCCondBitmap()
Allocate a temp value on the stack that MCDC can use to track condition results.
void EmitIgnoredExpr(const Expr *E)
EmitIgnoredExpr - Emit an expression in a context which ignores the result.
RValue EmitLoadOfLValue(LValue V, SourceLocation Loc)
EmitLoadOfLValue - Given an expression that represents a value lvalue, this method emits the address ...
static bool isInstrumentedCondition(const Expr *C)
isInstrumentedCondition - Determine whether the given condition is an instrumentable condition (i....
VlaSizePair getVLAElements1D(const VariableArrayType *vla)
Return the number of elements for a single dimension for the given array type.
RawAddress CreateIRTempWithoutCast(QualType T, const Twine &Name="tmp")
CreateIRTempWithoutCast - Create a temporary IR object of the given type, with appropriate alignment.
bool AlwaysEmitXRayCustomEvents() const
AlwaysEmitXRayCustomEvents - Return true if we must unconditionally emit XRay custom event handling c...
void StartFunction(GlobalDecl GD, QualType RetTy, llvm::Function *Fn, const CGFunctionInfo &FnInfo, const FunctionArgList &Args, SourceLocation Loc=SourceLocation(), SourceLocation StartLoc=SourceLocation())
Emit code for the start of a function.
llvm::Value * EmitPointerAuthSign(const CGPointerAuthInfo &Info, llvm::Value *Pointer)
void markAsIgnoreThreadCheckingAtRuntime(llvm::Function *Fn)
Annotate the function with an attribute that disables TSan checking at runtime.
llvm::Value * EvaluateExprAsBool(const Expr *E)
EvaluateExprAsBool - Perform the usual unary conversions on the specified expression and compare the ...
void EmitPointerAuthOperandBundle(const CGPointerAuthInfo &Info, SmallVectorImpl< llvm::OperandBundleDef > &Bundles)
void EmitCheck(ArrayRef< std::pair< llvm::Value *, SanitizerKind::SanitizerOrdinal > > Checked, SanitizerHandler Check, ArrayRef< llvm::Constant * > StaticArgs, ArrayRef< llvm::Value * > DynamicArgs, const TrapReason *TR=nullptr)
Create a basic block that will either trap or call a handler function in the UBSan runtime with the p...
void InsertHelper(llvm::Instruction *I, const llvm::Twine &Name, llvm::BasicBlock::iterator InsertPt) const
CGBuilder insert helper.
llvm::Value * emitArrayLength(const ArrayType *arrayType, QualType &baseType, Address &addr)
emitArrayLength - Compute the length of an array, even if it's a VLA, and drill down to the base elem...
bool HaveInsertPoint() const
HaveInsertPoint - True if an insertion point is defined.
bool AlwaysEmitXRayTypedEvents() const
AlwaysEmitXRayTypedEvents - Return true if clang must unconditionally emit XRay typed event handling ...
CGDebugInfo * getDebugInfo()
void EmitStartEHSpec(const Decl *D)
EmitStartEHSpec - Emit the start of the exception spec.
void EmitDestructorBody(FunctionArgList &Args)
EmitDestructorBody - Emits the body of the current destructor.
void EmitX86MultiVersionResolver(llvm::Function *Resolver, ArrayRef< FMVResolverOption > Options)
bool ShouldInstrumentFunction()
ShouldInstrumentFunction - Return true if the current function should be instrumented with __cyg_prof...
void maybeUpdateMCDCCondBitmap(const Expr *E, llvm::Value *Val)
Update the MCDC temp value with the condition's evaluated result.
void emitAlignmentAssumptionCheck(llvm::Value *Ptr, QualType Ty, SourceLocation Loc, SourceLocation AssumptionLoc, llvm::Value *Alignment, llvm::Value *OffsetValue, llvm::Value *TheCheck, llvm::Instruction *Assumption)
RValue EmitCall(const CGFunctionInfo &CallInfo, const CGCallee &Callee, ReturnValueSlot ReturnValue, const CallArgList &Args, llvm::CallBase **CallOrInvoke, bool IsMustTail, SourceLocation Loc, bool IsVirtualFunctionPointerThunk=false)
EmitCall - Generate a call of the given function, expecting the given result type,...
llvm::ConstantInt * getUBSanFunctionTypeHash(QualType T) const
Return a type hash constant for a function instrumented by -fsanitize=function.
void EmitBranchToCounterBlock(const Expr *Cond, BinaryOperator::Opcode LOp, llvm::BasicBlock *TrueBlock, llvm::BasicBlock *FalseBlock, uint64_t TrueCount=0, Stmt::Likelihood LH=Stmt::LH_None, const Expr *CntrIdx=nullptr)
EmitBranchToCounterBlock - Emit a conditional branch to a new block that increments a profile counter...
VlaSizePair getVLASize(const VariableArrayType *vla)
Returns an LLVM value that corresponds to the size, in non-variably-sized elements,...
ASTContext & getContext() const
bool isMCDCBranchExpr(const Expr *E) const
void EmitMultiVersionResolver(llvm::Function *Resolver, ArrayRef< FMVResolverOption > Options)
const Decl * CurFuncDecl
CurFuncDecl - Holds the Decl for the current outermost non-closure context.
static const Expr * stripCond(const Expr *C)
Ignore parentheses and logical-NOT to track conditions consistently.
void EmitFunctionProlog(const CGFunctionInfo &FI, llvm::Function *Fn, const FunctionArgList &Args)
EmitFunctionProlog - Emit the target specific LLVM code to load the arguments for the given function.
Address EmitAddressOfPFPField(Address RecordPtr, const PFPField &Field)
void SetFastMathFlags(FPOptions FPFeatures)
Set the codegen fast-math flags.
llvm::SmallVector< char, 256 > LifetimeExtendedCleanupStack
Address EmitVAListRef(const Expr *E)
void EmitLambdaInAllocaCallOpBody(const CXXMethodDecl *MD)
Address ReturnValuePointer
ReturnValuePointer - The temporary alloca to hold a pointer to sret.
static bool mightAddDeclToScope(const Stmt *S)
Determine if the given statement might introduce a declaration into the current scope,...
void EmitStmt(const Stmt *S, ArrayRef< const Attr * > Attrs={})
EmitStmt - Emit the code for the statement.
llvm::DenseMap< const ValueDecl *, FieldDecl * > LambdaCaptureFields
bool AutoreleaseResult
In ARC, whether we should autorelease the return value.
llvm::CallInst * EmitRuntimeCall(llvm::FunctionCallee callee, const Twine &name="")
uint64_t getCurrentProfileCount()
Get the profiler's current count.
llvm::Type * ConvertTypeForMem(QualType T)
void EmitEndEHSpec(const Decl *D)
EmitEndEHSpec - Emit the end of the exception spec.
LValue EmitLValueForLambdaField(const FieldDecl *Field)
void emitPFPPostCopyUpdates(Address DestPtr, Address SrcPtr, QualType Ty)
Copy all PFP fields from SrcPtr to DestPtr while updating signatures, assuming that DestPtr was alrea...
CodeGenTypes & getTypes() const
static TypeEvaluationKind getEvaluationKind(QualType T)
getEvaluationKind - Return the TypeEvaluationKind of QualType T.
bool IsSanitizerScope
True if CodeGen currently emits code implementing sanitizer checks.
static bool containsBreak(const Stmt *S)
containsBreak - Return true if the statement contains a break out of it.
void emitImplicitAssignmentOperatorBody(FunctionArgList &Args)
HLSLControlFlowHintAttr::Spelling HLSLControlFlowAttr
HLSL Branch attribute.
void EmitTypeCheck(TypeCheckKind TCK, SourceLocation Loc, LValue LV, QualType Type, SanitizerSet SkippedChecks=SanitizerSet(), llvm::Value *ArraySize=nullptr)
llvm::SmallVector< const ParmVarDecl *, 4 > FnArgs
Save Parameter Decl for coroutine.
const TargetInfo & Target
void EmitFunctionEpilog(const CGFunctionInfo &FI, bool EmitRetDbgLoc, SourceLocation EndLoc, uint64_t RetKeyInstructionsSourceAtom)
EmitFunctionEpilog - Emit the target specific LLVM code to return the given temporary.
Address EmitPointerWithAlignment(const Expr *Addr, LValueBaseInfo *BaseInfo=nullptr, TBAAAccessInfo *TBAAInfo=nullptr, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
EmitPointerWithAlignment - Given an expression with a pointer type, emit the value and compute our be...
void EmitBranch(llvm::BasicBlock *Block)
EmitBranch - Emit a branch to the specified basic block from the current insert block,...
RawAddress NormalCleanupDest
i32s containing the indexes of the cleanup destinations.
llvm::Type * convertTypeForLoadStore(QualType ASTTy, llvm::Type *LLVMTy=nullptr)
VarBypassDetector Bypasses
llvm::BasicBlock * GetIndirectGotoBlock()
EHScopeStack::stable_iterator PrologueCleanupDepth
PrologueCleanupDepth - The cleanup depth enclosing all the cleanups associated with the parameters.
Address EmitMSVAListRef(const Expr *E)
Emit a "reference" to a __builtin_ms_va_list; this is always the value of the expression,...
llvm::Value * EmitScalarExpr(const Expr *E, bool IgnoreResultAssign=false)
EmitScalarExpr - Emit the computation of the specified expression of LLVM scalar type,...
llvm::CallInst * EmitTrapCall(llvm::Intrinsic::ID IntrID)
Emit a call to trap or debugtrap and attach function attribute "trap-func-name" if specified.
LValue MakeAddrLValue(Address Addr, QualType T, AlignmentSource Source=AlignmentSource::Type)
void FinishFunction(SourceLocation EndLoc=SourceLocation())
FinishFunction - Complete IR generation of the current function.
const CGFunctionInfo * CurFnInfo
uint64_t getProfileCount(const Stmt *S)
Get the profiler's count for the given statement.
Address GetAddrOfLocalVar(const VarDecl *VD)
GetAddrOfLocalVar - Return the address of a local variable.
bool ConstantFoldsToSimpleInteger(const Expr *Cond, bool &Result, bool AllowLabels=false)
ConstantFoldsToSimpleInteger - If the specified expression does not fold to a constant,...
void ErrorUnsupported(const Stmt *S, const char *Type)
ErrorUnsupported - Print out an error that codegen doesn't support the specified stmt yet.
Address ReturnValue
ReturnValue - The temporary alloca to hold the return value.
LValue EmitLValue(const Expr *E, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
EmitLValue - Emit code to compute a designator that specifies the location of the expression.
bool ShouldXRayInstrumentFunction() const
ShouldXRayInstrument - Return true if the current function should be instrumented with XRay nop sleds...
void EnsureInsertPoint()
EnsureInsertPoint - Ensure that an insertion point is defined so that emitted IR has a place to go.
llvm::LLVMContext & getLLVMContext()
bool SawAsmBlock
Whether we processed a Microsoft-style asm block during CodeGen.
bool checkIfFunctionMustProgress()
Returns true if a function must make progress, which means the mustprogress attribute can be added.
void incrementProfileCounter(const Stmt *S, llvm::Value *StepV=nullptr)
Increment the profiler's counter for the given statement by StepV.
void emitAlignmentAssumption(llvm::Value *PtrValue, QualType Ty, SourceLocation Loc, SourceLocation AssumptionLoc, llvm::Value *Alignment, llvm::Value *OffsetValue=nullptr)
bool isMCDCDecisionExpr(const Expr *E) const
void EmitVariablyModifiedType(QualType Ty)
EmitVLASize - Capture all the sizes for the VLA expressions in the given variably-modified type and s...
void MaybeEmitDeferredVarDeclInit(const VarDecl *var)
void EmitBlockWithFallThrough(llvm::BasicBlock *BB, const Stmt *S)
When instrumenting to collect profile data, the counts for some blocks such as switch cases need to n...
void EmitBlock(llvm::BasicBlock *BB, bool IsFinished=false)
EmitBlock - Emit the given block.
LValue MakeNaturalAlignRawAddrLValue(llvm::Value *V, QualType T)
QualType BuildFunctionArgList(GlobalDecl GD, FunctionArgList &Args)
llvm::Value * EmitPointerAuthAuth(const CGPointerAuthInfo &Info, llvm::Value *Pointer)
This class organizes the cross-function state that is used while generating LLVM code.
CharUnits getNaturalTypeAlignment(QualType T, LValueBaseInfo *BaseInfo=nullptr, TBAAAccessInfo *TBAAInfo=nullptr, bool forPointeeType=false)
CodeGenTypes & getTypes()
const TargetCodeGenInfo & getTargetCodeGenInfo()
const CodeGenOptions & getCodeGenOpts() const
void GenKernelArgMetadata(llvm::Function *FN, const FunctionDecl *FD=nullptr, CodeGenFunction *CGF=nullptr)
OpenCL v1.2 s5.6.4.6 allows the compiler to store kernel argument information in the program executab...
llvm::Function * getIntrinsic(unsigned IID, ArrayRef< llvm::Type * > Tys={})
llvm::Type * ConvertType(QualType T)
ConvertType - Convert type T into a llvm::Type.
bool inheritingCtorHasParams(const InheritedConstructor &Inherited, CXXCtorType Type)
Determine if a C++ inheriting constructor should have parameters matching those of its inherited cons...
FunctionArgList - Type for representing both the decl and type of parameters to a function.
LValue - This represents an lvalue references.
llvm::Value * getPointer(CodeGenFunction &CGF) const
Address getAddress() const
RValue - This trivial value class is used to represent the result of an expression that is evaluated.
llvm::Value * getScalarVal() const
getScalarVal() - Return the Value* of this scalar value.
ReturnValueSlot - Contains the address where the return value of a function can be stored,...
virtual llvm::Constant * getUBSanFunctionSignature(CodeGen::CodeGenModule &CGM) const
Return a constant used by UBSan as a signature to identify functions possessing type information,...
CompoundStmt - This represents a group of statements like { stmt stmt }.
ConditionalOperator - The ?
A reference to a declared variable, function, enum, etc.
Decl - This represents one declaration (or definition), e.g.
ASTContext & getASTContext() const LLVM_READONLY
Decl * getNonClosureContext()
Find the innermost non-closure ancestor of this declaration, walking up through blocks,...
llvm::iterator_range< specific_attr_iterator< T > > specific_attrs() const
SourceLocation getLocation() const
This represents one expression.
FPOptions getFPFeaturesInEffect(const LangOptions &LO) const
Returns the set of floating point options that apply to this expression.
std::optional< llvm::APSInt > getIntegerConstantExpr(const ASTContext &Ctx) const
isIntegerConstantExpr - Return the value if this expression is a valid integer constant expression.
SourceLocation getExprLoc() const LLVM_READONLY
getExprLoc - Return the preferred location for the arrow when diagnosing a problem with a generic exp...
ExtVectorType - Extended vector type.
LangOptions::FPExceptionModeKind getExceptionMode() const
bool allowFPContractAcrossStatement() const
RoundingMode getRoundingMode() const
Represents a member of a struct/union/class.
Represents a function declaration or definition.
bool isMultiVersion() const
True if this function is considered a multiversioned function.
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
unsigned getBuiltinID(bool ConsiderWrapperFunctions=false) const
Returns a value indicating whether this function corresponds to a builtin function.
bool UsesFPIntrin() const
Determine whether the function was declared in source context that requires constrained FP intrinsics...
bool usesSEHTry() const
Indicates the function uses __try.
QualType getReturnType() const
ArrayRef< ParmVarDecl * > parameters() const
FunctionDecl * getTemplateInstantiationPattern(bool ForDefinition=true) const
Retrieve the function declaration from which this function could be instantiated, if it is an instant...
FunctionEffectsRef getFunctionEffects() const
bool isMSVCRTEntryPoint() const
Determines whether this function is a MSVCRT user defined entry point.
bool isInlineBuiltinDeclaration() const
Determine if this function provides an inline implementation of a builtin.
bool hasImplicitReturnZero() const
Whether falling off this function implicitly returns null/zero.
bool isMain() const
Determines whether this function is "main", which is the entry point into an executable program.
bool isDefaulted() const
Whether this function is defaulted.
OverloadedOperatorKind getOverloadedOperator() const
getOverloadedOperator - Which C++ overloaded operator this function represents, if any.
FunctionDecl * getPreviousDecl()
Return the previous declaration of this declaration or NULL if this is the first declaration.
Represents a prototype with parameter type info, e.g.
FunctionTypeExtraAttributeInfo getExtraAttributeInfo() const
Return the extra attribute information.
FunctionType - C99 6.7.5.3 - Function Declarators.
@ SME_PStateSMCompatibleMask
GlobalDecl - represents a global declaration.
CXXCtorType getCtorType() const
KernelReferenceKind getKernelReferenceKind() const
const Decl * getDecl() const
One of these records is kept for each identifier that is lexed.
bool isStr(const char(&Str)[StrLen]) const
Return true if this is the identifier for the specified string.
static ImplicitParamDecl * Create(ASTContext &C, DeclContext *DC, SourceLocation IdLoc, IdentifierInfo *Id, QualType T, ImplicitParamKind ParamKind)
Create implicit parameter.
Represents the declaration of a label.
FPExceptionModeKind
Possible floating point exception behavior.
@ FPE_Strict
Strictly preserve the floating-point exception semantics.
@ FPE_MayTrap
Transformations do not cause new exceptions but may hide some.
@ FPE_Ignore
Assume that floating-point exceptions are masked.
Keeps track of the various options that can be enabled, which controls the dialect of C or C++ that i...
SanitizerSet Sanitize
Set of enabled sanitizers.
RoundingMode getDefaultRoundingMode() const
DeclarationName getDeclName() const
Get the actual, stored name of the declaration, which may be a special name.
Represents a parameter to a function.
ParsedAttr - Represents a syntactic attribute.
PointerType - C99 6.7.5.1 - Pointer Declarators.
@ Forbid
Profiling is forbidden using the noprofile attribute.
@ Skip
Profiling is skipped using the skipprofile attribute.
@ Allow
Profiling is allowed.
A (possibly-)qualified type.
bool isVolatileQualified() const
Determine whether this type is volatile-qualified.
field_range fields() const
Encodes a location in the source.
A trivial tuple used to represent a source range.
Stmt - This represents one statement.
StmtClass getStmtClass() const
Likelihood
The likelihood of a branch being taken.
@ LH_Unlikely
Branch has the [[unlikely]] attribute.
@ LH_None
No attribute set or branches of the IfStmt have the same attribute.
@ LH_Likely
Branch has the [[likely]] attribute.
bool isMicrosoft() const
Is this ABI an MSVC-compatible ABI?
const llvm::Triple & getTriple() const
Returns the target triple of the primary target.
virtual std::optional< std::pair< unsigned, unsigned > > getVScaleRange(const LangOptions &LangOpts, ArmStreamingKind Mode, llvm::StringMap< bool > *FeatureMap=nullptr) const
Returns target-specific min and max values VScale_Range.
bool supportsIFunc() const
Identify whether this target supports IFuncs.
TargetCXXABI getCXXABI() const
Get the C++ ABI currently in use.
virtual ParsedTargetAttr parseTargetAttr(StringRef Str) const
bool isSignedIntegerType() const
Return true if this is an integer type that is signed, according to C99 6.2.5p4 [char,...
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool isVariablyModifiedType() const
Whether this type is a variably-modified type (C99 6.7.5).
TypeClass getTypeClass() const
const T * getAs() const
Member-template getAs<specific type>'.
bool isObjCRetainableType() const
bool isFunctionNoProtoType() const
bool isCFIUncheckedCalleeFunctionType() const
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
Represents a variable declaration or definition.
Represents a C array with a specified size that is not an integer-constant-expression.
Expr * getSizeExpr() const
QualType getElementType() const
Defines the clang::TargetInfo interface.
bool evaluateRequiredTargetFeatures(llvm::StringRef RequiredFatures, const llvm::StringMap< bool > &TargetFetureMap)
Returns true if the required target features of a builtin function are enabled.
@ Type
The l-value was considered opaque, so the alignment was determined from a type.
@ Decl
The l-value was an access to a declared entity or something equivalently strong, like the address of ...
TypeEvaluationKind
The kind of evaluation to perform on values of a particular type.
CGBuilderInserter CGBuilderInserterTy
constexpr XRayInstrMask Typed
constexpr XRayInstrMask FunctionExit
constexpr XRayInstrMask FunctionEntry
constexpr XRayInstrMask Custom
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const AstTypeMatcher< ArrayType > arrayType
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
Expr * IgnoreBuiltinExpectSingleStep(Expr *E)
@ NonNull
Values of this type can never be null.
Expr * IgnoreExprNodes(Expr *E, FnTys &&... Fns)
Given an expression E and functions Fn_1,...,Fn_n : Expr * -> Expr *, Recursively apply each of the f...
nullptr
This class represents a compute construct, representing a 'Kind' of ‘parallel’, 'serial',...
bool isLambdaCallOperator(const CXXMethodDecl *MD)
@ Result
The result type of a method or function.
Expr * IgnoreImplicitCastsSingleStep(Expr *E)
Expr * IgnoreUOpLNotSingleStep(Expr *E)
Expr * IgnoreParensSingleStep(Expr *E)
llvm::fp::ExceptionBehavior ToConstrainedExceptMD(LangOptions::FPExceptionModeKind Kind)
U cast(CodeGen::Address addr)
bool IsArmStreamingFunction(const FunctionDecl *FD, bool IncludeLocallyStreaming)
Returns whether the given FunctionDecl has an __arm[_locally]_streaming attribute.
@ Other
Other implicit parameter.
@ EST_None
no exception specification
@ Implicit
An implicit conversion.
A jump destination is an abstract label, branching to which may require a jump out through normal cle...
llvm::BasicBlock * getBlock() const
This structure provides a set of types that are commonly used during IR emission.
llvm::PointerType * VoidPtrTy
llvm::IntegerType * Int8Ty
i8, i16, i32, and i64
llvm::IntegerType * SizeTy
llvm::IntegerType * Int32Ty
llvm::IntegerType * IntPtrTy
llvm::PointerType * Int8PtrTy
CharUnits getPointerAlign() const
EvalResult is a struct with detailed info about an evaluated expression.
A FunctionEffect plus a potential boolean expression determining whether the effect is declared (e....
Contains information gathered from parsing the contents of TargetAttr.
std::vector< std::string > Features
void set(SanitizerMask K, bool Value)
Enable or disable a certain (single) sanitizer.
bool has(SanitizerMask K) const
Check if a certain (single) sanitizer is enabled.