38#include "llvm/ADT/ArrayRef.h"
39#include "llvm/Frontend/OpenMP/OMPIRBuilder.h"
40#include "llvm/IR/DataLayout.h"
41#include "llvm/IR/Dominators.h"
42#include "llvm/IR/FPEnv.h"
43#include "llvm/IR/IntrinsicInst.h"
44#include "llvm/IR/Intrinsics.h"
45#include "llvm/IR/MDBuilder.h"
46#include "llvm/IR/Operator.h"
47#include "llvm/Support/CRC.h"
48#include "llvm/Support/xxhash.h"
49#include "llvm/Transforms/Scalar/LowerExpectIntrinsic.h"
50#include "llvm/Transforms/Utils/PromoteMemToReg.h"
54using namespace CodeGen;
64 if (CGOpts.DisableLifetimeMarkers)
68 if (CGOpts.SanitizeAddressUseAfterScope ||
74 return CGOpts.OptimizationLevel != 0;
77CodeGenFunction::CodeGenFunction(
CodeGenModule &cgm,
bool suppressNewContext)
79 Builder(cgm, cgm.getModule().getContext(),
llvm::ConstantFolder(),
81 SanOpts(CGM.getLangOpts().Sanitize), CurFPFeatures(CGM.getLangOpts()),
82 DebugInfo(CGM.getModuleDebugInfo()), PGO(cgm),
83 ShouldEmitLifetimeMarkers(
85 if (!suppressNewContext)
86 CGM.getCXXABI().getMangleContext().startNewFunction();
89 SetFastMathFlags(CurFPFeatures);
92CodeGenFunction::~CodeGenFunction() {
95 "missed to deactivate a cleanup");
111llvm::fp::ExceptionBehavior
119 llvm_unreachable(
"Unsupported FP Exception Behavior");
124 llvm::FastMathFlags FMF;
125 FMF.setAllowReassoc(FPFeatures.getAllowFPReassociate());
126 FMF.setNoNaNs(FPFeatures.getNoHonorNaNs());
127 FMF.setNoInfs(FPFeatures.getNoHonorInfs());
128 FMF.setNoSignedZeros(FPFeatures.getNoSignedZero());
129 FMF.setAllowReciprocal(FPFeatures.getAllowReciprocal());
130 FMF.setApproxFunc(FPFeatures.getAllowApproxFunc());
144 ConstructorHelper(FPFeatures);
147void CodeGenFunction::CGFPOptionsRAII::ConstructorHelper(
FPOptions FPFeatures) {
148 OldFPFeatures = CGF.CurFPFeatures;
149 CGF.CurFPFeatures = FPFeatures;
151 OldExcept = CGF.Builder.getDefaultConstrainedExcept();
152 OldRounding = CGF.Builder.getDefaultConstrainedRounding();
154 if (OldFPFeatures == FPFeatures)
157 FMFGuard.emplace(CGF.Builder);
160 CGF.Builder.setDefaultConstrainedRounding(NewRoundingBehavior);
161 auto NewExceptionBehavior =
164 CGF.Builder.setDefaultConstrainedExcept(NewExceptionBehavior);
166 CGF.SetFastMathFlags(FPFeatures);
168 assert((CGF.CurFuncDecl ==
nullptr || CGF.Builder.getIsFPConstrained() ||
169 isa<CXXConstructorDecl>(CGF.CurFuncDecl) ||
170 isa<CXXDestructorDecl>(CGF.CurFuncDecl) ||
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-infs-fp-math", FPFeatures.getNoHonorInfs());
183 mergeFnAttrValue(
"no-nans-fp-math", FPFeatures.getNoHonorNaNs());
184 mergeFnAttrValue(
"no-signed-zeros-fp-math", FPFeatures.getNoSignedZero());
187 FPFeatures.getAllowFPReassociate() && FPFeatures.getAllowReciprocal() &&
188 FPFeatures.getAllowApproxFunc() && FPFeatures.getNoSignedZero() &&
193 CGF.CurFPFeatures = OldFPFeatures;
194 CGF.Builder.setDefaultConstrainedExcept(OldExcept);
195 CGF.Builder.setDefaultConstrainedRounding(OldRounding);
209 nullptr, IsKnownNonNull)
217 return ::makeNaturalAlignAddrLValue(
V,
T,
false,
224 return ::makeNaturalAlignAddrLValue(
V,
T,
true,
230 return ::makeNaturalAlignAddrLValue(
V,
T,
false,
236 return ::makeNaturalAlignAddrLValue(
V,
T,
true,
249 llvm::Type *LLVMTy) {
256 switch (
type->getTypeClass()) {
257#define TYPE(name, parent)
258#define ABSTRACT_TYPE(name, parent)
259#define NON_CANONICAL_TYPE(name, parent) case Type::name:
260#define DEPENDENT_TYPE(name, parent) case Type::name:
261#define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(name, parent) case Type::name:
262#include "clang/AST/TypeNodes.inc"
263 llvm_unreachable(
"non-canonical or dependent type in IR-generation");
266 case Type::DeducedTemplateSpecialization:
267 llvm_unreachable(
"undeduced type in IR-generation");
272 case Type::BlockPointer:
273 case Type::LValueReference:
274 case Type::RValueReference:
275 case Type::MemberPointer:
277 case Type::ExtVector:
278 case Type::ConstantMatrix:
279 case Type::FunctionProto:
280 case Type::FunctionNoProto:
282 case Type::ObjCObjectPointer:
292 case Type::ConstantArray:
293 case Type::IncompleteArray:
294 case Type::VariableArray:
296 case Type::ObjCObject:
297 case Type::ObjCInterface:
298 case Type::ArrayParameter:
303 type = cast<AtomicType>(
type)->getValueType();
306 llvm_unreachable(
"unknown type kind!");
313 llvm::BasicBlock *CurBB =
Builder.GetInsertBlock();
316 assert(!CurBB->getTerminator() &&
"Unexpected terminated block.");
326 return llvm::DebugLoc();
333 llvm::BranchInst *BI =
335 if (BI && BI->isUnconditional() &&
339 llvm::DebugLoc
Loc = BI->getDebugLoc();
340 Builder.SetInsertPoint(BI->getParent());
341 BI->eraseFromParent();
353 return llvm::DebugLoc();
358 if (!BB->use_empty()) {
366 assert(BreakContinueStack.empty() &&
367 "mismatched push/pop in break/continue stack!");
369 "mismatched push/pop of cleanups in EHStack!");
371 "mismatched activate/deactivate of cleanups!");
376 "mismatched push/pop in convergence stack!");
379 bool OnlySimpleReturnStmts = NumSimpleReturnExprs > 0
380 && NumSimpleReturnExprs == NumReturnExprs
395 if (OnlySimpleReturnStmts)
396 DI->EmitLocation(
Builder, LastStopPoint);
398 DI->EmitLocation(
Builder, EndLoc);
406 bool HasOnlyLifetimeMarkers =
408 bool EmitRetDbgLoc = !HasCleanups || HasOnlyLifetimeMarkers;
410 std::optional<ApplyDebugLocation> OAL;
415 if (OnlySimpleReturnStmts)
416 DI->EmitLocation(
Builder, EndLoc);
431 CurFn->addFnAttr(
"instrument-function-exit",
"__cyg_profile_func_exit");
433 CurFn->addFnAttr(
"instrument-function-exit-inlined",
434 "__cyg_profile_func_exit");
448 "did not remove all scopes from cleanup stack!");
452 if (IndirectBranch) {
459 if (!EscapedLocals.empty()) {
463 EscapeArgs.resize(EscapedLocals.size());
464 for (
auto &Pair : EscapedLocals)
465 EscapeArgs[Pair.second] = Pair.first;
466 llvm::Function *FrameEscapeFn = llvm::Intrinsic::getDeclaration(
474 Ptr->eraseFromParent();
478 if (PostAllocaInsertPt) {
479 llvm::Instruction *PostPtr = PostAllocaInsertPt;
480 PostAllocaInsertPt =
nullptr;
481 PostPtr->eraseFromParent();
486 if (IndirectBranch) {
487 llvm::PHINode *PN = cast<llvm::PHINode>(IndirectBranch->getAddress());
488 if (PN->getNumIncomingValues() == 0) {
489 PN->replaceAllUsesWith(llvm::UndefValue::get(PN->getType()));
490 PN->eraseFromParent();
499 for (
const auto &FuncletAndParent : TerminateFunclets)
505 for (
const auto &R : DeferredReplacements) {
506 if (llvm::Value *Old = R.first) {
507 Old->replaceAllUsesWith(R.second);
508 cast<llvm::Instruction>(Old)->eraseFromParent();
511 DeferredReplacements.clear();
520 llvm::DominatorTree DT(*
CurFn);
521 llvm::PromoteMemToReg(
527 for (llvm::Argument &A :
CurFn->args())
528 if (
auto *VT = dyn_cast<llvm::VectorType>(A.getType()))
530 std::max((uint64_t)LargestVectorWidth,
531 VT->getPrimitiveSizeInBits().getKnownMinValue());
534 if (
auto *VT = dyn_cast<llvm::VectorType>(
CurFn->getReturnType()))
536 std::max((uint64_t)LargestVectorWidth,
537 VT->getPrimitiveSizeInBits().getKnownMinValue());
549 if (
getContext().getTargetInfo().getTriple().isX86())
550 CurFn->addFnAttr(
"min-legal-vector-width",
551 llvm::utostr(LargestVectorWidth));
554 std::optional<std::pair<unsigned, unsigned>> VScaleRange =
557 CurFn->addFnAttr(llvm::Attribute::getWithVScaleRangeArgs(
569 if (RetAlloca && RetAlloca->use_empty()) {
570 RetAlloca->eraseFromParent();
623 llvm::raw_string_ostream Out(Mangled);
625 return llvm::ConstantInt::get(
629void CodeGenFunction::EmitKernelMetadata(
const FunctionDecl *FD,
630 llvm::Function *Fn) {
631 if (!FD->
hasAttr<OpenCLKernelAttr>() && !FD->
hasAttr<CUDAGlobalAttr>())
641 if (
const VecTypeHintAttr *A = FD->
getAttr<VecTypeHintAttr>()) {
642 QualType HintQTy = A->getTypeHint();
644 bool IsSignedInteger =
647 llvm::Metadata *AttrMDArgs[] = {
648 llvm::ConstantAsMetadata::get(llvm::UndefValue::get(
650 llvm::ConstantAsMetadata::get(llvm::ConstantInt::get(
651 llvm::IntegerType::get(Context, 32),
652 llvm::APInt(32, (uint64_t)(IsSignedInteger ? 1 : 0))))};
653 Fn->setMetadata(
"vec_type_hint", llvm::MDNode::get(Context, AttrMDArgs));
656 if (
const WorkGroupSizeHintAttr *A = FD->
getAttr<WorkGroupSizeHintAttr>()) {
657 llvm::Metadata *AttrMDArgs[] = {
658 llvm::ConstantAsMetadata::get(
Builder.getInt32(A->getXDim())),
659 llvm::ConstantAsMetadata::get(
Builder.getInt32(A->getYDim())),
660 llvm::ConstantAsMetadata::get(
Builder.getInt32(A->getZDim()))};
661 Fn->setMetadata(
"work_group_size_hint", llvm::MDNode::get(Context, AttrMDArgs));
664 if (
const ReqdWorkGroupSizeAttr *A = FD->
getAttr<ReqdWorkGroupSizeAttr>()) {
665 llvm::Metadata *AttrMDArgs[] = {
666 llvm::ConstantAsMetadata::get(
Builder.getInt32(A->getXDim())),
667 llvm::ConstantAsMetadata::get(
Builder.getInt32(A->getYDim())),
668 llvm::ConstantAsMetadata::get(
Builder.getInt32(A->getZDim()))};
669 Fn->setMetadata(
"reqd_work_group_size", llvm::MDNode::get(Context, AttrMDArgs));
672 if (
const OpenCLIntelReqdSubGroupSizeAttr *A =
673 FD->
getAttr<OpenCLIntelReqdSubGroupSizeAttr>()) {
674 llvm::Metadata *AttrMDArgs[] = {
675 llvm::ConstantAsMetadata::get(
Builder.getInt32(A->getSubGroupSize()))};
676 Fn->setMetadata(
"intel_reqd_sub_group_size",
677 llvm::MDNode::get(Context, AttrMDArgs));
683 const Stmt *Body =
nullptr;
684 if (
auto *FD = dyn_cast_or_null<FunctionDecl>(F))
686 else if (
auto *OMD = dyn_cast_or_null<ObjCMethodDecl>(F))
687 Body = OMD->getBody();
689 if (
auto *CS = dyn_cast_or_null<CompoundStmt>(Body)) {
690 auto LastStmt = CS->body_rbegin();
691 if (LastStmt != CS->body_rend())
692 return isa<ReturnStmt>(*LastStmt);
699 Fn->addFnAttr(
"sanitize_thread_no_checking_at_run_time");
700 Fn->removeFnAttr(llvm::Attribute::SanitizeThread);
705bool CodeGenFunction::requiresReturnValueCheck()
const {
706 return requiresReturnValueNullabilityCheck() ||
712 auto *MD = dyn_cast_or_null<CXXMethodDecl>(
D);
713 if (!MD || !MD->getDeclName().getAsIdentifierInfo() ||
714 !MD->getDeclName().getAsIdentifierInfo()->isStr(
"allocate") ||
715 (MD->getNumParams() != 1 && MD->getNumParams() != 2))
718 if (MD->parameters()[0]->getType().getCanonicalType() != Ctx.
getSizeType())
721 if (MD->getNumParams() == 2) {
722 auto *PT = MD->parameters()[1]->getType()->getAs<
PointerType>();
723 if (!PT || !PT->isVoidPointerType() ||
724 !PT->getPointeeType().isConstQualified())
736bool CodeGenFunction::hasInAllocaArg(
const CXXMethodDecl *MD) {
740 return isInAllocaArgument(CGM.getCXXABI(), P->getType());
747 if (
const auto *MD = dyn_cast<CXXMethodDecl>(FD))
760 "Do not use a CodeGenFunction object for more than one function");
764 DidCallStackSave =
false;
773 assert(
CurFn->isDeclaration() &&
"Function already has body?");
778#define SANITIZER(NAME, ID) \
779 if (SanOpts.empty()) \
781 if (SanOpts.has(SanitizerKind::ID)) \
782 if (CGM.isInNoSanitizeList(SanitizerKind::ID, Fn, Loc)) \
783 SanOpts.set(SanitizerKind::ID, false);
785#include "clang/Basic/Sanitizers.def"
792 bool NoSanitizeCoverage =
false;
795 no_sanitize_mask |=
Attr->getMask();
797 if (
Attr->hasCoverage())
798 NoSanitizeCoverage =
true;
803 if (no_sanitize_mask & SanitizerKind::Address)
804 SanOpts.
set(SanitizerKind::KernelAddress,
false);
805 if (no_sanitize_mask & SanitizerKind::KernelAddress)
807 if (no_sanitize_mask & SanitizerKind::HWAddress)
808 SanOpts.
set(SanitizerKind::KernelHWAddress,
false);
809 if (no_sanitize_mask & SanitizerKind::KernelHWAddress)
813 Fn->addFnAttr(llvm::Attribute::NoSanitizeBounds);
816 Fn->addFnAttr(llvm::Attribute::NoSanitizeCoverage);
820 if (no_sanitize_mask & SanitizerKind::Thread)
821 Fn->addFnAttr(
"no_sanitize_thread");
826 CurFn->addFnAttr(llvm::Attribute::DisableSanitizerInstrumentation);
829 if (
SanOpts.
hasOneOf(SanitizerKind::Address | SanitizerKind::KernelAddress))
830 Fn->addFnAttr(llvm::Attribute::SanitizeAddress);
832 SanitizerKind::KernelHWAddress))
833 Fn->addFnAttr(llvm::Attribute::SanitizeHWAddress);
835 Fn->addFnAttr(llvm::Attribute::SanitizeMemTag);
837 Fn->addFnAttr(llvm::Attribute::SanitizeThread);
838 if (
SanOpts.
has(SanitizerKind::NumericalStability))
839 Fn->addFnAttr(llvm::Attribute::SanitizeNumericalStability);
840 if (
SanOpts.
hasOneOf(SanitizerKind::Memory | SanitizerKind::KernelMemory))
841 Fn->addFnAttr(llvm::Attribute::SanitizeMemory);
844 Fn->addFnAttr(llvm::Attribute::SafeStack);
845 if (
SanOpts.
has(SanitizerKind::ShadowCallStack))
846 Fn->addFnAttr(llvm::Attribute::ShadowCallStack);
849 if (
SanOpts.
hasOneOf(SanitizerKind::Fuzzer | SanitizerKind::FuzzerNoLink))
850 Fn->addFnAttr(llvm::Attribute::OptForFuzzing);
855 if (
const auto *OMD = dyn_cast_or_null<ObjCMethodDecl>(
D)) {
856 const IdentifierInfo *II = OMD->getSelector().getIdentifierInfoForSlot(0);
859 (OMD->getSelector().isUnarySelector() && II->
isStr(
".cxx_destruct"))) {
868 if (
D &&
SanOpts.
has(SanitizerKind::CFIUnrelatedCast)) {
884 Fn->addFnAttr(
"ptrauth-returns");
886 Fn->addFnAttr(
"ptrauth-calls");
888 Fn->addFnAttr(
"ptrauth-auth-traps");
890 Fn->addFnAttr(
"ptrauth-indirect-gotos");
893 bool AlwaysXRayAttr =
false;
894 if (
const auto *XRayAttr =
D ?
D->
getAttr<XRayInstrumentAttr>() :
nullptr) {
900 Fn->addFnAttr(
"function-instrument",
"xray-always");
901 AlwaysXRayAttr =
true;
903 if (XRayAttr->neverXRayInstrument())
904 Fn->addFnAttr(
"function-instrument",
"xray-never");
905 if (
const auto *LogArgs =
D->
getAttr<XRayLogArgsAttr>())
907 Fn->addFnAttr(
"xray-log-args",
908 llvm::utostr(LogArgs->getArgumentCount()));
913 "xray-instruction-threshold",
919 Fn->addFnAttr(
"xray-ignore-loops");
923 Fn->addFnAttr(
"xray-skip-exit");
927 Fn->addFnAttr(
"xray-skip-entry");
930 if (FuncGroups > 1) {
932 CurFn->getName().bytes_end());
933 auto Group = crc32(FuncName) % FuncGroups;
936 Fn->addFnAttr(
"function-instrument",
"xray-never");
943 Fn->addFnAttr(llvm::Attribute::SkipProfile);
946 Fn->addFnAttr(llvm::Attribute::NoProfile);
953 unsigned Count, Offset;
954 if (
const auto *
Attr =
955 D ?
D->
getAttr<PatchableFunctionEntryAttr>() :
nullptr) {
956 Count =
Attr->getCount();
957 Offset =
Attr->getOffset();
962 if (Count && Offset <= Count) {
963 Fn->addFnAttr(
"patchable-function-entry", std::to_string(Count - Offset));
965 Fn->addFnAttr(
"patchable-function-prefix", std::to_string(Offset));
972 getContext().getTargetInfo().getTriple().isX86() &&
973 getContext().getTargetInfo().getTriple().getEnvironment() !=
974 llvm::Triple::CODE16)
975 Fn->addFnAttr(
"patchable-function",
"prologue-short-redirect");
979 Fn->addFnAttr(
"no-jump-tables",
"true");
983 Fn->addFnAttr(
"no-inline-line-tables");
987 Fn->addFnAttr(
"profile-sample-accurate");
990 Fn->addFnAttr(
"use-sample-profile");
992 if (
D &&
D->
hasAttr<CFICanonicalJumpTableAttr>())
993 Fn->addFnAttr(
"cfi-canonical-jump-table");
995 if (
D &&
D->
hasAttr<NoProfileFunctionAttr>())
996 Fn->addFnAttr(llvm::Attribute::NoProfile);
998 if (
D &&
D->
hasAttr<HybridPatchableAttr>())
999 Fn->addFnAttr(llvm::Attribute::HybridPatchable);
1003 if (
auto *A =
D->
getAttr<FunctionReturnThunksAttr>()) {
1004 switch (A->getThunkType()) {
1005 case FunctionReturnThunksAttr::Kind::Keep:
1007 case FunctionReturnThunksAttr::Kind::Extern:
1008 Fn->addFnAttr(llvm::Attribute::FnRetThunkExtern);
1012 Fn->addFnAttr(llvm::Attribute::FnRetThunkExtern);
1019 EmitKernelMetadata(FD, Fn);
1022 if (FD && FD->
hasAttr<ClspvLibclcBuiltinAttr>()) {
1023 Fn->setMetadata(
"clspv_libclc_builtin",
1029 if (FD &&
SanOpts.
has(SanitizerKind::Function)) {
1031 llvm::LLVMContext &Ctx =
Fn->getContext();
1032 llvm::MDBuilder MDB(Ctx);
1034 llvm::LLVMContext::MD_func_sanitize,
1035 MDB.createRTTIPointerPrologue(
1042 if (
SanOpts.
has(SanitizerKind::NullabilityReturn)) {
1046 if (!(
SanOpts.
has(SanitizerKind::ReturnsNonnullAttribute) &&
1048 RetValNullabilityPrecondition =
1067 Fn->addFnAttr(llvm::Attribute::NoRecurse);
1070 llvm::fp::ExceptionBehavior FPExceptionBehavior =
1072 Builder.setDefaultConstrainedRounding(RM);
1073 Builder.setDefaultConstrainedExcept(FPExceptionBehavior);
1075 (!FD && (FPExceptionBehavior != llvm::fp::ebIgnore ||
1076 RM != llvm::RoundingMode::NearestTiesToEven))) {
1077 Builder.setIsFPConstrained(
true);
1078 Fn->addFnAttr(llvm::Attribute::StrictFP);
1085 Fn->addFnAttr(
"stackrealign");
1089 Fn->removeFnAttr(
"zero-call-used-regs");
1096 llvm::Value *Undef = llvm::UndefValue::get(
Int32Ty);
1101 Builder.SetInsertPoint(EntryBB);
1105 if (requiresReturnValueCheck()) {
1116 DI->emitFunctionStart(GD,
Loc, StartLoc,
1117 DI->getFunctionType(FD, RetTy, Args),
CurFn,
1123 CurFn->addFnAttr(
"instrument-function-entry",
"__cyg_profile_func_enter");
1125 CurFn->addFnAttr(
"instrument-function-entry-inlined",
1126 "__cyg_profile_func_enter");
1128 CurFn->addFnAttr(
"instrument-function-entry-inlined",
1129 "__cyg_profile_func_enter_bare");
1141 Fn->addFnAttr(
"fentry-call",
"true");
1143 Fn->addFnAttr(
"instrument-function-entry-inlined",
1149 <<
"-mnop-mcount" <<
"-mfentry";
1150 Fn->addFnAttr(
"mnop-mcount");
1156 <<
"-mrecord-mcount" <<
"-mfentry";
1157 Fn->addFnAttr(
"mrecord-mcount");
1163 if (
getContext().getTargetInfo().getTriple().getArch() !=
1164 llvm::Triple::systemz)
1166 <<
"-mpacked-stack";
1167 Fn->addFnAttr(
"packed-stack");
1172 Fn->addFnAttr(
"warn-stack-size",
1185 auto AI =
CurFn->arg_begin();
1201 llvm::Function::arg_iterator EI =
CurFn->arg_end();
1206 cast<llvm::GetElementPtrInst>(Addr)->getResultElementType();
1233 if (FD->
hasAttr<HLSLShaderAttr>()) {
1241 if (
const CXXMethodDecl *MD = dyn_cast_if_present<CXXMethodDecl>(
D);
1262 CXXThisValue = ThisFieldLValue.
getPointer(*
this);
1271 if (FD->hasCapturedVLAType()) {
1274 auto VAT = FD->getCapturedVLAType();
1275 VLASizeMap[VAT->getSizeExpr()] = ExprArg;
1282 CXXThisValue = CXXABIThisValue;
1286 if (CXXABIThisValue) {
1288 SkippedChecks.
set(SanitizerKind::ObjectSize,
true);
1295 SkippedChecks.
set(SanitizerKind::Null,
true);
1299 Loc, CXXABIThisValue, ThisTy, CXXABIThisAlignment, SkippedChecks);
1306 if (!FD || !FD->
hasAttr<NakedAttr>()) {
1307 for (
const VarDecl *VD : Args) {
1312 if (
const ParmVarDecl *PVD = dyn_cast<ParmVarDecl>(VD))
1313 Ty = PVD->getOriginalType();
1323 DI->EmitLocation(
Builder, StartLoc);
1328 LargestVectorWidth = VecWidth->getVectorWidth();
1337 if (
const CompoundStmt *S = dyn_cast<CompoundStmt>(Body))
1349 llvm::BasicBlock *SkipCountBB =
nullptr;
1374 if (F->isInterposable())
return;
1376 for (llvm::BasicBlock &BB : *F)
1377 for (llvm::Instruction &I : BB)
1381 F->setDoesNotThrow();
1401 bool PassedParams =
true;
1403 if (
auto Inherited = CD->getInheritedConstructor())
1409 Args.push_back(Param);
1410 if (!Param->hasAttr<PassObjectSizeAttr>())
1414 getContext(), Param->getDeclContext(), Param->getLocation(),
1421 if (MD && (isa<CXXConstructorDecl>(MD) || isa<CXXDestructorDecl>(MD)))
1429 assert(Fn &&
"generating code for null Function");
1442 std::string FDInlineName = (
Fn->getName() +
".inline").str();
1443 llvm::Module *M =
Fn->getParent();
1444 llvm::Function *Clone = M->getFunction(FDInlineName);
1446 Clone = llvm::Function::Create(
Fn->getFunctionType(),
1447 llvm::GlobalValue::InternalLinkage,
1448 Fn->getAddressSpace(), FDInlineName, M);
1449 Clone->addFnAttr(llvm::Attribute::AlwaysInline);
1451 Fn->setLinkage(llvm::GlobalValue::ExternalLinkage);
1461 if (LLVM_UNLIKELY(PD->isInlineBuiltinDeclaration())) {
1462 std::string FDInlineName = (
Fn->getName() +
".inline").str();
1463 llvm::Module *M =
Fn->getParent();
1464 if (llvm::Function *Clone = M->getFunction(FDInlineName)) {
1465 Clone->replaceAllUsesWith(Fn);
1466 Clone->eraseFromParent();
1474 if (FD->
hasAttr<NoDebugAttr>()) {
1477 Fn->setSubprogram(
nullptr);
1479 DebugInfo =
nullptr;
1489 CurEHLocation = BodyRange.
getEnd();
1501 if (SpecDecl->hasBody(SpecDecl))
1502 Loc = SpecDecl->getLocation();
1508 if (isa<CoroutineBodyStmt>(Body))
1509 ShouldEmitLifetimeMarkers =
true;
1513 if (ShouldEmitLifetimeMarkers)
1521 if (Body && isa_and_nonnull<CoroutineBodyStmt>(Body))
1529 CurFn->addFnAttr(llvm::Attribute::MustProgress);
1533 if (isa<CXXDestructorDecl>(FD))
1535 else if (isa<CXXConstructorDecl>(FD))
1539 FD->
hasAttr<CUDAGlobalAttr>())
1541 else if (isa<CXXMethodDecl>(FD) &&
1542 cast<CXXMethodDecl>(FD)->isLambdaStaticInvoker()) {
1546 }
else if (isa<CXXMethodDecl>(FD) &&
1549 cast<CXXMethodDecl>(FD)->getParent()->getLambdaStaticInvoker() &&
1550 hasInAllocaArg(cast<CXXMethodDecl>(FD))) {
1557 }
else if (FD->
isDefaulted() && isa<CXXMethodDecl>(FD) &&
1558 (cast<CXXMethodDecl>(FD)->isCopyAssignmentOperator() ||
1559 cast<CXXMethodDecl>(FD)->isMoveAssignmentOperator())) {
1566 llvm_unreachable(
"no definition for emitted function");
1576 bool ShouldEmitUnreachable =
1580 SanitizerScope SanScope(
this);
1581 llvm::Value *IsFalse =
Builder.getFalse();
1582 EmitCheck(std::make_pair(IsFalse, SanitizerKind::Return),
1583 SanitizerHandler::MissingReturn,
1585 }
else if (ShouldEmitUnreachable) {
1589 if (
SanOpts.
has(SanitizerKind::Return) || ShouldEmitUnreachable) {
1591 Builder.ClearInsertionPoint();
1600 if (!
CurFn->doesNotThrow())
1609 if (!S)
return false;
1616 if (isa<LabelStmt>(S))
1621 if (isa<SwitchCase>(S) && !IgnoreCaseStmts)
1625 if (isa<SwitchStmt>(S))
1626 IgnoreCaseStmts =
true;
1629 for (
const Stmt *SubStmt : S->children())
1641 if (!S)
return false;
1645 if (isa<SwitchStmt>(S) || isa<WhileStmt>(S) || isa<DoStmt>(S) ||
1649 if (isa<BreakStmt>(S))
1653 for (
const Stmt *SubStmt : S->children())
1661 if (!S)
return false;
1667 if (isa<IfStmt>(S) || isa<SwitchStmt>(S) || isa<WhileStmt>(S) ||
1668 isa<DoStmt>(S) || isa<ForStmt>(S) || isa<CompoundStmt>(S) ||
1669 isa<CXXForRangeStmt>(S) || isa<CXXTryStmt>(S) ||
1670 isa<ObjCForCollectionStmt>(S) || isa<ObjCAtTryStmt>(S))
1673 if (isa<DeclStmt>(S))
1676 for (
const Stmt *SubStmt : S->children())
1696 llvm::APSInt ResultInt;
1700 ResultBool = ResultInt.getBoolValue();
1708 llvm::APSInt &ResultInt,
1726 while (
const UnaryOperator *Op = dyn_cast<UnaryOperator>(
C->IgnoreParens())) {
1727 if (Op->getOpcode() != UO_LNot)
1729 C = Op->getSubExpr();
1731 return C->IgnoreParens();
1747 llvm::BasicBlock *FalseBlock, uint64_t TrueCount ,
1754 llvm::BasicBlock *ThenBlock =
nullptr;
1755 llvm::BasicBlock *ElseBlock =
nullptr;
1756 llvm::BasicBlock *NextBlock =
nullptr;
1773 if (LOp == BO_LAnd) {
1774 ThenBlock = CounterIncrBlock;
1775 ElseBlock = FalseBlock;
1776 NextBlock = TrueBlock;
1791 else if (LOp == BO_LOr) {
1792 ThenBlock = TrueBlock;
1793 ElseBlock = CounterIncrBlock;
1794 NextBlock = FalseBlock;
1796 llvm_unreachable(
"Expected Opcode must be that of a Logical Operator");
1820 const Expr *Cond, llvm::BasicBlock *TrueBlock, llvm::BasicBlock *FalseBlock,
1824 if (
const BinaryOperator *CondBOp = dyn_cast<BinaryOperator>(Cond)) {
1826 if (CondBOp->getOpcode() == BO_LAnd) {
1831 bool ConstantBool =
false;
1837 FalseBlock, TrueCount, LH);
1848 FalseBlock, TrueCount, LH, CondBOp);
1860 ConditionalEvaluation eval(*
this);
1877 FalseBlock, TrueCount, LH);
1883 if (CondBOp->getOpcode() == BO_LOr) {
1888 bool ConstantBool =
false;
1894 FalseBlock, TrueCount, LH);
1905 FalseBlock, TrueCount, LH, CondBOp);
1917 uint64_t RHSCount = TrueCount - LHSCount;
1919 ConditionalEvaluation eval(*
this);
1944 if (
const UnaryOperator *CondUOp = dyn_cast<UnaryOperator>(Cond)) {
1952 if (CondUOp->getOpcode() == UO_LNot && !MCDCCondition) {
1970 ConditionalEvaluation cond(*
this);
1983 LHSScaledTrueCount = TrueCount * LHSRatio;
1992 LHSScaledTrueCount, LH, CondOp);
1999 TrueCount - LHSScaledTrueCount, LH, CondOp);
2005 if (
const CXXThrowExpr *Throw = dyn_cast<CXXThrowExpr>(Cond)) {
2025 const Expr *MCDCBaseExpr = Cond;
2032 MCDCBaseExpr = ConditionalOp;
2037 llvm::MDNode *Weights =
nullptr;
2038 llvm::MDNode *Unpredictable =
nullptr;
2045 auto *FD = dyn_cast_or_null<FunctionDecl>(
Call->getCalleeDecl());
2046 if (FD && FD->
getBuiltinID() == Builtin::BI__builtin_unpredictable) {
2048 Unpredictable = MDHelper.createUnpredictable();
2054 llvm::Value *NewCondV = emitCondLikelihoodViaExpectIntrinsic(CondV, LH);
2055 if (CondV != NewCondV)
2060 Weights = createProfileWeights(TrueCount, CurrentCount - TrueCount);
2063 Builder.CreateCondBr(CondV, TrueBlock, FalseBlock, Weights, Unpredictable);
2081 llvm::Value *sizeInChars) {
2085 llvm::Value *baseSizeInChars
2091 sizeInChars,
"vla.end");
2093 llvm::BasicBlock *originBB = CGF.
Builder.GetInsertBlock();
2101 llvm::PHINode *cur =
Builder.CreatePHI(begin.
getType(), 2,
"vla.cur");
2116 llvm::Value *done =
Builder.CreateICmpEQ(next, end,
"vla-init.isdone");
2117 Builder.CreateCondBr(done, contBB, loopBB);
2118 cur->addIncoming(next, loopBB);
2128 if (cast<CXXRecordDecl>(RT->getDecl())->isEmpty())
2139 llvm::Value *SizeVal;
2146 dyn_cast_or_null<VariableArrayType>(
2149 SizeVal = VlaSize.NumElts;
2151 if (!eltSize.
isOne())
2172 llvm::GlobalVariable *NullVariable =
2173 new llvm::GlobalVariable(
CGM.
getModule(), NullConstant->getType(),
2175 llvm::GlobalVariable::PrivateLinkage,
2176 NullConstant, Twine());
2178 NullVariable->setAlignment(NullAlign.
getAsAlign());
2196 if (!IndirectBranch)
2202 IndirectBranch->addDestination(BB);
2203 return llvm::BlockAddress::get(
CurFn, BB);
2208 if (IndirectBranch)
return IndirectBranch->getParent();
2213 llvm::Value *DestVal = TmpBuilder.CreatePHI(
Int8PtrTy, 0,
2214 "indirect.goto.dest");
2217 IndirectBranch = TmpBuilder.CreateIndirectBr(DestVal);
2218 return IndirectBranch->getParent();
2230 llvm::Value *numVLAElements =
nullptr;
2231 if (isa<VariableArrayType>(
arrayType)) {
2242 baseType = elementType;
2243 return numVLAElements;
2245 }
while (isa<VariableArrayType>(
arrayType));
2257 llvm::ConstantInt *zero =
Builder.getInt32(0);
2258 gepIndices.push_back(zero);
2263 llvm::ArrayType *llvmArrayType =
2265 while (llvmArrayType) {
2266 assert(isa<ConstantArrayType>(
arrayType));
2267 assert(cast<ConstantArrayType>(
arrayType)->getZExtSize() ==
2268 llvmArrayType->getNumElements());
2270 gepIndices.push_back(zero);
2271 countFromCLAs *= llvmArrayType->getNumElements();
2275 dyn_cast<llvm::ArrayType>(llvmArrayType->getElementType());
2278 "LLVM and Clang types are out-of-synch");
2286 countFromCLAs *= cast<ConstantArrayType>(
arrayType)->getZExtSize();
2297 gepIndices,
"array.begin"),
2303 llvm::Value *numElements
2304 = llvm::ConstantInt::get(
SizeTy, countFromCLAs);
2308 numElements =
Builder.CreateNUWMul(numVLAElements, numElements);
2315 assert(vla &&
"type was not a variable array type!");
2319CodeGenFunction::VlaSizePair
2322 llvm::Value *numElements =
nullptr;
2326 elementType =
type->getElementType();
2327 llvm::Value *vlaSize = VLASizeMap[
type->getSizeExpr()];
2328 assert(vlaSize &&
"no size for VLA!");
2329 assert(vlaSize->getType() ==
SizeTy);
2332 numElements = vlaSize;
2336 numElements =
Builder.CreateNUWMul(numElements, vlaSize);
2338 }
while ((
type =
getContext().getAsVariableArrayType(elementType)));
2340 return { numElements, elementType };
2343CodeGenFunction::VlaSizePair
2346 assert(vla &&
"type was not a variable array type!");
2350CodeGenFunction::VlaSizePair
2352 llvm::Value *VlaSize = VLASizeMap[Vla->
getSizeExpr()];
2353 assert(VlaSize &&
"no size for VLA!");
2354 assert(VlaSize->getType() ==
SizeTy);
2359 assert(
type->isVariablyModifiedType() &&
2360 "Must pass variably modified type to EmitVLASizes!");
2367 assert(
type->isVariablyModifiedType());
2369 const Type *ty =
type.getTypePtr();
2372#define TYPE(Class, Base)
2373#define ABSTRACT_TYPE(Class, Base)
2374#define NON_CANONICAL_TYPE(Class, Base)
2375#define DEPENDENT_TYPE(Class, Base) case Type::Class:
2376#define NON_CANONICAL_UNLESS_DEPENDENT_TYPE(Class, Base)
2377#include "clang/AST/TypeNodes.inc"
2378 llvm_unreachable(
"unexpected dependent type!");
2384 case Type::ExtVector:
2385 case Type::ConstantMatrix:
2389 case Type::TemplateSpecialization:
2390 case Type::ObjCTypeParam:
2391 case Type::ObjCObject:
2392 case Type::ObjCInterface:
2393 case Type::ObjCObjectPointer:
2395 llvm_unreachable(
"type class is never variably-modified!");
2397 case Type::Elaborated:
2398 type = cast<ElaboratedType>(ty)->getNamedType();
2401 case Type::Adjusted:
2402 type = cast<AdjustedType>(ty)->getAdjustedType();
2406 type = cast<DecayedType>(ty)->getPointeeType();
2410 type = cast<PointerType>(ty)->getPointeeType();
2413 case Type::BlockPointer:
2414 type = cast<BlockPointerType>(ty)->getPointeeType();
2417 case Type::LValueReference:
2418 case Type::RValueReference:
2419 type = cast<ReferenceType>(ty)->getPointeeType();
2422 case Type::MemberPointer:
2423 type = cast<MemberPointerType>(ty)->getPointeeType();
2426 case Type::ArrayParameter:
2427 case Type::ConstantArray:
2428 case Type::IncompleteArray:
2430 type = cast<ArrayType>(ty)->getElementType();
2433 case Type::VariableArray: {
2442 llvm::Value *&entry = VLASizeMap[sizeExpr];
2451 SanitizerScope SanScope(
this);
2452 llvm::Value *
Zero = llvm::Constant::getNullValue(size->getType());
2454 llvm::Value *CheckCondition =
2456 ?
Builder.CreateICmpSGT(size, Zero)
2457 :
Builder.CreateICmpUGT(size, Zero);
2458 llvm::Constant *StaticArgs[] = {
2461 EmitCheck(std::make_pair(CheckCondition, SanitizerKind::VLABound),
2462 SanitizerHandler::VLABoundNotPositive, StaticArgs, size);
2475 case Type::FunctionProto:
2476 case Type::FunctionNoProto:
2477 type = cast<FunctionType>(ty)->getReturnType();
2482 case Type::UnaryTransform:
2483 case Type::Attributed:
2484 case Type::BTFTagAttributed:
2485 case Type::SubstTemplateTypeParm:
2486 case Type::MacroQualified:
2487 case Type::CountAttributed:
2493 case Type::Decltype:
2495 case Type::DeducedTemplateSpecialization:
2496 case Type::PackIndexing:
2500 case Type::TypeOfExpr:
2506 type = cast<AtomicType>(ty)->getValueType();
2510 type = cast<PipeType>(ty)->getElementType();
2513 }
while (
type->isVariablyModifiedType());
2517 if (
getContext().getBuiltinVaListType()->isArrayType())
2528 assert(
Init.hasValue() &&
"Invalid DeclRefExpr initializer!");
2531 Dbg->EmitGlobalVariable(
E->getDecl(),
Init);
2534CodeGenFunction::PeepholeProtection
2540 if (!rvalue.
isScalar())
return PeepholeProtection();
2542 if (!isa<llvm::ZExtInst>(value))
return PeepholeProtection();
2546 llvm::Instruction *inst =
new llvm::BitCastInst(value, value->getType(),
"",
2549 PeepholeProtection protection;
2550 protection.Inst = inst;
2555 if (!protection.Inst)
return;
2558 protection.Inst->eraseFromParent();
2564 llvm::Value *Alignment,
2565 llvm::Value *OffsetValue) {
2566 if (Alignment->getType() !=
IntPtrTy)
2569 if (OffsetValue && OffsetValue->getType() !=
IntPtrTy)
2572 llvm::Value *TheCheck =
nullptr;
2574 llvm::Value *PtrIntValue =
2578 bool IsOffsetZero =
false;
2579 if (
const auto *CI = dyn_cast<llvm::ConstantInt>(OffsetValue))
2580 IsOffsetZero = CI->isZero();
2583 PtrIntValue =
Builder.CreateSub(PtrIntValue, OffsetValue,
"offsetptr");
2586 llvm::Value *
Zero = llvm::ConstantInt::get(
IntPtrTy, 0);
2589 llvm::Value *MaskedPtr =
Builder.CreateAnd(PtrIntValue, Mask,
"maskedptr");
2590 TheCheck =
Builder.CreateICmpEQ(MaskedPtr, Zero,
"maskcond");
2592 llvm::Instruction *Assumption =
Builder.CreateAlignmentAssumption(
2598 OffsetValue, TheCheck, Assumption);
2604 llvm::Value *Alignment,
2605 llvm::Value *OffsetValue) {
2614 llvm::Value *AnnotatedVal,
2615 StringRef AnnotationStr,
2617 const AnnotateAttr *
Attr) {
2626 return Builder.CreateCall(AnnotationFn, Args);
2630 assert(
D->
hasAttr<AnnotateAttr>() &&
"no annotate attribute");
2633 {V->getType(), CGM.ConstGlobalsPtrTy}),
2639 assert(
D->
hasAttr<AnnotateAttr>() &&
"no annotate attribute");
2641 llvm::Type *VTy =
V->getType();
2642 auto *PTy = dyn_cast<llvm::PointerType>(VTy);
2643 unsigned AS = PTy ? PTy->getAddressSpace() : 0;
2644 llvm::PointerType *IntrinTy =
2653 if (VTy != IntrinTy)
2671 CGF->IsSanitizerScope =
false;
2675 const llvm::Twine &Name,
2676 llvm::BasicBlock::iterator InsertPt)
const {
2679 I->setNoSanitizeMetadata();
2683 llvm::Instruction *I,
const llvm::Twine &Name,
2684 llvm::BasicBlock::iterator InsertPt)
const {
2685 llvm::IRBuilderDefaultInserter::InsertHelper(I, Name, InsertPt);
2687 CGF->InsertHelper(I, Name, InsertPt);
2698 if (BuiltinID == X86::BI__builtin_ia32_cmpps ||
2699 BuiltinID == X86::BI__builtin_ia32_cmpss ||
2700 BuiltinID == X86::BI__builtin_ia32_cmppd ||
2701 BuiltinID == X86::BI__builtin_ia32_cmpsd) {
2703 llvm::StringMap<bool> TargetFetureMap;
2707 if (
Result.getSExtValue() > 7 && !TargetFetureMap.lookup(
"avx"))
2733 std::string MissingFeature;
2734 llvm::StringMap<bool> CallerFeatureMap;
2744 FeatureList, CallerFeatureMap) && !IsHipStdPar) {
2750 TargetDecl->
hasAttr<TargetAttr>()) {
2753 const TargetAttr *TD = TargetDecl->
getAttr<TargetAttr>();
2758 llvm::StringMap<bool> CalleeFeatureMap;
2762 if (F[0] ==
'+' && CalleeFeatureMap.lookup(F.substr(1)))
2763 ReqFeatures.push_back(StringRef(F).substr(1));
2766 for (
const auto &F : CalleeFeatureMap) {
2769 ReqFeatures.push_back(F.getKey());
2771 if (!llvm::all_of(ReqFeatures, [&](StringRef Feature) {
2772 if (!CallerFeatureMap.lookup(Feature)) {
2773 MissingFeature = Feature.str();
2781 llvm::StringMap<bool> CalleeFeatureMap;
2784 for (
const auto &F : CalleeFeatureMap) {
2785 if (F.getValue() && (!CallerFeatureMap.lookup(F.getKey()) ||
2786 !CallerFeatureMap.find(F.getKey())->getValue()) &&
2798 llvm::IRBuilder<> IRB(
Builder.GetInsertBlock(),
Builder.GetInsertPoint());
2799 IRB.SetCurrentDebugLocation(
Builder.getCurrentDebugLocation());
2806 Callee.getAbstractInfo().getCalleeFunctionProtoType();
2811llvm::Value *CodeGenFunction::FormAArch64ResolverCondition(
2812 const MultiVersionResolverOption &RO) {
2814 for (
const StringRef &Feature : RO.Conditions.Features)
2815 CondFeatures.push_back(Feature);
2816 if (!CondFeatures.empty()) {
2817 return EmitAArch64CpuSupports(CondFeatures);
2822llvm::Value *CodeGenFunction::FormX86ResolverCondition(
2823 const MultiVersionResolverOption &RO) {
2826 if (!RO.Conditions.Architecture.empty()) {
2827 StringRef Arch = RO.Conditions.Architecture;
2830 if (Arch.starts_with(
"x86-64"))
2836 if (!RO.Conditions.Features.empty()) {
2837 llvm::Value *FeatureCond = EmitX86CpuSupports(RO.Conditions.Features);
2845 llvm::Function *Resolver,
2847 llvm::Function *FuncToReturn,
2848 bool SupportsIFunc) {
2849 if (SupportsIFunc) {
2850 Builder.CreateRet(FuncToReturn);
2855 llvm::make_pointer_range(Resolver->args()));
2857 llvm::CallInst *
Result =
Builder.CreateCall(FuncToReturn, Args);
2858 Result->setTailCallKind(llvm::CallInst::TCK_MustTail);
2860 if (Resolver->getReturnType()->isVoidTy())
2869 llvm::Triple::ArchType ArchType =
2873 case llvm::Triple::x86:
2874 case llvm::Triple::x86_64:
2877 case llvm::Triple::aarch64:
2882 assert(
false &&
"Only implemented for x86 and AArch64 targets");
2888 assert(!Options.empty() &&
"No multiversion resolver options found");
2889 assert(Options.back().Conditions.Features.size() == 0 &&
2890 "Default case must be last");
2892 assert(SupportsIFunc &&
2893 "Multiversion resolver requires target IFUNC support");
2894 bool AArch64CpuInitialized =
false;
2897 for (
const MultiVersionResolverOption &RO : Options) {
2898 Builder.SetInsertPoint(CurBlock);
2899 llvm::Value *
Condition = FormAArch64ResolverCondition(RO);
2908 if (!AArch64CpuInitialized) {
2909 Builder.SetInsertPoint(CurBlock, CurBlock->begin());
2910 EmitAArch64CpuInit();
2911 AArch64CpuInitialized =
true;
2912 Builder.SetInsertPoint(CurBlock);
2915 llvm::BasicBlock *RetBlock =
createBasicBlock(
"resolver_return", Resolver);
2924 Builder.SetInsertPoint(CurBlock);
2925 llvm::CallInst *TrapCall =
EmitTrapCall(llvm::Intrinsic::trap);
2926 TrapCall->setDoesNotReturn();
2927 TrapCall->setDoesNotThrow();
2929 Builder.ClearInsertionPoint();
2939 Builder.SetInsertPoint(CurBlock);
2942 for (
const MultiVersionResolverOption &RO : Options) {
2943 Builder.SetInsertPoint(CurBlock);
2944 llvm::Value *
Condition = FormX86ResolverCondition(RO);
2948 assert(&RO == Options.end() - 1 &&
2949 "Default or Generic case must be last");
2955 llvm::BasicBlock *RetBlock =
createBasicBlock(
"resolver_return", Resolver);
2964 Builder.SetInsertPoint(CurBlock);
2965 llvm::CallInst *TrapCall =
EmitTrapCall(llvm::Intrinsic::trap);
2966 TrapCall->setDoesNotReturn();
2967 TrapCall->setDoesNotThrow();
2969 Builder.ClearInsertionPoint();
2981 llvm::Value *OffsetValue, llvm::Value *TheCheck,
2982 llvm::Instruction *Assumption) {
2983 assert(isa_and_nonnull<llvm::CallInst>(Assumption) &&
2984 cast<llvm::CallInst>(Assumption)->getCalledOperand() ==
2985 llvm::Intrinsic::getDeclaration(
2986 Builder.GetInsertBlock()->getParent()->getParent(),
2987 llvm::Intrinsic::assume) &&
2988 "Assumption should be a call to llvm.assume().");
2989 assert(&(
Builder.GetInsertBlock()->back()) == Assumption &&
2990 "Assumption should be the last instruction of the basic block, "
2991 "since the basic block is still being generated.");
3003 Assumption->removeFromParent();
3006 SanitizerScope SanScope(
this);
3009 OffsetValue =
Builder.getInt1(
false);
3017 EmitCheck({std::make_pair(TheCheck, SanitizerKind::Alignment)},
3018 SanitizerHandler::AlignmentAssumption, StaticData, DynamicData);
3029 return DI->SourceLocToDebugLoc(Location);
3031 return llvm::DebugLoc();
3035CodeGenFunction::emitCondLikelihoodViaExpectIntrinsic(llvm::Value *Cond,
3046 llvm::Type *CondTy = Cond->getType();
3047 assert(CondTy->isIntegerTy(1) &&
"expecting condition to be a boolean");
3048 llvm::Function *FnExpect =
3050 llvm::Value *ExpectedValueOfCond =
3052 return Builder.CreateCall(FnExpect, {Cond, ExpectedValueOfCond},
3053 Cond->getName() +
".expval");
3055 llvm_unreachable(
"Unknown Likelihood");
3059 unsigned NumElementsDst,
3060 const llvm::Twine &Name) {
3061 auto *SrcTy = cast<llvm::FixedVectorType>(SrcVec->getType());
3062 unsigned NumElementsSrc = SrcTy->getNumElements();
3063 if (NumElementsSrc == NumElementsDst)
3066 std::vector<int> ShuffleMask(NumElementsDst, -1);
3067 for (
unsigned MaskIdx = 0;
3068 MaskIdx < std::min<>(NumElementsDst, NumElementsSrc); ++MaskIdx)
3069 ShuffleMask[MaskIdx] = MaskIdx;
3071 return Builder.CreateShuffleVector(SrcVec, ShuffleMask, Name);
3086 llvm::Value *Args[] = {Key, Discriminator};
3087 Bundles.emplace_back(
"ptrauth", Args);
3093 unsigned IntrinsicID) {
3100 if (!Discriminator) {
3105 auto OrigType =
Pointer->getType();
3123 llvm::Intrinsic::ptrauth_sign);
3129 auto StripIntrinsic = CGF.
CGM.
getIntrinsic(llvm::Intrinsic::ptrauth_strip);
3133 auto OrigType =
Pointer->getType();
3150 llvm::Intrinsic::ptrauth_auth);
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....
llvm::MachO::Target Target
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 ...
ParsedTargetAttr filterFunctionTargetAttrs(const TargetAttr *TD) const
Parses the target attributes passed in, and returns only the ones that are valid feature names.
Builtin::Context & BuiltinInfo
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.
CanQualType getSizeType() const
Return the unique type for "size_t" (C99 7.17), defined in <stddef.h>.
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.
const VariableArrayType * getAsVariableArrayType(QualType T) const
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)
const char * getRequiredFeatures(unsigned ID) const
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.
Represents a C++ struct/union/class.
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
A C++ throw-expression (C++ [except.throw]).
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
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...
std::string SampleProfileFile
Name of the profile file to use with -fprofile-sample-use.
bool hasProfileClangInstr() const
Check if Clang profile instrumenation is on.
XRayInstrSet XRayInstrumentationBundle
Set of XRay instrumentation kinds to emit.
bool hasSanitizeCoverage() const
PointerAuthOptions PointerAuth
Configuration for pointer-signing.
bool hasReducedDebugInfo() const
Check if type and variable info should be emitted.
bool hasSanitizeBinaryMetadata() const
unsigned getInAllocaFieldIndex() const
bool getIndirectByVal() const
@ 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...
bool isSRetAfterThis() const
CharUnits getIndirectAlign() const
Like RawAddress, an abstract representation of an aligned address, but the pointer contained in this ...
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 debug location to the specified location or preferred location of ...
static ApplyDebugLocation CreateDefaultArtificial(CodeGenFunction &CGF, SourceLocation TemporaryLocation)
Apply TemporaryLocation if it is valid.
This is an IRBuilder insertion helper that forwards to CodeGenFunction::InsertHelper,...
void InsertHelper(llvm::Instruction *I, const llvm::Twine &Name, llvm::BasicBlock::iterator InsertPt) const override
This forwards to CodeGenFunction::InsertHelper.
llvm::StoreInst * CreateStore(llvm::Value *Val, Address Addr, bool IsVolatile=false)
llvm::CallInst * CreateMemSet(Address Dest, llvm::Value *Value, llvm::Value *Size, bool IsVolatile=false)
Address CreateStructGEP(Address Addr, unsigned Index, const llvm::Twine &Name="")
llvm::CallInst * CreateMemCpy(Address Dest, Address Src, llvm::Value *Size, bool IsVolatile=false)
llvm::LoadInst * CreateAlignedLoad(llvm::Type *Ty, llvm::Value *Addr, CharUnits Align, const llvm::Twine &Name="")
llvm::ConstantInt * getSize(CharUnits N)
Address CreateInBoundsGEP(Address Addr, ArrayRef< llvm::Value * > IdxList, llvm::Type *ElementType, CharUnits Align, const Twine &Name="")
virtual void emitDeviceStub(CodeGenFunction &CGF, FunctionArgList &Args)=0
Emits a kernel launch stub.
Implements C++ ABI-specific code generation functions.
virtual bool hasMostDerivedReturn(GlobalDecl GD) const
virtual bool HasThisReturn(GlobalDecl GD) const
Returns true if the given constructor or destructor is one of the kinds that the ABI says returns 'th...
virtual void EmitInstanceFunctionProlog(CodeGenFunction &CGF)=0
Emit the ABI-specific prolog for the function.
@ RAA_DirectInMemory
Pass it on the stack using its defined layout.
void buildThisParam(CodeGenFunction &CGF, FunctionArgList &Params)
Build a parameter variable suitable for 'this'.
virtual void addImplicitStructorParams(CodeGenFunction &CGF, QualType &ResTy, FunctionArgList &Params)=0
Insert any ABI-specific implicit parameters into the parameter list for a function.
virtual RecordArgABI getRecordArgABI(const CXXRecordDecl *RD) const =0
Returns how an argument of the given record type should be passed.
MangleContext & getMangleContext()
Gets the mangle context.
All available information about a concrete callee.
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.
ABIArgInfo & getReturnInfo()
bool isReturnsRetained() const
In ARC, whether this function retains its return value.
CanQualType getReturnType() const
bool isDelegateCall() const
unsigned getMaxVectorWidth() const
Return the maximum vector width in the arguments.
llvm::StructType * getArgStruct() const
Get the struct type used to represent all the arguments in memory.
void setHLSLFunctionAttributes(const FunctionDecl *FD, llvm::Function *Fn)
void emitEntryFunction(const FunctionDecl *FD, llvm::Function *Fn)
virtual void functionFinished(CodeGenFunction &CGF)
Cleans up references to the objects in finished function.
llvm::OpenMPIRBuilder & getOMPBuilder()
virtual void emitFunctionProlog(CodeGenFunction &CGF, const Decl *D)
Emits OpenMP-specific function prolog.
llvm::Value * getDiscriminator() const
virtual ~CGCapturedStmtInfo()
CGFPOptionsRAII(CodeGenFunction &CGF, FPOptions FPFeatures)
SanitizerScope(CodeGenFunction *CGF)
CodeGenFunction - This class organizes the per-function state that is used while generating LLVM code...
llvm::Value * EmitPointerAuthAuth(const CGPointerAuthInfo &Info, llvm::Value *Pointer)
void EmitDestructorBody(FunctionArgList &Args)
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...
void FinishFunction(SourceLocation EndLoc=SourceLocation())
FinishFunction - Complete IR generation of the current function.
void EmitNullInitialization(Address DestPtr, QualType Ty)
EmitNullInitialization - Generate code to set a value of the given type to null, If the type contains...
void EmitPointerAuthOperandBundle(const CGPointerAuthInfo &Info, SmallVectorImpl< llvm::OperandBundleDef > &Bundles)
GlobalDecl CurGD
CurGD - The GlobalDecl for the current function being compiled.
static TypeEvaluationKind getEvaluationKind(QualType T)
getEvaluationKind - Return the TypeEvaluationKind of QualType T.
static bool ContainsLabel(const Stmt *S, bool IgnoreCaseStmts=false)
ContainsLabel - Return true if the statement contains a label in it.
void EmitBranchOnBoolExpr(const Expr *Cond, llvm::BasicBlock *TrueBlock, llvm::BasicBlock *FalseBlock, uint64_t TrueCount, Stmt::Likelihood LH=Stmt::LH_None, const Expr *ConditionalOp=nullptr)
EmitBranchOnBoolExpr - Emit a branch on a boolean condition (e.g.
llvm::CallInst * EmitTrapCall(llvm::Intrinsic::ID IntrID)
Emit a call to trap or debugtrap and attach function attribute "trap-func-name" if specified.
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...
void EmitSanitizerStatReport(llvm::SanitizerStatKind SSK)
SanitizerSet SanOpts
Sanitizers enabled for this function.
void unprotectFromPeepholes(PeepholeProtection protection)
void EmitLambdaStaticInvokeBody(const CXXMethodDecl *MD)
bool ShouldInstrumentFunction()
ShouldInstrumentFunction - Return true if the current function should be instrumented with __cyg_prof...
Address EmitCompoundStmtWithoutScope(const CompoundStmt &S, bool GetLast=false, AggValueSlot AVS=AggValueSlot::ignored())
static bool hasScalarEvaluationKind(QualType T)
FieldDecl * LambdaThisCaptureField
LValue MakeNaturalAlignPointeeRawAddrLValue(llvm::Value *V, QualType T)
Same as MakeNaturalAlignPointeeAddrLValue except that the pointer is known to be unsigned.
void EmitKCFIOperandBundle(const CGCallee &Callee, SmallVectorImpl< llvm::OperandBundleDef > &Bundles)
void emitAlignmentAssumptionCheck(llvm::Value *Ptr, QualType Ty, SourceLocation Loc, SourceLocation AssumptionLoc, llvm::Value *Alignment, llvm::Value *OffsetValue, llvm::Value *TheCheck, llvm::Instruction *Assumption)
llvm::BlockAddress * GetAddrOfLabel(const LabelDecl *L)
RawAddress CreateDefaultAlignTempAlloca(llvm::Type *Ty, const Twine &Name="tmp")
CreateDefaultAlignedTempAlloca - This creates an alloca with the default ABI alignment of the given L...
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...
VlaSizePair getVLASize(const VariableArrayType *vla)
Returns an LLVM value that corresponds to the size, in non-variably-sized elements,...
void EmitEndEHSpec(const Decl *D)
EmitEndEHSpec - Emit the end of the exception spec.
bool CurFuncIsThunk
In C++, whether we are code generating a thunk.
void EmitAArch64MultiVersionResolver(llvm::Function *Resolver, ArrayRef< MultiVersionResolverOption > Options)
LValue EmitLValue(const Expr *E, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
EmitLValue - Emit code to compute a designator that specifies the location of the expression.
llvm::SmallVector< DeferredDeactivateCleanup > DeferredDeactivationCleanupStack
void EmitVariablyModifiedType(QualType Ty)
EmitVLASize - Capture all the sizes for the VLA expressions in the given variably-modified type and s...
llvm::BasicBlock * createBasicBlock(const Twine &name="", llvm::Function *parent=nullptr, llvm::BasicBlock *before=nullptr)
createBasicBlock - Create an LLVM basic block.
const LangOptions & getLangOpts() const
void EmitFunctionEpilog(const CGFunctionInfo &FI, bool EmitRetDbgLoc, SourceLocation EndLoc)
EmitFunctionEpilog - Emit the target specific LLVM code to return the given temporary.
llvm::Constant * EmitCheckTypeDescriptor(QualType T)
Emit a description of a type in a format suitable for passing to a runtime sanitizer handler.
llvm::BasicBlock * EHResumeBlock
EHResumeBlock - Unified block containing a call to llvm.eh.resume.
void EmitBlock(llvm::BasicBlock *BB, bool IsFinished=false)
EmitBlock - Emit the given block.
static bool isInstrumentedCondition(const Expr *C)
isInstrumentedCondition - Determine whether the given condition is an instrumentable condition (i....
void EmitX86MultiVersionResolver(llvm::Function *Resolver, ArrayRef< MultiVersionResolverOption > Options)
void EmitFunctionBody(const Stmt *Body)
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.
RValue EmitLoadOfLValue(LValue V, SourceLocation Loc)
EmitLoadOfLValue - Given an expression that represents a value lvalue, this method emits the address ...
@ 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.
void setCurrentProfileCount(uint64_t Count)
Set the profiler's current count.
void EmitIgnoredExpr(const Expr *E)
EmitIgnoredExpr - Emit an expression in a context which ignores the result.
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.
llvm::Type * ConvertTypeForMem(QualType T)
const Decl * CurCodeDecl
CurCodeDecl - This is the inner-most code context, which includes blocks.
llvm::AssertingVH< llvm::Instruction > AllocaInsertPt
AllocaInsertPoint - This is an instruction in the entry block before which we prefer to insert alloca...
bool AlwaysEmitXRayCustomEvents() const
AlwaysEmitXRayCustomEvents - Return true if we must unconditionally emit XRay custom event handling c...
JumpDest ReturnBlock
ReturnBlock - Unified return block.
void EmitVarAnnotations(const VarDecl *D, llvm::Value *V)
Emit local annotations for the local variable V, declared by D.
llvm::Value * EmitPointerAuthSign(const CGPointerAuthInfo &Info, llvm::Value *Pointer)
static const Expr * stripCond(const Expr *C)
Ignore parentheses and logical-NOT to track conditions consistently.
PeepholeProtection protectFromPeepholes(RValue rvalue)
protectFromPeepholes - Protect a value that we're intending to store to the side, but which will prob...
const TargetInfo & getTarget() const
llvm::DebugLoc SourceLocToDebugLoc(SourceLocation Location)
Converts Location to a DebugLoc, if debug information is enabled.
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 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 EmitCheck(ArrayRef< std::pair< llvm::Value *, SanitizerMask > > Checked, SanitizerHandler Check, ArrayRef< llvm::Constant * > StaticArgs, ArrayRef< llvm::Value * > DynamicArgs)
Create a basic block that will either trap or call a handler function in the UBSan runtime with the p...
void EmitBlockWithFallThrough(llvm::BasicBlock *BB, const Stmt *S)
bool ShouldSkipSanitizerInstrumentation()
ShouldSkipSanitizerInstrumentation - Return true if the current function should not be instrumented w...
uint64_t getCurrentProfileCount()
Get the profiler's current count.
SmallVector< const BinaryOperator *, 16 > MCDCLogOpStack
Stack to track the Logical Operator recursion nest for MC/DC.
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.
bool HaveInsertPoint() const
HaveInsertPoint - True if an insertion point is defined.
llvm::Constant * EmitCheckSourceLocation(SourceLocation Loc)
Emit a description of a source location in a format suitable for passing to a runtime sanitizer handl...
void markAsIgnoreThreadCheckingAtRuntime(llvm::Function *Fn)
Annotate the function with an attribute that disables TSan checking at runtime.
void ErrorUnsupported(const Stmt *S, const char *Type)
ErrorUnsupported - Print out an error that codegen doesn't support the specified stmt yet.
CGDebugInfo * getDebugInfo()
Address EmitVAListRef(const Expr *E)
void EmitBranch(llvm::BasicBlock *Block)
EmitBranch - Emit a branch to the specified basic block from the current insert block,...
void maybeCreateMCDCCondBitmap()
Allocate a temp value on the stack that MCDC can use to track condition results.
SmallVector< llvm::IntrinsicInst *, 4 > ConvergenceTokenStack
Stack to track the controlled convergence tokens.
void emitAlignmentAssumption(llvm::Value *PtrValue, QualType Ty, SourceLocation Loc, SourceLocation AssumptionLoc, llvm::Value *Alignment, llvm::Value *OffsetValue=nullptr)
llvm::Value * emitBoolVecConversion(llvm::Value *SrcVec, unsigned NumElementsDst, const llvm::Twine &Name="")
LValue MakeNaturalAlignRawAddrLValue(llvm::Value *V, QualType T)
bool ShouldXRayInstrumentFunction() const
ShouldXRayInstrument - Return true if the current function should be instrumented with XRay nop sleds...
void EmitStartEHSpec(const Decl *D)
EmitStartEHSpec - Emit the start of the exception spec.
void EmitMultiVersionResolver(llvm::Function *Resolver, ArrayRef< MultiVersionResolverOption > Options)
llvm::Value * EmitCheckValue(llvm::Value *V)
Convert a value into a format suitable for passing to a runtime sanitizer handler.
VlaSizePair getVLAElements1D(const VariableArrayType *vla)
Return the number of elements for a single dimension for the given array type.
bool AlwaysEmitXRayTypedEvents() const
AlwaysEmitXRayTypedEvents - Return true if clang must unconditionally emit XRay typed event handling ...
void EmitConstructorBody(FunctionArgList &Args)
void SetFastMathFlags(FPOptions FPFeatures)
Set the codegen fast-math flags.
ASTContext & getContext() const
const Decl * CurFuncDecl
CurFuncDecl - Holds the Decl for the current outermost non-closure context.
void maybeUpdateMCDCCondBitmap(const Expr *E, llvm::Value *Val)
Update the MCDC temp value with the condition's evaluated result.
void checkTargetFeatures(const CallExpr *E, const FunctionDecl *TargetDecl)
void EmitLambdaInAllocaCallOpBody(const CXXMethodDecl *MD)
llvm::SmallVector< char, 256 > LifetimeExtendedCleanupStack
void EmitDeclRefExprDbgValue(const DeclRefExpr *E, const APValue &Init)
void EmitCXXThrowExpr(const CXXThrowExpr *E, bool KeepInsertionPoint=true)
bool ConstantFoldsToSimpleInteger(const Expr *Cond, bool &Result, bool AllowLabels=false)
ConstantFoldsToSimpleInteger - If the specified expression does not fold to a constant,...
Address ReturnValuePointer
ReturnValuePointer - The temporary alloca to hold a pointer to sret.
llvm::ConstantInt * getUBSanFunctionTypeHash(QualType T) const
Return a type hash constant for a function instrumented by -fsanitize=function.
JumpDest getJumpDestForLabel(const LabelDecl *S)
getBasicBlockForLabel - Return the LLVM basicblock that the specified label maps to.
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="")
llvm::Type * ConvertType(QualType T)
CodeGenTypes & getTypes() const
bool IsSanitizerScope
True if CodeGen currently emits code implementing sanitizer checks.
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.
QualType BuildFunctionArgList(GlobalDecl GD, FunctionArgList &Args)
RawAddress NormalCleanupDest
i32s containing the indexes of the cleanup destinations.
llvm::Value * EvaluateExprAsBool(const Expr *E)
EvaluateExprAsBool - Perform the usual unary conversions on the specified expression and compare the ...
Address EmitMSVAListRef(const Expr *E)
Emit a "reference" to a __builtin_ms_va_list; this is always the value of the expression,...
VarBypassDetector Bypasses
EHScopeStack::stable_iterator PrologueCleanupDepth
PrologueCleanupDepth - The cleanup depth enclosing all the cleanups associated with the parameters.
static bool mightAddDeclToScope(const Stmt *S)
Determine if the given statement might introduce a declaration into the current scope,...
uint64_t getProfileCount(const Stmt *S)
Get the profiler's count for the given statement.
RawAddress CreateIRTemp(QualType T, const Twine &Name="tmp")
CreateIRTemp - Create a temporary IR object of the given type, with appropriate alignment.
void emitImplicitAssignmentOperatorBody(FunctionArgList &Args)
LValue MakeAddrLValue(Address Addr, QualType T, AlignmentSource Source=AlignmentSource::Type)
const CGFunctionInfo * CurFnInfo
LValue MakeNaturalAlignAddrLValue(llvm::Value *V, QualType T, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
void InsertHelper(llvm::Instruction *I, const llvm::Twine &Name, llvm::BasicBlock::iterator InsertPt) const
CGBuilder insert helper.
Address EmitFieldAnnotations(const FieldDecl *D, Address V)
Emit field annotations for the given field & value.
Address ReturnValue
ReturnValue - The temporary alloca to hold the return value.
void EmitStmt(const Stmt *S, ArrayRef< const Attr * > Attrs=std::nullopt)
EmitStmt - Emit the code for the statement.
void EnsureInsertPoint()
EnsureInsertPoint - Ensure that an insertion point is defined so that emitted IR has a place to go.
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 ...
llvm::LLVMContext & getLLVMContext()
bool SawAsmBlock
Whether we processed a Microsoft-style asm block during CodeGen.
llvm::Value * EmitScalarExpr(const Expr *E, bool IgnoreResultAssign=false)
EmitScalarExpr - Emit the computation of the specified expression of LLVM scalar type,...
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.
llvm::Value * EmitAnnotationCall(llvm::Function *AnnotationFn, llvm::Value *AnnotatedVal, StringRef AnnotationStr, SourceLocation Location, const AnnotateAttr *Attr)
Emit an annotation call (intrinsic).
llvm::BasicBlock * GetIndirectGotoBlock()
llvm::Type * convertTypeForLoadStore(QualType ASTTy, llvm::Type *LLVMTy=nullptr)
llvm::DebugLoc EmitReturnBlock()
Emit the unified return block, trying to avoid its emission when possible.
void GenerateCode(GlobalDecl GD, llvm::Function *Fn, const CGFunctionInfo &FnInfo)
LValue EmitLValueForLambdaField(const FieldDecl *Field)
static bool containsBreak(const Stmt *S)
containsBreak - Return true if the statement contains a break out of it.
This class organizes the cross-function state that is used while generating LLVM code.
CGHLSLRuntime & getHLSLRuntime()
Return a reference to the configured HLSL runtime.
llvm::Constant * EmitAnnotationArgs(const AnnotateAttr *Attr)
Emit additional args of the annotation.
llvm::Module & getModule() const
DiagnosticsEngine & getDiags() const
void ErrorUnsupported(const Stmt *S, const char *Type)
Print out an error that codegen doesn't support the specified stmt yet.
const LangOptions & getLangOpts() const
CGCUDARuntime & getCUDARuntime()
Return a reference to the configured CUDA runtime.
llvm::Constant * EmitAnnotationLineNo(SourceLocation L)
Emit the annotation line number.
CharUnits getNaturalTypeAlignment(QualType T, LValueBaseInfo *BaseInfo=nullptr, TBAAAccessInfo *TBAAInfo=nullptr, bool forPointeeType=false)
CodeGenTypes & getTypes()
const llvm::DataLayout & getDataLayout() const
bool shouldEmitConvergenceTokens() const
CGCXXABI & getCXXABI() const
CGOpenMPRuntime & getOpenMPRuntime()
Return a reference to the configured OpenMP runtime.
bool imbueXRayAttrs(llvm::Function *Fn, SourceLocation Loc, StringRef Category=StringRef()) const
Imbue XRay attributes to a function, applying the always/never attribute lists in the process.
ProfileList::ExclusionType isFunctionBlockedFromProfileInstr(llvm::Function *Fn, SourceLocation Loc) const
ASTContext & getContext() const
llvm::SanitizerStatReport & getSanStats()
llvm::Constant * EmitAnnotationString(StringRef Str)
Emit an annotation string.
const TargetCodeGenInfo & getTargetCodeGenInfo()
const CodeGenOptions & getCodeGenOpts() const
llvm::LLVMContext & getLLVMContext()
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::Constant * EmitNullConstant(QualType T)
Return the result of value-initializing the given type, i.e.
llvm::ConstantInt * CreateKCFITypeId(QualType T)
Generate a KCFI type identifier for T.
llvm::Function * getIntrinsic(unsigned IID, ArrayRef< llvm::Type * > Tys=std::nullopt)
bool MayDropFunctionReturn(const ASTContext &Context, QualType ReturnType) const
Whether this function's return type has no side effects, and thus may be trivially discarded if it is...
llvm::Constant * EmitAnnotationUnit(SourceLocation Loc)
Emit the annotation's translation unit.
llvm::ConstantInt * getSize(CharUnits numChars)
Emit the given number of characters as a value of type size_t.
void assignRegionCounters(GlobalDecl GD, llvm::Function *Fn)
Assign counters to regions and configure them for PGO of a given function.
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...
llvm::Type * convertTypeForLoadStore(QualType T, llvm::Type *LLVMTy=nullptr)
Given that T is a scalar type, return the IR type that should be used for load and store operations.
llvm::Type * ConvertTypeForMem(QualType T)
ConvertTypeForMem - Convert type T into a llvm::Type.
bool isZeroInitializable(QualType T)
IsZeroInitializable - Return whether a type can be zero-initialized (in the C++ sense) with an LLVM z...
stable_iterator stable_begin() const
Create a stable reference to the top of the EH stack.
bool containsOnlyLifetimeMarkers(stable_iterator Old) const
bool empty() const
Determines whether the exception-scopes stack is empty.
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
void InsertHelper(llvm::Instruction *I) const
Function called by the CodeGenFunction when an instruction is created.
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.
llvm::Value * getPointer() const
virtual void checkFunctionABI(CodeGenModule &CGM, const FunctionDecl *Decl) const
Any further codegen related checks that need to be done on a function signature in a target specific ...
virtual llvm::Constant * getUBSanFunctionSignature(CodeGen::CodeGenModule &CGM) const
Return a constant used by UBSan as a signature to identify functions possessing type information,...
void Init(const Stmt *Body)
Clear the object and pre-process for the given statement, usually function body statement.
CompoundStmt - This represents a group of statements like { stmt stmt }.
ConditionalOperator - The ?: ternary operator.
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
DiagnosticBuilder Report(SourceLocation Loc, unsigned DiagID)
Issue the message to the client.
bool isIgnored(unsigned DiagID, SourceLocation Loc) const
Determine whether the diagnostic is known to be ignored.
This represents one expression.
bool EvaluateAsInt(EvalResult &Result, const ASTContext &Ctx, SideEffectsKind AllowSideEffects=SE_NoSideEffects, bool InConstantContext=false) const
EvaluateAsInt - Return true if this is a constant which we can fold and convert to an integer,...
FPOptions getFPFeaturesInEffect(const LangOptions &LO) const
Returns the set of floating point options that apply to this expression.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
Expr * IgnoreImpCasts() LLVM_READONLY
Skip past any implicit casts which might surround this expression until reaching a fixed point.
SourceLocation getExprLoc() const LLVM_READONLY
getExprLoc - Return the preferred location for the arrow when diagnosing a problem with a generic exp...
std::optional< llvm::APSInt > getIntegerConstantExpr(const ASTContext &Ctx, SourceLocation *Loc=nullptr) const
isIntegerConstantExpr - Return the value if this expression is a valid integer constant expression.
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...
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.
Represents a prototype with parameter type info, e.g.
GlobalDecl - represents a global declaration.
CXXCtorType getCtorType() 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
virtual void mangleCanonicalTypeName(QualType T, raw_ostream &, bool NormalizeIntegers=false)=0
Generates a unique string for an externally visible type for use with TBAA or type uniquing.
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
A helper class that allows the use of isa/cast/dyncast to detect TagType objects of structs/unions/cl...
decl_type * getPreviousDecl()
Return the previous declaration of this declaration or NULL if this is the first declaration.
Encodes a location in the source.
A trivial tuple used to represent a source range.
SourceLocation getEnd() const
SourceLocation getBegin() const
Stmt - This represents one statement.
StmtClass getStmtClass() const
SourceRange getSourceRange() const LLVM_READONLY
SourceLocation tokens are not useful in isolation - they are low level value objects created/interpre...
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.
SourceLocation getBeginLoc() const LLVM_READONLY
bool isMicrosoft() const
Is this ABI an MSVC-compatible ABI?
const llvm::Triple & getTriple() const
Returns the target triple of the primary target.
bool supportsIFunc() const
Identify whether this target supports IFuncs.
TargetCXXABI getCXXABI() const
Get the C++ ABI currently in use.
virtual std::optional< std::pair< unsigned, unsigned > > getVScaleRange(const LangOptions &LangOpts) const
Returns target-specific min and max values VScale_Range.
The base class of the type hierarchy.
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
bool isSignedIntegerType() const
Return true if this is an integer type that is signed, according to C99 6.2.5p4 [char,...
bool isPointerType() const
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 isRecordType() const
bool isObjCRetainableType() const
std::optional< NullabilityKind > getNullability() const
Determine the nullability of the given type.
bool isFunctionNoProtoType() 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.
TypeEvaluationKind
The kind of evaluation to perform on values of a particular type.
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
Matches all kinds of arrays.
bool Zero(InterpState &S, CodePtr OpPC)
The JSON file list parser is used to communicate input to InstallAPI.
@ NonNull
Values of this type can never be null.
bool isLambdaCallOperator(const CXXMethodDecl *MD)
@ Result
The result type of a method or function.
const FunctionProtoType * T
llvm::fp::ExceptionBehavior ToConstrainedExceptMD(LangOptions::FPExceptionModeKind Kind)
@ Other
Other implicit parameter.
@ EST_None
no exception specification
@ Implicit
An implicit conversion.
Diagnostic wrappers for TextAPI types for error reporting.
cl::opt< bool > EnableSingleByteCoverage
llvm::BasicBlock * getBlock() const
This structure provides a set of types that are commonly used during IR emission.
llvm::PointerType * ConstGlobalsPtrTy
void* in the address space for constant globals
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.
Contains information gathered from parsing the contents of TargetAttr.
bool ReturnAddresses
Should return addresses be authenticated?
PointerAuthSchema FunctionPointers
The ABI for C function pointers.
bool AuthTraps
Do authentication failures cause a trap?
bool IndirectGotos
Do indirect goto label addresses need to be authenticated?
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.
SanitizerMask Mask
Bitmask of enabled sanitizers.
bool hasOneOf(SanitizerMask K) const
Check if one or more sanitizers are enabled.
bool has(XRayInstrMask K) const