32#include "llvm/ADT/APFixedPoint.h"
33#include "llvm/IR/CFG.h"
34#include "llvm/IR/Constants.h"
35#include "llvm/IR/DataLayout.h"
36#include "llvm/IR/DerivedTypes.h"
37#include "llvm/IR/FixedPointBuilder.h"
38#include "llvm/IR/Function.h"
39#include "llvm/IR/GEPNoWrapFlags.h"
40#include "llvm/IR/GetElementPtrTypeIterator.h"
41#include "llvm/IR/GlobalVariable.h"
42#include "llvm/IR/Intrinsics.h"
43#include "llvm/IR/IntrinsicsPowerPC.h"
44#include "llvm/IR/MatrixBuilder.h"
45#include "llvm/IR/Module.h"
46#include "llvm/Support/TypeSize.h"
51using namespace CodeGen;
69bool mayHaveIntegerOverflow(llvm::ConstantInt *LHS, llvm::ConstantInt *RHS,
71 llvm::APInt &Result) {
74 const auto &LHSAP = LHS->getValue();
75 const auto &RHSAP = RHS->getValue();
76 if (Opcode == BO_Add) {
77 Result =
Signed ? LHSAP.sadd_ov(RHSAP, Overflow)
78 : LHSAP.uadd_ov(RHSAP, Overflow);
79 }
else if (Opcode == BO_Sub) {
80 Result =
Signed ? LHSAP.ssub_ov(RHSAP, Overflow)
81 : LHSAP.usub_ov(RHSAP, Overflow);
82 }
else if (Opcode == BO_Mul) {
83 Result =
Signed ? LHSAP.smul_ov(RHSAP, Overflow)
84 : LHSAP.umul_ov(RHSAP, Overflow);
85 }
else if (Opcode == BO_Div || Opcode == BO_Rem) {
86 if (
Signed && !RHS->isZero())
87 Result = LHSAP.sdiv_ov(RHSAP, Overflow);
103 bool mayHaveIntegerOverflow()
const {
105 auto *LHSCI = dyn_cast<llvm::ConstantInt>(LHS);
106 auto *RHSCI = dyn_cast<llvm::ConstantInt>(RHS);
107 if (!LHSCI || !RHSCI)
111 return ::mayHaveIntegerOverflow(
116 bool isDivremOp()
const {
122 bool mayHaveIntegerDivisionByZero()
const {
124 if (
auto *CI = dyn_cast<llvm::ConstantInt>(RHS))
130 bool mayHaveFloatDivisionByZero()
const {
132 if (
auto *CFP = dyn_cast<llvm::ConstantFP>(RHS))
133 return CFP->isZero();
140 bool isFixedPointOp()
const {
143 if (
const auto *BinOp = dyn_cast<BinaryOperator>(E)) {
144 QualType LHSType = BinOp->getLHS()->getType();
145 QualType RHSType = BinOp->getRHS()->getType();
148 if (
const auto *UnOp = dyn_cast<UnaryOperator>(E))
149 return UnOp->getSubExpr()->getType()->isFixedPointType();
154 bool rhsHasSignedIntegerRepresentation()
const {
155 if (
const auto *BinOp = dyn_cast<BinaryOperator>(E)) {
156 QualType RHSType = BinOp->getRHS()->getType();
163static bool MustVisitNullValue(
const Expr *
E) {
171static std::optional<QualType> getUnwidenedIntegerType(
const ASTContext &Ctx,
187 return getUnwidenedIntegerType(Ctx,
E).has_value();
191static bool CanElideOverflowCheck(
const ASTContext &Ctx,
const BinOpInfo &Op) {
192 assert((isa<UnaryOperator>(Op.E) || isa<BinaryOperator>(Op.E)) &&
193 "Expected a unary or binary operator");
197 if (!Op.mayHaveIntegerOverflow())
200 if (Op.Ty->isSignedIntegerType() &&
206 if (Op.Ty->isUnsignedIntegerType() &&
216 LangOptions::OverflowPatternExclusionKind::NegUnsignedConst) &&
226 const auto *BO = cast<BinaryOperator>(Op.E);
227 if (BO->hasExcludedOverflowPattern())
230 auto OptionalLHSTy = getUnwidenedIntegerType(Ctx, BO->getLHS());
234 auto OptionalRHSTy = getUnwidenedIntegerType(Ctx, BO->getRHS());
243 if ((Op.Opcode != BO_Mul && Op.Opcode != BO_MulAssign) ||
249 unsigned PromotedSize = Ctx.
getTypeSize(Op.E->getType());
250 return (2 * Ctx.
getTypeSize(LHSTy)) < PromotedSize ||
254class ScalarExprEmitter
258 bool IgnoreResultAssign;
259 llvm::LLVMContext &VMContext;
263 : CGF(cgf), Builder(CGF.Builder), IgnoreResultAssign(ira),
264 VMContext(cgf.getLLVMContext()) {
271 bool TestAndClearIgnoreResultAssign() {
272 bool I = IgnoreResultAssign;
273 IgnoreResultAssign =
false;
279 LValue EmitCheckedLValue(
const Expr *
E, CodeGenFunction::TypeCheckKind TCK) {
284 ArrayRef<std::pair<Value *, SanitizerKind::SanitizerOrdinal>> Checks,
285 const BinOpInfo &Info);
291 void EmitLValueAlignmentAssumption(
const Expr *
E,
Value *
V) {
292 const AlignValueAttr *AVAttr =
nullptr;
293 if (
const auto *DRE = dyn_cast<DeclRefExpr>(
E)) {
297 if (
const auto *TTy =
299 AVAttr = TTy->getDecl()->
getAttr<AlignValueAttr>();
306 if (isa<ParmVarDecl>(VD) && !CGF.
SanOpts.
has(SanitizerKind::Alignment))
309 AVAttr = VD->
getAttr<AlignValueAttr>();
315 AVAttr = TTy->getDecl()->getAttr<AlignValueAttr>();
321 llvm::ConstantInt *AlignmentCI = cast<llvm::ConstantInt>(AlignmentValue);
329 Value *
V = EmitLoadOfLValue(EmitCheckedLValue(
E, CodeGenFunction::TCK_Load),
332 EmitLValueAlignmentAssumption(
E,
V);
342 void EmitFloatConversionCheck(
Value *OrigSrc,
QualType OrigSrcType,
349 enum ImplicitConversionCheckKind :
unsigned char {
350 ICCK_IntegerTruncation = 0,
351 ICCK_UnsignedIntegerTruncation = 1,
352 ICCK_SignedIntegerTruncation = 2,
353 ICCK_IntegerSignChange = 3,
354 ICCK_SignedIntegerTruncationOrSignChange = 4,
370 struct ScalarConversionOpts {
371 bool TreatBooleanAsSigned;
372 bool EmitImplicitIntegerTruncationChecks;
373 bool EmitImplicitIntegerSignChangeChecks;
375 ScalarConversionOpts()
376 : TreatBooleanAsSigned(
false),
377 EmitImplicitIntegerTruncationChecks(
false),
378 EmitImplicitIntegerSignChangeChecks(
false) {}
381 : TreatBooleanAsSigned(
false),
382 EmitImplicitIntegerTruncationChecks(
384 EmitImplicitIntegerSignChangeChecks(
388 llvm::Type *SrcTy, llvm::Type *DstTy,
389 ScalarConversionOpts Opts);
393 ScalarConversionOpts Opts = ScalarConversionOpts());
402 Value *EmitComplexToScalarConversion(CodeGenFunction::ComplexPairTy Src,
412 llvm::Value *
Zero = llvm::Constant::getNullValue(
V->getType());
413 return Builder.CreateFCmpUNE(
V, Zero,
"tobool");
420 return Builder.CreateICmpNE(
V, Zero,
"tobool");
427 if (llvm::ZExtInst *ZI = dyn_cast<llvm::ZExtInst>(
V)) {
428 if (ZI->getOperand(0)->getType() == Builder.getInt1Ty()) {
429 Value *Result = ZI->getOperand(0);
434 ZI->eraseFromParent();
439 return Builder.CreateIsNotNull(
V,
"tobool");
453 llvm_unreachable(
"Stmt can't have complex result type!");
471 return Visit(
E->getSubExpr());
477 return Visit(
E->getReplacement());
480 return Visit(
GE->getResultExpr());
489 return Visit(
E->getSubExpr());
494 return Builder.getInt(
E->getValue());
497 return Builder.getInt(
E->getValue());
500 return llvm::ConstantFP::get(VMContext,
E->getValue());
503 return llvm::ConstantInt::get(ConvertType(
E->
getType()),
E->getValue());
506 return llvm::ConstantInt::get(ConvertType(
E->
getType()),
E->getValue());
509 return llvm::ConstantInt::get(ConvertType(
E->
getType()),
E->getValue());
524 return Builder.CreateBitCast(
V, ConvertType(
E->
getType()));
528 return llvm::ConstantInt::get(ConvertType(
E->
getType()),
E->getPackLength());
548 llvm_unreachable(
"Codegen for this isn't defined/implemented");
555 return EmitLoadOfLValue(
E);
565 return EmitLoadOfLValue(
E);
568 if (
E->getMethodDecl() &&
569 E->getMethodDecl()->getReturnType()->isReferenceType())
570 return EmitLoadOfLValue(
E);
581 VersionTuple Version =
E->getVersion();
586 return llvm::ConstantInt::get(Builder.getInt1Ty(), 1);
596 Value *VisitExtVectorElementExpr(
Expr *
E) {
return EmitLoadOfLValue(
E); }
603 return EmitLoadOfLValue(
E);
610 "ArrayInitIndexExpr not inside an ArrayInitLoopExpr?");
619 return VisitCastExpr(
E);
624 if (
E->getCallReturnType(CGF.
getContext())->isReferenceType())
625 return EmitLoadOfLValue(
E);
629 EmitLValueAlignmentAssumption(
E,
V);
637 LValue LV = EmitLValue(
E->getSubExpr());
638 return EmitScalarPrePostIncDec(
E, LV,
false,
false);
641 LValue LV = EmitLValue(
E->getSubExpr());
642 return EmitScalarPrePostIncDec(
E, LV,
true,
false);
645 LValue LV = EmitLValue(
E->getSubExpr());
646 return EmitScalarPrePostIncDec(
E, LV,
false,
true);
649 LValue LV = EmitLValue(
E->getSubExpr());
650 return EmitScalarPrePostIncDec(
E, LV,
true,
true);
653 llvm::Value *EmitIncDecConsiderOverflowBehavior(
const UnaryOperator *
E,
658 bool isInc,
bool isPre);
662 if (isa<MemberPointerType>(
E->
getType()))
665 return EmitLValue(
E->getSubExpr()).getPointer(CGF);
669 return Visit(
E->getSubExpr());
670 return EmitLoadOfLValue(
E);
689 return Visit(
E->getSubExpr());
694 return EmitLoadOfLValue(
E);
705 CodeGenFunction::CXXDefaultArgExprScope
Scope(CGF, DAE);
709 CodeGenFunction::CXXDefaultInitExprScope
Scope(CGF, DIE);
726 return llvm::ConstantInt::get(ConvertType(
E->
getType()),
E->getValue());
730 return Builder.getInt1(
E->isSatisfied());
734 return Builder.getInt1(
E->isSatisfied());
738 return llvm::ConstantInt::get(ConvertType(
E->
getType()),
E->getValue());
742 return llvm::ConstantInt::get(Builder.getInt1Ty(),
E->getValue());
765 return Builder.getInt1(
E->getValue());
769 Value *EmitMul(
const BinOpInfo &Ops) {
770 if (Ops.Ty->isSignedIntegerOrEnumerationType()) {
771 switch (CGF.
getLangOpts().getSignedOverflowBehavior()) {
773 if (!CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow))
774 return Builder.CreateMul(Ops.LHS, Ops.RHS,
"mul");
777 if (!CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow))
778 return Builder.CreateNSWMul(Ops.LHS, Ops.RHS,
"mul");
781 if (CanElideOverflowCheck(CGF.
getContext(), Ops))
782 return Builder.CreateNSWMul(Ops.LHS, Ops.RHS,
"mul");
783 return EmitOverflowCheckedBinOp(Ops);
787 if (Ops.Ty->isConstantMatrixType()) {
788 llvm::MatrixBuilder MB(Builder);
791 auto *BO = cast<BinaryOperator>(Ops.E);
792 auto *LHSMatTy = dyn_cast<ConstantMatrixType>(
793 BO->getLHS()->getType().getCanonicalType());
794 auto *RHSMatTy = dyn_cast<ConstantMatrixType>(
795 BO->getRHS()->getType().getCanonicalType());
796 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, Ops.FPFeatures);
797 if (LHSMatTy && RHSMatTy)
798 return MB.CreateMatrixMultiply(Ops.LHS, Ops.RHS, LHSMatTy->getNumRows(),
799 LHSMatTy->getNumColumns(),
800 RHSMatTy->getNumColumns());
801 return MB.CreateScalarMultiply(Ops.LHS, Ops.RHS);
804 if (Ops.Ty->isUnsignedIntegerType() &&
805 CGF.
SanOpts.
has(SanitizerKind::UnsignedIntegerOverflow) &&
806 !CanElideOverflowCheck(CGF.
getContext(), Ops))
807 return EmitOverflowCheckedBinOp(Ops);
809 if (Ops.LHS->getType()->isFPOrFPVectorTy()) {
811 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, Ops.FPFeatures);
812 return Builder.CreateFMul(Ops.LHS, Ops.RHS,
"mul");
814 if (Ops.isFixedPointOp())
815 return EmitFixedPointBinOp(Ops);
816 return Builder.CreateMul(Ops.LHS, Ops.RHS,
"mul");
820 Value *EmitOverflowCheckedBinOp(
const BinOpInfo &Ops);
823 void EmitUndefinedBehaviorIntegerDivAndRemCheck(
const BinOpInfo &Ops,
824 llvm::Value *Zero,
bool isDiv);
826 static Value *GetMaximumShiftAmount(
Value *LHS,
Value *RHS,
bool RHSIsSigned);
832 Value *EmitDiv(
const BinOpInfo &Ops);
833 Value *EmitRem(
const BinOpInfo &Ops);
834 Value *EmitAdd(
const BinOpInfo &Ops);
835 Value *EmitSub(
const BinOpInfo &Ops);
836 Value *EmitShl(
const BinOpInfo &Ops);
837 Value *EmitShr(
const BinOpInfo &Ops);
838 Value *EmitAnd(
const BinOpInfo &Ops) {
839 return Builder.CreateAnd(Ops.LHS, Ops.RHS,
"and");
841 Value *EmitXor(
const BinOpInfo &Ops) {
842 return Builder.CreateXor(Ops.LHS, Ops.RHS,
"xor");
844 Value *EmitOr (
const BinOpInfo &Ops) {
845 return Builder.CreateOr(Ops.LHS, Ops.RHS,
"or");
849 Value *EmitFixedPointBinOp(
const BinOpInfo &Ops);
859 Value *(ScalarExprEmitter::*F)(
const BinOpInfo &),
863 Value *(ScalarExprEmitter::*F)(
const BinOpInfo &));
868 QualType ElementType = CT->getElementType();
875 unsigned NumElements = VT->getNumElements();
885#define HANDLEBINOP(OP) \
886 Value *VisitBin##OP(const BinaryOperator *E) { \
887 QualType promotionTy = getPromotionType(E->getType()); \
888 auto result = Emit##OP(EmitBinOps(E, promotionTy)); \
889 if (result && !promotionTy.isNull()) \
890 result = EmitUnPromotedValue(result, E->getType()); \
893 Value *VisitBin##OP##Assign(const CompoundAssignOperator *E) { \
894 return EmitCompoundAssign(E, &ScalarExprEmitter::Emit##OP); \
910 llvm::CmpInst::Predicate SICmpOpc,
911 llvm::CmpInst::Predicate FCmpOpc,
bool IsSignaling);
912#define VISITCOMP(CODE, UI, SI, FP, SIG) \
913 Value *VisitBin##CODE(const BinaryOperator *E) { \
914 return EmitCompare(E, llvm::ICmpInst::UI, llvm::ICmpInst::SI, \
915 llvm::FCmpInst::FP, SIG); }
930 Value *VisitBinPtrMemD(
const Expr *
E) {
return EmitLoadOfLValue(
E); }
931 Value *VisitBinPtrMemI(
const Expr *
E) {
return EmitLoadOfLValue(
E); }
934 return Visit(
E->getSemanticForm());
957 return Visit(
E->getSelectedExpr());
969 assert(SrcType.
isCanonical() &&
"EmitScalarConversion strips typedefs");
972 return EmitFloatToBoolConversion(Src);
978 "Unknown scalar type to convert");
980 if (isa<llvm::IntegerType>(Src->
getType()))
981 return EmitIntToBoolConversion(Src);
983 assert(isa<llvm::PointerType>(Src->
getType()));
984 return EmitPointerToBoolConversion(Src, SrcType);
987void ScalarExprEmitter::EmitFloatConversionCheck(
990 assert(SrcType->
isFloatingType() &&
"not a conversion from floating point");
991 if (!isa<llvm::IntegerType>(DstTy))
994 CodeGenFunction::SanitizerScope SanScope(&CGF);
998 llvm::Value *Check =
nullptr;
999 const llvm::fltSemantics &SrcSema =
1009 APFloat MinSrc(SrcSema, APFloat::uninitialized);
1010 if (MinSrc.convertFromAPInt(
Min, !
Unsigned, APFloat::rmTowardZero) &
1011 APFloat::opOverflow)
1014 MinSrc = APFloat::getInf(SrcSema,
true);
1018 MinSrc.subtract(
APFloat(SrcSema, 1), APFloat::rmTowardNegative);
1021 APFloat MaxSrc(SrcSema, APFloat::uninitialized);
1022 if (MaxSrc.convertFromAPInt(
Max, !
Unsigned, APFloat::rmTowardZero) &
1023 APFloat::opOverflow)
1026 MaxSrc = APFloat::getInf(SrcSema,
false);
1030 MaxSrc.add(
APFloat(SrcSema, 1), APFloat::rmTowardPositive);
1035 const llvm::fltSemantics &
Sema =
1038 MinSrc.convert(
Sema, APFloat::rmTowardZero, &IsInexact);
1039 MaxSrc.convert(
Sema, APFloat::rmTowardZero, &IsInexact);
1043 Builder.CreateFCmpOGT(Src, llvm::ConstantFP::get(VMContext, MinSrc));
1045 Builder.CreateFCmpOLT(Src, llvm::ConstantFP::get(VMContext, MaxSrc));
1046 Check = Builder.CreateAnd(GE, LE);
1051 CGF.
EmitCheck(std::make_pair(Check, SanitizerKind::SO_FloatCastOverflow),
1052 SanitizerHandler::FloatCastOverflow, StaticArgs, OrigSrc);
1057static std::pair<ScalarExprEmitter::ImplicitConversionCheckKind,
1058 std::pair<llvm::Value *, SanitizerKind::SanitizerOrdinal>>
1061 llvm::Type *SrcTy = Src->
getType();
1062 llvm::Type *DstTy = Dst->
getType();
1067 assert(SrcTy->getScalarSizeInBits() > Dst->
getType()->getScalarSizeInBits());
1068 assert(isa<llvm::IntegerType>(SrcTy) && isa<llvm::IntegerType>(DstTy) &&
1069 "non-integer llvm type");
1076 ScalarExprEmitter::ImplicitConversionCheckKind Kind;
1078 if (!SrcSigned && !DstSigned) {
1079 Kind = ScalarExprEmitter::ICCK_UnsignedIntegerTruncation;
1080 Ordinal = SanitizerKind::SO_ImplicitUnsignedIntegerTruncation;
1082 Kind = ScalarExprEmitter::ICCK_SignedIntegerTruncation;
1083 Ordinal = SanitizerKind::SO_ImplicitSignedIntegerTruncation;
1086 llvm::Value *Check =
nullptr;
1088 Check = Builder.CreateIntCast(Dst, SrcTy, DstSigned,
"anyext");
1090 Check = Builder.CreateICmpEQ(Check, Src,
"truncheck");
1092 return std::make_pair(Kind, std::make_pair(Check, Ordinal));
1100void ScalarExprEmitter::EmitIntegerTruncationCheck(
Value *Src,
QualType SrcType,
1112 unsigned SrcBits = Src->
getType()->getScalarSizeInBits();
1113 unsigned DstBits = Dst->
getType()->getScalarSizeInBits();
1115 if (SrcBits <= DstBits)
1118 assert(!DstType->
isBooleanType() &&
"we should not get here with booleans.");
1125 if (CGF.
SanOpts.
has(SanitizerKind::ImplicitIntegerSignChange) &&
1126 (!SrcSigned && DstSigned))
1129 CodeGenFunction::SanitizerScope SanScope(&CGF);
1131 std::pair<ScalarExprEmitter::ImplicitConversionCheckKind,
1132 std::pair<llvm::Value *, SanitizerKind::SanitizerOrdinal>>
1146 llvm::Constant *StaticArgs[] = {
1149 llvm::ConstantInt::get(Builder.getInt8Ty(), Check.first),
1150 llvm::ConstantInt::get(Builder.getInt32Ty(), 0)};
1152 CGF.
EmitCheck(Check.second, SanitizerHandler::ImplicitConversion, StaticArgs,
1160 llvm::Type *VTy =
V->getType();
1163 return llvm::ConstantInt::getFalse(VTy->getContext());
1165 llvm::Constant *Zero = llvm::ConstantInt::get(VTy, 0);
1166 return Builder.CreateICmp(llvm::ICmpInst::ICMP_SLT,
V, Zero,
1167 llvm::Twine(Name) +
"." +
V->getName() +
1168 ".negativitycheck");
1173static std::pair<ScalarExprEmitter::ImplicitConversionCheckKind,
1174 std::pair<llvm::Value *, SanitizerKind::SanitizerOrdinal>>
1177 llvm::Type *SrcTy = Src->
getType();
1178 llvm::Type *DstTy = Dst->
getType();
1180 assert(isa<llvm::IntegerType>(SrcTy) && isa<llvm::IntegerType>(DstTy) &&
1181 "non-integer llvm type");
1187 unsigned SrcBits = SrcTy->getScalarSizeInBits();
1188 unsigned DstBits = DstTy->getScalarSizeInBits();
1192 assert(((SrcBits != DstBits) || (SrcSigned != DstSigned)) &&
1193 "either the widths should be different, or the signednesses.");
1196 llvm::Value *SrcIsNegative =
1199 llvm::Value *DstIsNegative =
1205 llvm::Value *Check =
nullptr;
1206 Check = Builder.CreateICmpEQ(SrcIsNegative, DstIsNegative,
"signchangecheck");
1208 return std::make_pair(
1209 ScalarExprEmitter::ICCK_IntegerSignChange,
1210 std::make_pair(Check, SanitizerKind::SO_ImplicitIntegerSignChange));
1213void ScalarExprEmitter::EmitIntegerSignChangeCheck(
Value *Src,
QualType SrcType,
1216 if (!CGF.
SanOpts.
has(SanitizerKind::SO_ImplicitIntegerSignChange))
1219 llvm::Type *SrcTy = Src->
getType();
1220 llvm::Type *DstTy = Dst->
getType();
1230 unsigned SrcBits = SrcTy->getScalarSizeInBits();
1231 unsigned DstBits = DstTy->getScalarSizeInBits();
1238 if (SrcSigned == DstSigned && SrcBits == DstBits)
1242 if (!SrcSigned && !DstSigned)
1247 if ((DstBits > SrcBits) && DstSigned)
1249 if (CGF.
SanOpts.
has(SanitizerKind::ImplicitSignedIntegerTruncation) &&
1250 (SrcBits > DstBits) && SrcSigned) {
1259 SanitizerKind::ImplicitSignedIntegerTruncation, DstType))
1263 SanitizerKind::ImplicitUnsignedIntegerTruncation, DstType))
1267 CodeGenFunction::SanitizerScope SanScope(&CGF);
1269 std::pair<ScalarExprEmitter::ImplicitConversionCheckKind,
1270 std::pair<llvm::Value *, SanitizerKind::SanitizerOrdinal>>
1274 ImplicitConversionCheckKind CheckKind;
1282 CheckKind = Check.first;
1283 Checks.emplace_back(Check.second);
1285 if (CGF.
SanOpts.
has(SanitizerKind::ImplicitSignedIntegerTruncation) &&
1286 (SrcBits > DstBits) && !SrcSigned && DstSigned) {
1292 CheckKind = ICCK_SignedIntegerTruncationOrSignChange;
1293 Checks.emplace_back(Check.second);
1297 llvm::Constant *StaticArgs[] = {
1300 llvm::ConstantInt::get(Builder.getInt8Ty(), CheckKind),
1301 llvm::ConstantInt::get(Builder.getInt32Ty(), 0)};
1303 CGF.
EmitCheck(Checks, SanitizerHandler::ImplicitConversion, StaticArgs,
1309static std::pair<ScalarExprEmitter::ImplicitConversionCheckKind,
1310 std::pair<llvm::Value *, SanitizerKind::SanitizerOrdinal>>
1316 ScalarExprEmitter::ImplicitConversionCheckKind Kind;
1317 if (!SrcSigned && !DstSigned)
1318 Kind = ScalarExprEmitter::ICCK_UnsignedIntegerTruncation;
1320 Kind = ScalarExprEmitter::ICCK_SignedIntegerTruncation;
1322 llvm::Value *Check =
nullptr;
1324 Check = Builder.CreateIntCast(Dst, Src->
getType(), DstSigned,
"bf.anyext");
1326 Check = Builder.CreateICmpEQ(Check, Src,
"bf.truncheck");
1329 return std::make_pair(
1331 std::make_pair(Check, SanitizerKind::SO_ImplicitBitfieldConversion));
1336static std::pair<ScalarExprEmitter::ImplicitConversionCheckKind,
1337 std::pair<llvm::Value *, SanitizerKind::SanitizerOrdinal>>
1341 llvm::Value *SrcIsNegative =
1344 llvm::Value *DstIsNegative =
1350 llvm::Value *Check =
nullptr;
1352 Builder.CreateICmpEQ(SrcIsNegative, DstIsNegative,
"bf.signchangecheck");
1354 return std::make_pair(
1355 ScalarExprEmitter::ICCK_IntegerSignChange,
1356 std::make_pair(Check, SanitizerKind::SO_ImplicitBitfieldConversion));
1359void CodeGenFunction::EmitBitfieldConversionCheck(
Value *Src,
QualType SrcType,
1364 if (!
SanOpts.
has(SanitizerKind::ImplicitBitfieldConversion))
1377 assert(isa<llvm::IntegerType>(Src->
getType()) &&
1378 isa<llvm::IntegerType>(Dst->
getType()) &&
"non-integer llvm type");
1382 unsigned SrcBits =
ConvertType(SrcType)->getScalarSizeInBits();
1383 unsigned DstBits = Info.
Size;
1388 CodeGenFunction::SanitizerScope SanScope(
this);
1390 std::pair<ScalarExprEmitter::ImplicitConversionCheckKind,
1391 std::pair<llvm::Value *, SanitizerKind::SanitizerOrdinal>>
1395 bool EmitTruncation = DstBits < SrcBits;
1399 bool EmitTruncationFromUnsignedToSigned =
1400 EmitTruncation && DstSigned && !SrcSigned;
1402 bool SameTypeSameSize = SrcSigned == DstSigned && SrcBits == DstBits;
1403 bool BothUnsigned = !SrcSigned && !DstSigned;
1404 bool LargerSigned = (DstBits > SrcBits) && DstSigned;
1411 bool EmitSignChange = !SameTypeSameSize && !BothUnsigned && !LargerSigned;
1416 else if (EmitSignChange) {
1417 assert(((SrcBits != DstBits) || (SrcSigned != DstSigned)) &&
1418 "either the widths should be different, or the signednesses.");
1424 ScalarExprEmitter::ImplicitConversionCheckKind CheckKind = Check.first;
1425 if (EmitTruncationFromUnsignedToSigned)
1426 CheckKind = ScalarExprEmitter::ICCK_SignedIntegerTruncationOrSignChange;
1428 llvm::Constant *StaticArgs[] = {
1431 llvm::ConstantInt::get(
Builder.getInt8Ty(), CheckKind),
1432 llvm::ConstantInt::get(
Builder.getInt32Ty(), Info.
Size)};
1434 EmitCheck(Check.second, SanitizerHandler::ImplicitConversion, StaticArgs,
1439 QualType DstType, llvm::Type *SrcTy,
1441 ScalarConversionOpts Opts) {
1443 llvm::Type *SrcElementTy;
1444 llvm::Type *DstElementTy;
1448 SrcElementTy = cast<llvm::VectorType>(SrcTy)->getElementType();
1449 DstElementTy = cast<llvm::VectorType>(DstTy)->getElementType();
1454 "cannot cast between matrix and non-matrix types");
1455 SrcElementTy = SrcTy;
1456 DstElementTy = DstTy;
1457 SrcElementType = SrcType;
1458 DstElementType = DstType;
1461 if (isa<llvm::IntegerType>(SrcElementTy)) {
1463 if (SrcElementType->
isBooleanType() && Opts.TreatBooleanAsSigned) {
1467 if (isa<llvm::IntegerType>(DstElementTy))
1468 return Builder.CreateIntCast(Src, DstTy, InputSigned,
"conv");
1470 return Builder.CreateSIToFP(Src, DstTy,
"conv");
1471 return Builder.CreateUIToFP(Src, DstTy,
"conv");
1474 if (isa<llvm::IntegerType>(DstElementTy)) {
1475 assert(SrcElementTy->isFloatingPointTy() &&
"Unknown real conversion");
1482 llvm::Intrinsic::ID IID =
1483 IsSigned ? llvm::Intrinsic::fptosi_sat : llvm::Intrinsic::fptoui_sat;
1484 return Builder.CreateCall(CGF.
CGM.
getIntrinsic(IID, {DstTy, SrcTy}), Src);
1488 return Builder.CreateFPToSI(Src, DstTy,
"conv");
1489 return Builder.CreateFPToUI(Src, DstTy,
"conv");
1492 if ((DstElementTy->is16bitFPTy() && SrcElementTy->is16bitFPTy())) {
1493 Value *FloatVal = Builder.CreateFPExt(Src, Builder.getFloatTy(),
"fpext");
1494 return Builder.CreateFPTrunc(FloatVal, DstTy,
"fptrunc");
1496 if (DstElementTy->getTypeID() < SrcElementTy->getTypeID())
1497 return Builder.CreateFPTrunc(Src, DstTy,
"conv");
1498 return Builder.CreateFPExt(Src, DstTy,
"conv");
1506 ScalarConversionOpts Opts) {
1521 return Builder.CreateIsNotNull(Src,
"tobool");
1524 return EmitFixedPointConversion(Src, SrcType, DstType,
Loc);
1527 "Unhandled scalar conversion from a fixed point type to another type.");
1531 return EmitFixedPointConversion(Src, SrcType, DstType,
Loc);
1534 "Unhandled scalar conversion to a fixed point type from another type.");
1537 QualType NoncanonicalSrcType = SrcType;
1538 QualType NoncanonicalDstType = DstType;
1542 if (SrcType == DstType)
return Src;
1546 llvm::Value *OrigSrc = Src;
1548 llvm::Type *SrcTy = Src->
getType();
1552 return EmitConversionToBool(Src, SrcType);
1554 llvm::Type *DstTy = ConvertType(DstType);
1559 if (DstTy->isFloatingPointTy()) {
1561 return Builder.CreateCall(
1569 Src = Builder.CreateCall(
1574 Src = Builder.CreateFPExt(Src, CGF.
CGM.
FloatTy,
"conv");
1582 if (SrcTy == DstTy) {
1583 if (Opts.EmitImplicitIntegerSignChangeChecks)
1584 EmitIntegerSignChangeCheck(Src, NoncanonicalSrcType, Src,
1585 NoncanonicalDstType,
Loc);
1593 if (
auto DstPT = dyn_cast<llvm::PointerType>(DstTy)) {
1595 if (isa<llvm::PointerType>(SrcTy))
1598 assert(SrcType->
isIntegerType() &&
"Not ptr->ptr or int->ptr conversion?");
1603 llvm::Value* IntResult =
1604 Builder.CreateIntCast(Src, MiddleTy, InputSigned,
"conv");
1606 return Builder.CreateIntToPtr(IntResult, DstTy,
"conv");
1609 if (isa<llvm::PointerType>(SrcTy)) {
1611 assert(isa<llvm::IntegerType>(DstTy) &&
"not ptr->int?");
1612 return Builder.CreatePtrToInt(Src, DstTy,
"conv");
1621 "Splatted expr doesn't match with vector element type?");
1624 unsigned NumElements = cast<llvm::FixedVectorType>(DstTy)->getNumElements();
1625 return Builder.CreateVectorSplat(NumElements, Src,
"splat");
1629 return EmitScalarCast(Src, SrcType, DstType, SrcTy, DstTy, Opts);
1631 if (isa<llvm::VectorType>(SrcTy) || isa<llvm::VectorType>(DstTy)) {
1633 llvm::TypeSize SrcSize = SrcTy->getPrimitiveSizeInBits();
1634 llvm::TypeSize DstSize = DstTy->getPrimitiveSizeInBits();
1635 if (SrcSize == DstSize)
1636 return Builder.CreateBitCast(Src, DstTy,
"conv");
1645 llvm::Type *SrcElementTy = cast<llvm::VectorType>(SrcTy)->getElementType();
1646 llvm::Type *DstElementTy = cast<llvm::VectorType>(DstTy)->getElementType();
1649 assert(((SrcElementTy->isIntegerTy() &&
1650 DstElementTy->isIntegerTy()) ||
1651 (SrcElementTy->isFloatingPointTy() &&
1652 DstElementTy->isFloatingPointTy())) &&
1653 "unexpected conversion between a floating-point vector and an "
1657 if (SrcElementTy->isIntegerTy())
1658 return Builder.CreateIntCast(Src, DstTy,
false,
"conv");
1661 if (SrcSize > DstSize)
1662 return Builder.CreateFPTrunc(Src, DstTy,
"conv");
1665 return Builder.CreateFPExt(Src, DstTy,
"conv");
1669 Value *Res =
nullptr;
1670 llvm::Type *ResTy = DstTy;
1677 if (CGF.
SanOpts.
has(SanitizerKind::FloatCastOverflow) &&
1679 EmitFloatConversionCheck(OrigSrc, OrigSrcType, Src, SrcType, DstType, DstTy,
1685 if (SrcTy->isFloatingPointTy()) {
1689 return Builder.CreateCall(
1692 return Builder.CreateFPTrunc(Src, DstTy);
1697 Res = EmitScalarCast(Src, SrcType, DstType, SrcTy, DstTy, Opts);
1699 if (DstTy != ResTy) {
1701 assert(ResTy->isIntegerTy(16) &&
"Only half FP requires extra conversion");
1702 Res = Builder.CreateCall(
1706 Res = Builder.CreateFPTrunc(Res, ResTy,
"conv");
1710 if (Opts.EmitImplicitIntegerTruncationChecks)
1711 EmitIntegerTruncationCheck(Src, NoncanonicalSrcType, Res,
1712 NoncanonicalDstType,
Loc);
1714 if (Opts.EmitImplicitIntegerSignChangeChecks)
1715 EmitIntegerSignChangeCheck(Src, NoncanonicalSrcType, Res,
1716 NoncanonicalDstType,
Loc);
1724 llvm::FixedPointBuilder<CGBuilderTy> FPBuilder(Builder);
1727 Result = FPBuilder.CreateFloatingToFixed(Src,
1730 Result = FPBuilder.CreateFixedToFloating(Src,
1732 ConvertType(DstTy));
1738 Result = FPBuilder.CreateFixedToInteger(Src, SrcFPSema,
1739 DstFPSema.getWidth(),
1740 DstFPSema.isSigned());
1742 Result = FPBuilder.CreateIntegerToFixed(Src, SrcFPSema.isSigned(),
1745 Result = FPBuilder.CreateFixedToFixed(Src, SrcFPSema, DstFPSema);
1752Value *ScalarExprEmitter::EmitComplexToScalarConversion(
1761 Src.first = EmitScalarConversion(Src.first, SrcTy, DstTy,
Loc);
1762 Src.second = EmitScalarConversion(Src.second, SrcTy, DstTy,
Loc);
1763 return Builder.CreateOr(Src.first, Src.second,
"tobool");
1770 return EmitScalarConversion(Src.first, SrcTy, DstTy,
Loc);
1781void ScalarExprEmitter::EmitBinOpCheck(
1782 ArrayRef<std::pair<Value *, SanitizerKind::SanitizerOrdinal>> Checks,
1783 const BinOpInfo &Info) {
1795 if (UO && UO->
getOpcode() == UO_Minus) {
1796 Check = SanitizerHandler::NegateOverflow;
1798 DynamicData.push_back(Info.RHS);
1802 Check = SanitizerHandler::ShiftOutOfBounds;
1804 StaticData.push_back(
1806 StaticData.push_back(
1808 }
else if (Opcode == BO_Div || Opcode == BO_Rem) {
1810 Check = SanitizerHandler::DivremOverflow;
1815 case BO_Add: Check = SanitizerHandler::AddOverflow;
break;
1816 case BO_Sub: Check = SanitizerHandler::SubOverflow;
break;
1817 case BO_Mul: Check = SanitizerHandler::MulOverflow;
break;
1818 default: llvm_unreachable(
"unexpected opcode for bin op check");
1822 DynamicData.push_back(Info.LHS);
1823 DynamicData.push_back(Info.RHS);
1826 CGF.
EmitCheck(Checks, Check, StaticData, DynamicData);
1833Value *ScalarExprEmitter::VisitExpr(
Expr *
E) {
1843 unsigned AddrSpace =
1845 llvm::Constant *GlobalConstStr = Builder.CreateGlobalString(
1846 E->ComputeName(Context),
"__usn_str", AddrSpace);
1848 llvm::Type *ExprTy = ConvertType(
E->
getType());
1849 return Builder.CreatePointerBitCastOrAddrSpaceCast(GlobalConstStr, ExprTy,
1854 assert(
E->getDataElementCount() == 1);
1855 auto It =
E->begin();
1856 return Builder.getInt((*It)->getValue());
1861 if (
E->getNumSubExprs() == 2) {
1866 auto *LTy = cast<llvm::FixedVectorType>(LHS->
getType());
1867 unsigned LHSElts = LTy->getNumElements();
1871 auto *MTy = cast<llvm::FixedVectorType>(Mask->
getType());
1875 llvm::ConstantInt::get(MTy, llvm::NextPowerOf2(LHSElts - 1) - 1);
1876 Mask = Builder.CreateAnd(Mask, MaskBits,
"mask");
1884 auto *RTy = llvm::FixedVectorType::get(LTy->getElementType(),
1885 MTy->getNumElements());
1886 Value* NewV = llvm::PoisonValue::get(RTy);
1887 for (
unsigned i = 0, e = MTy->getNumElements(); i != e; ++i) {
1888 Value *IIndx = llvm::ConstantInt::get(CGF.
SizeTy, i);
1889 Value *Indx = Builder.CreateExtractElement(Mask, IIndx,
"shuf_idx");
1891 Value *VExt = Builder.CreateExtractElement(LHS, Indx,
"shuf_elt");
1892 NewV = Builder.CreateInsertElement(NewV, VExt, IIndx,
"shuf_ins");
1901 for (
unsigned i = 2; i <
E->getNumSubExprs(); ++i) {
1902 llvm::APSInt Idx =
E->getShuffleMaskIdx(CGF.
getContext(), i-2);
1904 if (Idx.isSigned() && Idx.isAllOnes())
1905 Indices.push_back(-1);
1907 Indices.push_back(Idx.getZExtValue());
1910 return Builder.CreateShuffleVector(V1, V2, Indices,
"shuffle");
1921 if (SrcType == DstType)
return Src;
1924 "ConvertVector source type must be a vector");
1926 "ConvertVector destination type must be a vector");
1928 llvm::Type *SrcTy = Src->
getType();
1929 llvm::Type *DstTy = ConvertType(DstType);
1938 assert(SrcTy->isVectorTy() &&
1939 "ConvertVector source IR type must be a vector");
1940 assert(DstTy->isVectorTy() &&
1941 "ConvertVector destination IR type must be a vector");
1943 llvm::Type *SrcEltTy = cast<llvm::VectorType>(SrcTy)->getElementType(),
1944 *DstEltTy = cast<llvm::VectorType>(DstTy)->getElementType();
1946 if (DstEltType->isBooleanType()) {
1947 assert((SrcEltTy->isFloatingPointTy() ||
1948 isa<llvm::IntegerType>(SrcEltTy)) &&
"Unknown boolean conversion");
1950 llvm::Value *
Zero = llvm::Constant::getNullValue(SrcTy);
1951 if (SrcEltTy->isFloatingPointTy()) {
1952 return Builder.CreateFCmpUNE(Src, Zero,
"tobool");
1954 return Builder.CreateICmpNE(Src, Zero,
"tobool");
1959 Value *Res =
nullptr;
1961 if (isa<llvm::IntegerType>(SrcEltTy)) {
1963 if (isa<llvm::IntegerType>(DstEltTy))
1964 Res = Builder.CreateIntCast(Src, DstTy, InputSigned,
"conv");
1965 else if (InputSigned)
1966 Res = Builder.CreateSIToFP(Src, DstTy,
"conv");
1968 Res = Builder.CreateUIToFP(Src, DstTy,
"conv");
1969 }
else if (isa<llvm::IntegerType>(DstEltTy)) {
1970 assert(SrcEltTy->isFloatingPointTy() &&
"Unknown real conversion");
1971 if (DstEltType->isSignedIntegerOrEnumerationType())
1972 Res = Builder.CreateFPToSI(Src, DstTy,
"conv");
1974 Res = Builder.CreateFPToUI(Src, DstTy,
"conv");
1976 assert(SrcEltTy->isFloatingPointTy() && DstEltTy->isFloatingPointTy() &&
1977 "Unknown real conversion");
1978 if (DstEltTy->getTypeID() < SrcEltTy->getTypeID())
1979 Res = Builder.CreateFPTrunc(Src, DstTy,
"conv");
1981 Res = Builder.CreateFPExt(Src, DstTy,
"conv");
1996 return Builder.getInt(
Value);
2000 llvm::Value *
Result = EmitLoadOfLValue(
E);
2006 if (llvm::LoadInst *Load = dyn_cast<llvm::LoadInst>(
Result)) {
2007 if (llvm::GetElementPtrInst *GEP =
2008 dyn_cast<llvm::GetElementPtrInst>(
Load->getPointerOperand())) {
2009 if (llvm::Instruction *
Pointer =
2010 dyn_cast<llvm::Instruction>(GEP->getPointerOperand())) {
2023 TestAndClearIgnoreResultAssign();
2031 return EmitLoadOfLValue(
E);
2036 Value *Idx = Visit(
E->getIdx());
2039 if (CGF.
SanOpts.
has(SanitizerKind::ArrayBounds))
2042 return Builder.CreateExtractElement(
Base, Idx,
"vecext");
2046 TestAndClearIgnoreResultAssign();
2055 llvm::MatrixBuilder MB(Builder);
2056 Value *Idx = MB.CreateIndex(RowIdx, ColumnIdx, NumRows);
2058 MB.CreateIndexAssumption(Idx, MatrixTy->getNumElementsFlattened());
2060 Value *Matrix = Visit(
E->getBase());
2063 return Builder.CreateExtractElement(Matrix, Idx,
"matrixext");
2068 int MV = SVI->getMaskValue(Idx);
2075 assert(llvm::ConstantInt::isValueValidForType(I32Ty,
C->getZExtValue()) &&
2076 "Index operand too large for shufflevector mask!");
2077 return C->getZExtValue();
2081 bool Ignore = TestAndClearIgnoreResultAssign();
2083 assert (Ignore ==
false &&
"init list ignored");
2084 unsigned NumInitElements =
E->getNumInits();
2086 if (
E->hadArrayRangeDesignator())
2089 llvm::VectorType *VType =
2090 dyn_cast<llvm::VectorType>(ConvertType(
E->
getType()));
2093 if (NumInitElements == 0) {
2095 return EmitNullValue(
E->
getType());
2098 return Visit(
E->getInit(0));
2101 if (isa<llvm::ScalableVectorType>(VType)) {
2102 if (NumInitElements == 0) {
2104 return EmitNullValue(
E->
getType());
2107 if (NumInitElements == 1) {
2108 Expr *InitVector =
E->getInit(0);
2113 return Visit(InitVector);
2116 llvm_unreachable(
"Unexpected initialization of a scalable vector!");
2119 unsigned ResElts = cast<llvm::FixedVectorType>(VType)->getNumElements();
2126 unsigned CurIdx = 0;
2127 bool VIsPoisonShuffle =
false;
2128 llvm::Value *
V = llvm::PoisonValue::get(VType);
2129 for (
unsigned i = 0; i != NumInitElements; ++i) {
2130 Expr *IE =
E->getInit(i);
2134 llvm::VectorType *VVT = dyn_cast<llvm::VectorType>(
Init->getType());
2140 if (isa<ExtVectorElementExpr>(IE)) {
2141 llvm::ExtractElementInst *EI = cast<llvm::ExtractElementInst>(
Init);
2143 if (cast<llvm::FixedVectorType>(EI->getVectorOperandType())
2144 ->getNumElements() == ResElts) {
2145 llvm::ConstantInt *
C = cast<llvm::ConstantInt>(EI->getIndexOperand());
2146 Value *LHS =
nullptr, *RHS =
nullptr;
2151 Args.resize(ResElts, -1);
2153 LHS = EI->getVectorOperand();
2155 VIsPoisonShuffle =
true;
2156 }
else if (VIsPoisonShuffle) {
2158 llvm::ShuffleVectorInst *SVV = cast<llvm::ShuffleVectorInst>(
V);
2159 for (
unsigned j = 0; j != CurIdx; ++j)
2161 Args.push_back(ResElts +
C->getZExtValue());
2162 Args.resize(ResElts, -1);
2164 LHS = cast<llvm::ShuffleVectorInst>(
V)->getOperand(0);
2165 RHS = EI->getVectorOperand();
2166 VIsPoisonShuffle =
false;
2168 if (!Args.empty()) {
2169 V = Builder.CreateShuffleVector(LHS, RHS, Args);
2175 V = Builder.CreateInsertElement(
V,
Init, Builder.getInt32(CurIdx),
2177 VIsPoisonShuffle =
false;
2182 unsigned InitElts = cast<llvm::FixedVectorType>(VVT)->getNumElements();
2187 unsigned Offset = (CurIdx == 0) ? 0 : ResElts;
2188 if (isa<ExtVectorElementExpr>(IE)) {
2189 llvm::ShuffleVectorInst *SVI = cast<llvm::ShuffleVectorInst>(
Init);
2190 Value *SVOp = SVI->getOperand(0);
2191 auto *OpTy = cast<llvm::FixedVectorType>(SVOp->
getType());
2193 if (OpTy->getNumElements() == ResElts) {
2194 for (
unsigned j = 0; j != CurIdx; ++j) {
2197 if (VIsPoisonShuffle) {
2198 Args.push_back(
getMaskElt(cast<llvm::ShuffleVectorInst>(
V), j, 0));
2203 for (
unsigned j = 0, je = InitElts; j != je; ++j)
2205 Args.resize(ResElts, -1);
2207 if (VIsPoisonShuffle)
2208 V = cast<llvm::ShuffleVectorInst>(
V)->getOperand(0);
2217 for (
unsigned j = 0; j != InitElts; ++j)
2219 Args.resize(ResElts, -1);
2220 Init = Builder.CreateShuffleVector(
Init, Args,
"vext");
2223 for (
unsigned j = 0; j != CurIdx; ++j)
2225 for (
unsigned j = 0; j != InitElts; ++j)
2226 Args.push_back(j + Offset);
2227 Args.resize(ResElts, -1);
2234 V = Builder.CreateShuffleVector(
V,
Init, Args,
"vecinit");
2235 VIsPoisonShuffle = isa<llvm::PoisonValue>(
Init);
2241 llvm::Type *EltTy = VType->getElementType();
2244 for (; CurIdx < ResElts; ++CurIdx) {
2245 Value *Idx = Builder.getInt32(CurIdx);
2246 llvm::Value *
Init = llvm::Constant::getNullValue(EltTy);
2247 V = Builder.CreateInsertElement(
V,
Init, Idx,
"vecinit");
2255 if (CE->
getCastKind() == CK_UncheckedDerivedToBase)
2265 if (ICE->isGLValue())
2279 CodeGenFunction::CGFPOptionsRAII
FPOptions(CGF, CE);
2283 bool Ignored = TestAndClearIgnoreResultAssign();
2289 case CK_Dependent: llvm_unreachable(
"dependent cast kind in IR gen!");
2290 case CK_BuiltinFnToFnPtr:
2291 llvm_unreachable(
"builtin functions are handled elsewhere");
2293 case CK_LValueBitCast:
2294 case CK_ObjCObjectLValueCast: {
2295 Address Addr = EmitLValue(
E).getAddress();
2298 return EmitLoadOfLValue(LV, CE->
getExprLoc());
2301 case CK_LValueToRValueBitCast: {
2307 return EmitLoadOfLValue(DestLV, CE->
getExprLoc());
2310 case CK_CPointerToObjCPointerCast:
2311 case CK_BlockPointerToObjCPointerCast:
2312 case CK_AnyPointerToBlockPointerCast:
2315 llvm::Type *SrcTy = Src->
getType();
2316 llvm::Type *DstTy = ConvertType(DestTy);
2318 (!SrcTy->isPtrOrPtrVectorTy() || !DstTy->isPtrOrPtrVectorTy() ||
2319 SrcTy->getPointerAddressSpace() == DstTy->getPointerAddressSpace()) &&
2320 "Address-space cast must be used to convert address spaces");
2322 if (CGF.
SanOpts.
has(SanitizerKind::CFIUnrelatedCast)) {
2325 PT->getPointeeType(),
2341 Src = Builder.CreateLaunderInvariantGroup(Src);
2349 Src = Builder.CreateStripInvariantGroup(Src);
2354 if (
auto *CI = dyn_cast<llvm::CallBase>(Src)) {
2355 if (CI->getMetadata(
"heapallocsite") && isa<ExplicitCastExpr>(CE) &&
2356 !isa<CastExpr>(
E)) {
2358 if (!PointeeType.
isNull())
2367 if (
auto *FixedSrcTy = dyn_cast<llvm::FixedVectorType>(SrcTy)) {
2368 if (
auto *ScalableDstTy = dyn_cast<llvm::ScalableVectorType>(DstTy)) {
2371 if (ScalableDstTy->getElementType()->isIntegerTy(1) &&
2372 ScalableDstTy->getElementCount().isKnownMultipleOf(8) &&
2373 FixedSrcTy->getElementType()->isIntegerTy(8)) {
2374 ScalableDstTy = llvm::ScalableVectorType::get(
2375 FixedSrcTy->getElementType(),
2376 ScalableDstTy->getElementCount().getKnownMinValue() / 8);
2378 if (FixedSrcTy->getElementType() == ScalableDstTy->getElementType()) {
2379 llvm::Value *PoisonVec = llvm::PoisonValue::get(ScalableDstTy);
2380 llvm::Value *
Zero = llvm::Constant::getNullValue(CGF.
CGM.
Int64Ty);
2381 llvm::Value *
Result = Builder.CreateInsertVector(
2382 ScalableDstTy, PoisonVec, Src, Zero,
"cast.scalable");
2383 if (
Result->getType() != DstTy)
2393 if (
auto *ScalableSrcTy = dyn_cast<llvm::ScalableVectorType>(SrcTy)) {
2394 if (
auto *FixedDstTy = dyn_cast<llvm::FixedVectorType>(DstTy)) {
2397 if (ScalableSrcTy->getElementType()->isIntegerTy(1) &&
2398 ScalableSrcTy->getElementCount().isKnownMultipleOf(8) &&
2399 FixedDstTy->getElementType()->isIntegerTy(8)) {
2400 ScalableSrcTy = llvm::ScalableVectorType::get(
2401 FixedDstTy->getElementType(),
2402 ScalableSrcTy->getElementCount().getKnownMinValue() / 8);
2403 Src = Builder.CreateBitCast(Src, ScalableSrcTy);
2405 if (ScalableSrcTy->getElementType() == FixedDstTy->getElementType()) {
2406 llvm::Value *
Zero = llvm::Constant::getNullValue(CGF.
CGM.
Int64Ty);
2407 return Builder.CreateExtractVector(DstTy, Src, Zero,
"cast.fixed");
2418 if ((isa<llvm::FixedVectorType>(SrcTy) &&
2419 isa<llvm::ScalableVectorType>(DstTy)) ||
2420 (isa<llvm::ScalableVectorType>(SrcTy) &&
2421 isa<llvm::FixedVectorType>(DstTy))) {
2428 return EmitLoadOfLValue(DestLV, CE->
getExprLoc());
2431 llvm::Value *
Result = Builder.CreateBitCast(Src, DstTy);
2434 case CK_AddressSpaceConversion: {
2437 Result.Val.isNullPointer()) {
2441 if (
Result.HasSideEffects)
2444 ConvertType(DestTy)), DestTy);
2452 case CK_AtomicToNonAtomic:
2453 case CK_NonAtomicToAtomic:
2454 case CK_UserDefinedConversion:
2455 return Visit(
const_cast<Expr*
>(
E));
2459 : Visit(const_cast<
Expr *>(
E));
2462 case CK_BaseToDerived: {
2464 assert(DerivedClassDecl &&
"BaseToDerived arg isn't a C++ object pointer!");
2478 if (CGF.
SanOpts.
has(SanitizerKind::CFIDerivedCast))
2486 case CK_UncheckedDerivedToBase:
2487 case CK_DerivedToBase: {
2500 case CK_ArrayToPointerDecay:
2503 case CK_FunctionToPointerDecay:
2504 return EmitLValue(
E).getPointer(CGF);
2506 case CK_NullToPointer:
2507 if (MustVisitNullValue(
E))
2513 case CK_NullToMemberPointer: {
2514 if (MustVisitNullValue(
E))
2521 case CK_ReinterpretMemberPointer:
2522 case CK_BaseToDerivedMemberPointer:
2523 case CK_DerivedToBaseMemberPointer: {
2535 case CK_ARCProduceObject:
2537 case CK_ARCConsumeObject:
2539 case CK_ARCReclaimReturnedObject:
2541 case CK_ARCExtendBlockObject:
2544 case CK_CopyAndAutoreleaseBlockObject:
2547 case CK_FloatingRealToComplex:
2548 case CK_FloatingComplexCast:
2549 case CK_IntegralRealToComplex:
2550 case CK_IntegralComplexCast:
2551 case CK_IntegralComplexToFloatingComplex:
2552 case CK_FloatingComplexToIntegralComplex:
2553 case CK_ConstructorConversion:
2555 case CK_HLSLArrayRValue:
2556 llvm_unreachable(
"scalar cast to non-scalar value");
2558 case CK_LValueToRValue:
2560 assert(
E->
isGLValue() &&
"lvalue-to-rvalue applied to r-value!");
2561 return Visit(
const_cast<Expr*
>(
E));
2563 case CK_IntegralToPointer: {
2568 auto DestLLVMTy = ConvertType(DestTy);
2571 llvm::Value* IntResult =
2572 Builder.CreateIntCast(Src, MiddleTy, InputSigned,
"conv");
2574 auto *IntToPtr = Builder.CreateIntToPtr(IntResult, DestLLVMTy);
2580 IntToPtr = Builder.CreateLaunderInvariantGroup(IntToPtr);
2586 case CK_PointerToIntegral: {
2587 assert(!DestTy->
isBooleanType() &&
"bool should use PointerToBool");
2588 auto *PtrExpr = Visit(
E);
2596 PtrExpr = Builder.CreateStripInvariantGroup(PtrExpr);
2600 return Builder.CreatePtrToInt(PtrExpr, ConvertType(DestTy));
2606 case CK_MatrixCast: {
2607 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2610 case CK_VectorSplat: {
2611 llvm::Type *DstTy = ConvertType(DestTy);
2614 llvm::ElementCount NumElements =
2615 cast<llvm::VectorType>(DstTy)->getElementCount();
2616 return Builder.CreateVectorSplat(NumElements, Elt,
"splat");
2619 case CK_FixedPointCast:
2620 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2623 case CK_FixedPointToBoolean:
2625 "Expected src type to be fixed point type");
2626 assert(DestTy->
isBooleanType() &&
"Expected dest type to be boolean type");
2627 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2630 case CK_FixedPointToIntegral:
2632 "Expected src type to be fixed point type");
2633 assert(DestTy->
isIntegerType() &&
"Expected dest type to be an integer");
2634 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2637 case CK_IntegralToFixedPoint:
2639 "Expected src type to be an integer");
2641 "Expected dest type to be fixed point type");
2642 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2645 case CK_IntegralCast: {
2648 return Builder.CreateIntCast(Visit(
E), ConvertType(DestTy),
2652 ScalarConversionOpts Opts;
2653 if (
auto *ICE = dyn_cast<ImplicitCastExpr>(CE)) {
2654 if (!ICE->isPartOfExplicitCast())
2655 Opts = ScalarConversionOpts(CGF.
SanOpts);
2657 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2660 case CK_IntegralToFloating: {
2665 return Builder.CreateSIToFP(Visit(
E), ConvertType(DestTy),
"conv");
2666 return Builder.CreateUIToFP(Visit(
E), ConvertType(DestTy),
"conv");
2668 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, CE);
2669 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2672 case CK_FloatingToIntegral: {
2677 return Builder.CreateFPToSI(Visit(
E), ConvertType(DestTy),
"conv");
2678 return Builder.CreateFPToUI(Visit(
E), ConvertType(DestTy),
"conv");
2680 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, CE);
2681 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2684 case CK_FloatingCast: {
2691 return Builder.CreateFPTrunc(Visit(
E), ConvertType(DestTy),
"conv");
2692 return Builder.CreateFPExt(Visit(
E), ConvertType(DestTy),
"conv");
2694 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, CE);
2695 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2698 case CK_FixedPointToFloating:
2699 case CK_FloatingToFixedPoint: {
2700 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, CE);
2701 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2704 case CK_BooleanToSignedIntegral: {
2705 ScalarConversionOpts Opts;
2706 Opts.TreatBooleanAsSigned =
true;
2707 return EmitScalarConversion(Visit(
E),
E->
getType(), DestTy,
2710 case CK_IntegralToBoolean:
2711 return EmitIntToBoolConversion(Visit(
E));
2712 case CK_PointerToBoolean:
2713 return EmitPointerToBoolConversion(Visit(
E),
E->
getType());
2714 case CK_FloatingToBoolean: {
2715 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, CE);
2716 return EmitFloatToBoolConversion(Visit(
E));
2718 case CK_MemberPointerToBoolean: {
2719 llvm::Value *MemPtr = Visit(
E);
2724 case CK_FloatingComplexToReal:
2725 case CK_IntegralComplexToReal:
2728 case CK_FloatingComplexToBoolean:
2729 case CK_IntegralComplexToBoolean: {
2733 return EmitComplexToScalarConversion(
V,
E->
getType(), DestTy,
2737 case CK_ZeroToOCLOpaqueType: {
2740 "CK_ZeroToOCLEvent cast on non-event type");
2741 return llvm::Constant::getNullValue(ConvertType(DestTy));
2744 case CK_IntToOCLSampler:
2747 case CK_HLSLVectorTruncation: {
2749 "Destination type must be a vector or builtin type.");
2753 unsigned NumElts = VecTy->getNumElements();
2754 for (
unsigned I = 0; I != NumElts; ++I)
2757 return Builder.CreateShuffleVector(Vec, Mask,
"trunc");
2759 llvm::Value *
Zero = llvm::Constant::getNullValue(CGF.
SizeTy);
2760 return Builder.CreateExtractElement(Vec, Zero,
"cast.vtrunc");
2765 llvm_unreachable(
"unknown scalar cast");
2769 CodeGenFunction::StmtExprEvaluation eval(CGF);
2779 CodeGenFunction::RunCleanupsScope
Scope(CGF);
2780 Value *
V = Visit(
E->getSubExpr());
2783 Scope.ForceCleanup({&
V});
2792 llvm::Value *InVal,
bool IsInc,
2796 BinOp.RHS = llvm::ConstantInt::get(InVal->getType(), 1,
false);
2798 BinOp.Opcode = IsInc ? BO_Add : BO_Sub;
2799 BinOp.FPFeatures = FPFeatures;
2804llvm::Value *ScalarExprEmitter::EmitIncDecConsiderOverflowBehavior(
2806 llvm::Value *Amount =
2807 llvm::ConstantInt::get(InVal->getType(), IsInc ? 1 : -1,
true);
2808 StringRef Name = IsInc ?
"inc" :
"dec";
2809 switch (CGF.
getLangOpts().getSignedOverflowBehavior()) {
2811 if (!CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow))
2812 return Builder.CreateAdd(InVal, Amount, Name);
2815 if (!CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow))
2816 return Builder.CreateNSWAdd(InVal, Amount, Name);
2821 if (!
E->canOverflow() || CanElideOverflowCheck(CGF.
getContext(), Info))
2822 return Builder.CreateNSWAdd(InVal, Amount, Name);
2823 return EmitOverflowCheckedBinOp(Info);
2825 llvm_unreachable(
"Unknown SignedOverflowBehaviorTy");
2850class OMPLastprivateConditionalUpdateRAII {
2859 ~OMPLastprivateConditionalUpdateRAII() {
2862 CGF,
E->getSubExpr());
2869 bool isInc,
bool isPre) {
2870 OMPLastprivateConditionalUpdateRAII OMPRegion(CGF,
E);
2872 llvm::PHINode *atomicPHI =
nullptr;
2878 int amount = (isInc ? 1 : -1);
2879 bool isSubtraction = !isInc;
2882 type = atomicTy->getValueType();
2883 if (isInc &&
type->isBooleanType()) {
2887 ->setAtomic(llvm::AtomicOrdering::SequentiallyConsistent);
2888 return Builder.getTrue();
2892 return Builder.CreateAtomicRMW(
2894 llvm::AtomicOrdering::SequentiallyConsistent);
2899 if (!
type->isBooleanType() &&
type->isIntegerType() &&
2900 !(
type->isUnsignedIntegerType() &&
2901 CGF.
SanOpts.
has(SanitizerKind::UnsignedIntegerOverflow)) &&
2904 llvm::AtomicRMWInst::BinOp aop = isInc ? llvm::AtomicRMWInst::Add :
2905 llvm::AtomicRMWInst::Sub;
2906 llvm::Instruction::BinaryOps op = isInc ? llvm::Instruction::Add :
2907 llvm::Instruction::Sub;
2909 llvm::ConstantInt::get(ConvertType(
type), 1,
true),
type);
2911 Builder.CreateAtomicRMW(aop, LV.
getAddress(), amt,
2912 llvm::AtomicOrdering::SequentiallyConsistent);
2913 return isPre ? Builder.CreateBinOp(op, old, amt) : old;
2917 if (
type->isFloatingType()) {
2918 llvm::Type *Ty = ConvertType(
type);
2919 if (llvm::has_single_bit(Ty->getScalarSizeInBits())) {
2920 llvm::AtomicRMWInst::BinOp aop =
2921 isInc ? llvm::AtomicRMWInst::FAdd : llvm::AtomicRMWInst::FSub;
2922 llvm::Instruction::BinaryOps op =
2923 isInc ? llvm::Instruction::FAdd : llvm::Instruction::FSub;
2924 llvm::Value *amt = llvm::ConstantFP::get(Ty, 1.0);
2925 llvm::AtomicRMWInst *old =
2927 llvm::AtomicOrdering::SequentiallyConsistent);
2929 return isPre ? Builder.CreateBinOp(op, old, amt) : old;
2935 llvm::BasicBlock *startBB = Builder.GetInsertBlock();
2938 Builder.CreateBr(opBB);
2939 Builder.SetInsertPoint(opBB);
2940 atomicPHI = Builder.CreatePHI(value->getType(), 2);
2941 atomicPHI->addIncoming(value, startBB);
2955 if (isInc &&
type->isBooleanType()) {
2956 value = Builder.getTrue();
2959 }
else if (
type->isIntegerType()) {
2961 bool canPerformLossyDemotionCheck =
false;
2963 bool excludeOverflowPattern =
2968 assert(promotedType !=
type &&
"Shouldn't promote to the same type.");
2969 canPerformLossyDemotionCheck =
true;
2970 canPerformLossyDemotionCheck &=
2973 canPerformLossyDemotionCheck &=
2975 type, promotedType);
2976 assert((!canPerformLossyDemotionCheck ||
2977 type->isSignedIntegerOrEnumerationType() ||
2979 ConvertType(
type)->getScalarSizeInBits() ==
2980 ConvertType(promotedType)->getScalarSizeInBits()) &&
2981 "The following check expects that if we do promotion to different "
2982 "underlying canonical type, at least one of the types (either "
2983 "base or promoted) will be signed, or the bitwidths will match.");
2986 SanitizerKind::ImplicitIntegerArithmeticValueChange |
2987 SanitizerKind::ImplicitBitfieldConversion) &&
2988 canPerformLossyDemotionCheck) {
3002 value = EmitScalarConversion(value,
type, promotedType,
E->
getExprLoc());
3003 Value *amt = llvm::ConstantInt::get(value->getType(), amount,
true);
3004 value = Builder.CreateAdd(value, amt, isInc ?
"inc" :
"dec");
3008 ScalarConversionOpts Opts;
3010 Opts = ScalarConversionOpts(CGF.
SanOpts);
3011 else if (CGF.
SanOpts.
has(SanitizerKind::ImplicitBitfieldConversion)) {
3013 SrcType = promotedType;
3016 value = EmitScalarConversion(value, promotedType,
type,
E->
getExprLoc(),
3022 }
else if (
E->canOverflow() &&
type->isSignedIntegerOrEnumerationType()) {
3023 value = EmitIncDecConsiderOverflowBehavior(
E, value, isInc);
3024 }
else if (
E->canOverflow() &&
type->isUnsignedIntegerType() &&
3025 CGF.
SanOpts.
has(SanitizerKind::UnsignedIntegerOverflow) &&
3026 !excludeOverflowPattern &&
3028 SanitizerKind::UnsignedIntegerOverflow,
E->
getType())) {
3032 llvm::Value *amt = llvm::ConstantInt::get(value->getType(), amount,
true);
3033 value = Builder.CreateAdd(value, amt, isInc ?
"inc" :
"dec");
3044 if (!isInc) numElts = Builder.CreateNSWNeg(numElts,
"vla.negsize");
3047 value = Builder.CreateGEP(elemTy, value, numElts,
"vla.inc");
3050 elemTy, value, numElts,
false, isSubtraction,
3054 }
else if (
type->isFunctionType()) {
3055 llvm::Value *amt = Builder.getInt32(amount);
3058 value = Builder.CreateGEP(CGF.
Int8Ty, value, amt,
"incdec.funcptr");
3062 false, isSubtraction,
3067 llvm::Value *amt = Builder.getInt32(amount);
3070 value = Builder.CreateGEP(elemTy, value, amt,
"incdec.ptr");
3073 elemTy, value, amt,
false, isSubtraction,
3078 }
else if (
type->isVectorType()) {
3079 if (
type->hasIntegerRepresentation()) {
3080 llvm::Value *amt = llvm::ConstantInt::get(value->getType(), amount);
3082 value = Builder.CreateAdd(value, amt, isInc ?
"inc" :
"dec");
3084 value = Builder.CreateFAdd(
3086 llvm::ConstantFP::get(value->getType(), amount),
3087 isInc ?
"inc" :
"dec");
3091 }
else if (
type->isRealFloatingType()) {
3094 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF,
E);
3099 value = Builder.CreateCall(
3102 input,
"incdec.conv");
3104 value = Builder.CreateFPExt(input, CGF.
CGM.
FloatTy,
"incdec.conv");
3108 if (value->getType()->isFloatTy())
3109 amt = llvm::ConstantFP::get(VMContext,
3110 llvm::APFloat(
static_cast<float>(amount)));
3111 else if (value->getType()->isDoubleTy())
3112 amt = llvm::ConstantFP::get(VMContext,
3113 llvm::APFloat(
static_cast<double>(amount)));
3117 llvm::APFloat F(
static_cast<float>(amount));
3119 const llvm::fltSemantics *FS;
3122 if (value->getType()->isFP128Ty())
3124 else if (value->getType()->isHalfTy())
3126 else if (value->getType()->isBFloatTy())
3128 else if (value->getType()->isPPC_FP128Ty())
3132 F.convert(*FS, llvm::APFloat::rmTowardZero, &ignored);
3133 amt = llvm::ConstantFP::get(VMContext, F);
3135 value = Builder.CreateFAdd(value, amt, isInc ?
"inc" :
"dec");
3139 value = Builder.CreateCall(
3142 value,
"incdec.conv");
3144 value = Builder.CreateFPTrunc(value, input->getType(),
"incdec.conv");
3149 }
else if (
type->isFixedPointType()) {
3156 Info.Opcode = isInc ? BO_Add : BO_Sub;
3158 Info.RHS = llvm::ConstantInt::get(value->getType(), 1,
false);
3161 if (
type->isSignedFixedPointType()) {
3162 Info.Opcode = isInc ? BO_Sub : BO_Add;
3163 Info.RHS = Builder.CreateNeg(Info.RHS);
3168 llvm::FixedPointBuilder<CGBuilderTy> FPBuilder(Builder);
3170 Info.RHS = FPBuilder.CreateIntegerToFixed(Info.RHS,
true, DstSema);
3171 value = EmitFixedPointBinOp(Info);
3178 if (!isInc) size = -size;
3179 llvm::Value *sizeValue =
3180 llvm::ConstantInt::get(CGF.
SizeTy, size.getQuantity());
3183 value = Builder.CreateGEP(CGF.
Int8Ty, value, sizeValue,
"incdec.objptr");
3186 CGF.
Int8Ty, value, sizeValue,
false, isSubtraction,
3188 value = Builder.CreateBitCast(value, input->getType());
3192 llvm::BasicBlock *curBlock = Builder.GetInsertBlock();
3197 llvm::Value *
success = Pair.second;
3198 atomicPHI->addIncoming(old, curBlock);
3199 Builder.CreateCondBr(success, contBB, atomicPHI->getParent());
3200 Builder.SetInsertPoint(contBB);
3201 return isPre ? value : input;
3215 return isPre ? value : input;
3222 ? getPromotionType(
E->getSubExpr()->
getType())
3224 Value *result = VisitPlus(
E, promotionTy);
3225 if (result && !promotionTy.
isNull())
3226 result = EmitUnPromotedValue(result,
E->
getType());
3233 TestAndClearIgnoreResultAssign();
3234 if (!PromotionType.
isNull())
3236 return Visit(
E->getSubExpr());
3242 ? getPromotionType(
E->getSubExpr()->
getType())
3244 Value *result = VisitMinus(
E, promotionTy);
3245 if (result && !promotionTy.
isNull())
3246 result = EmitUnPromotedValue(result,
E->
getType());
3252 TestAndClearIgnoreResultAssign();
3254 if (!PromotionType.
isNull())
3257 Op = Visit(
E->getSubExpr());
3260 if (Op->
getType()->isFPOrFPVectorTy())
3261 return Builder.CreateFNeg(Op,
"fneg");
3266 BinOp.LHS = llvm::Constant::getNullValue(BinOp.RHS->getType());
3268 BinOp.Opcode = BO_Sub;
3271 return EmitSub(BinOp);
3275 TestAndClearIgnoreResultAssign();
3276 Value *Op = Visit(
E->getSubExpr());
3277 return Builder.CreateNot(Op,
"not");
3285 Value *Oper = Visit(
E->getSubExpr());
3288 if (Oper->
getType()->isFPOrFPVectorTy()) {
3289 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(
3291 Result = Builder.CreateFCmp(llvm::CmpInst::FCMP_OEQ, Oper, Zero,
"cmp");
3293 Result = Builder.CreateICmp(llvm::CmpInst::ICMP_EQ, Oper, Zero,
"cmp");
3294 return Builder.CreateSExt(
Result, ConvertType(
E->
getType()),
"sext");
3303 BoolVal = Builder.CreateNot(BoolVal,
"lnot");
3306 return Builder.CreateZExt(BoolVal, ConvertType(
E->
getType()),
"lnot.ext");
3314 return Builder.getInt(
Value);
3318 unsigned n =
E->getNumComponents();
3319 llvm::Type* ResultType = ConvertType(
E->
getType());
3320 llvm::Value*
Result = llvm::Constant::getNullValue(ResultType);
3322 for (
unsigned i = 0; i != n; ++i) {
3324 llvm::Value *Offset =
nullptr;
3331 Idx = Builder.CreateIntCast(Idx, ResultType, IdxSigned,
"conv");
3338 llvm::Value* ElemSize = llvm::ConstantInt::get(ResultType,
3342 Offset = Builder.CreateMul(Idx, ElemSize);
3356 Field != FieldEnd; ++Field, ++i) {
3357 if (*Field == MemberDecl)
3360 assert(i < RL.
getFieldCount() &&
"offsetof field in wrong type");
3365 Offset = llvm::ConstantInt::get(ResultType, OffsetInt);
3368 CurrentType = MemberDecl->
getType();
3373 llvm_unreachable(
"dependent __builtin_offsetof");
3389 auto *BaseRD = cast<CXXRecordDecl>(BaseRT->getDecl());
3391 Offset = llvm::ConstantInt::get(ResultType, OffsetInt.
getQuantity());
3403ScalarExprEmitter::VisitUnaryExprOrTypeTraitExpr(
3405 QualType TypeToSize =
E->getTypeOfArgument();
3406 if (
auto Kind =
E->getKind();
3407 Kind == UETT_SizeOf || Kind == UETT_DataSizeOf) {
3410 if (
E->isArgumentType()) {
3420 llvm::Value *size = VlaSize.
NumElts;
3424 if (!eltSize.
isOne())
3429 }
else if (
E->getKind() == UETT_OpenMPRequiredSimdAlign) {
3433 E->getTypeOfArgument()->getPointeeType()))
3435 return llvm::ConstantInt::get(CGF.
SizeTy, Alignment);
3436 }
else if (
E->getKind() == UETT_VectorElements) {
3437 auto *VecTy = cast<llvm::VectorType>(ConvertType(
E->getTypeOfArgument()));
3438 return Builder.CreateElementCount(CGF.
SizeTy, VecTy->getElementCount());
3449 ? getPromotionType(
E->getSubExpr()->
getType())
3451 Value *result = VisitReal(
E, promotionTy);
3452 if (result && !promotionTy.
isNull())
3453 result = EmitUnPromotedValue(result,
E->
getType());
3459 Expr *Op =
E->getSubExpr();
3465 if (!PromotionType.
isNull()) {
3467 Op, IgnoreResultAssign,
true);
3470 return result.first;
3480 if (!PromotionType.
isNull())
3488 ? getPromotionType(
E->getSubExpr()->
getType())
3490 Value *result = VisitImag(
E, promotionTy);
3491 if (result && !promotionTy.
isNull())
3492 result = EmitUnPromotedValue(result,
E->
getType());
3498 Expr *Op =
E->getSubExpr();
3504 if (!PromotionType.
isNull()) {
3506 Op,
true, IgnoreResultAssign);
3509 return result.second;
3523 else if (!PromotionType.
isNull())
3527 if (!PromotionType.
isNull())
3528 return llvm::Constant::getNullValue(ConvertType(PromotionType));
3529 return llvm::Constant::getNullValue(ConvertType(
E->
getType()));
3536Value *ScalarExprEmitter::EmitPromotedValue(
Value *result,
3538 return CGF.
Builder.CreateFPExt(result, ConvertType(PromotionType),
"ext");
3541Value *ScalarExprEmitter::EmitUnPromotedValue(
Value *result,
3543 return CGF.
Builder.CreateFPTrunc(result, ConvertType(ExprType),
"unpromotion");
3548 if (
auto BO = dyn_cast<BinaryOperator>(
E)) {
3550#define HANDLE_BINOP(OP) \
3552 return Emit##OP(EmitBinOps(BO, PromotionType));
3561 }
else if (
auto UO = dyn_cast<UnaryOperator>(
E)) {
3564 return VisitImag(UO, PromotionType);
3566 return VisitReal(UO, PromotionType);
3568 return VisitMinus(UO, PromotionType);
3570 return VisitPlus(UO, PromotionType);
3575 auto result = Visit(
const_cast<Expr *
>(
E));
3577 if (!PromotionType.
isNull())
3578 return EmitPromotedValue(result, PromotionType);
3580 return EmitUnPromotedValue(result,
E->
getType());
3587 TestAndClearIgnoreResultAssign();
3591 if (!PromotionType.
isNull())
3592 Result.Ty = PromotionType;
3595 Result.Opcode =
E->getOpcode();
3601LValue ScalarExprEmitter::EmitCompoundAssignLValue(
3603 Value *(ScalarExprEmitter::*
Func)(
const BinOpInfo &),
3608 if (
E->getComputationResultType()->isAnyComplexType())
3615 PromotionTypeCR = getPromotionType(
E->getComputationResultType());
3616 if (PromotionTypeCR.
isNull())
3617 PromotionTypeCR =
E->getComputationResultType();
3618 QualType PromotionTypeLHS = getPromotionType(
E->getComputationLHSType());
3620 if (!PromotionTypeRHS.
isNull())
3623 OpInfo.RHS = Visit(
E->getRHS());
3624 OpInfo.Ty = PromotionTypeCR;
3625 OpInfo.Opcode =
E->getOpcode();
3631 llvm::PHINode *atomicPHI =
nullptr;
3634 if (!
type->isBooleanType() &&
type->isIntegerType() &&
3635 !(
type->isUnsignedIntegerType() &&
3636 CGF.
SanOpts.
has(SanitizerKind::UnsignedIntegerOverflow)) &&
3639 llvm::AtomicRMWInst::BinOp AtomicOp = llvm::AtomicRMWInst::BAD_BINOP;
3640 llvm::Instruction::BinaryOps Op;
3641 switch (OpInfo.Opcode) {
3643 case BO_MulAssign:
case BO_DivAssign:
3649 AtomicOp = llvm::AtomicRMWInst::Add;
3650 Op = llvm::Instruction::Add;
3653 AtomicOp = llvm::AtomicRMWInst::Sub;
3654 Op = llvm::Instruction::Sub;
3657 AtomicOp = llvm::AtomicRMWInst::And;
3658 Op = llvm::Instruction::And;
3661 AtomicOp = llvm::AtomicRMWInst::Xor;
3662 Op = llvm::Instruction::Xor;
3665 AtomicOp = llvm::AtomicRMWInst::Or;
3666 Op = llvm::Instruction::Or;
3669 llvm_unreachable(
"Invalid compound assignment type");
3671 if (AtomicOp != llvm::AtomicRMWInst::BAD_BINOP) {
3673 EmitScalarConversion(OpInfo.RHS,
E->getRHS()->
getType(), LHSTy,
3677 llvm::AtomicRMWInst *OldVal =
3682 Result = Builder.CreateBinOp(Op, OldVal, Amt);
3688 llvm::BasicBlock *startBB = Builder.GetInsertBlock();
3690 OpInfo.LHS = EmitLoadOfLValue(LHSLV,
E->
getExprLoc());
3692 Builder.CreateBr(opBB);
3693 Builder.SetInsertPoint(opBB);
3694 atomicPHI = Builder.CreatePHI(OpInfo.LHS->getType(), 2);
3695 atomicPHI->addIncoming(OpInfo.LHS, startBB);
3696 OpInfo.LHS = atomicPHI;
3699 OpInfo.LHS = EmitLoadOfLValue(LHSLV,
E->
getExprLoc());
3701 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, OpInfo.FPFeatures);
3703 if (!PromotionTypeLHS.
isNull())
3704 OpInfo.LHS = EmitScalarConversion(OpInfo.LHS, LHSTy, PromotionTypeLHS,
3707 OpInfo.LHS = EmitScalarConversion(OpInfo.LHS, LHSTy,
3708 E->getComputationLHSType(),
Loc);
3723 ScalarConversionOpts(CGF.
SanOpts));
3726 llvm::BasicBlock *curBlock = Builder.GetInsertBlock();
3730 llvm::Value *old = CGF.
EmitToMemory(Pair.first.getScalarVal(), LHSTy);
3731 llvm::Value *
success = Pair.second;
3732 atomicPHI->addIncoming(old, curBlock);
3733 Builder.CreateCondBr(success, contBB, atomicPHI->getParent());
3734 Builder.SetInsertPoint(contBB);
3759 Value *(ScalarExprEmitter::*
Func)(
const BinOpInfo &)) {
3760 bool Ignore = TestAndClearIgnoreResultAssign();
3761 Value *RHS =
nullptr;
3762 LValue LHS = EmitCompoundAssignLValue(
E,
Func, RHS);
3780void ScalarExprEmitter::EmitUndefinedBehaviorIntegerDivAndRemCheck(
3781 const BinOpInfo &Ops, llvm::Value *Zero,
bool isDiv) {
3785 if (CGF.
SanOpts.
has(SanitizerKind::IntegerDivideByZero)) {
3786 Checks.push_back(std::make_pair(Builder.CreateICmpNE(Ops.RHS, Zero),
3787 SanitizerKind::SO_IntegerDivideByZero));
3790 const auto *BO = cast<BinaryOperator>(Ops.E);
3791 if (CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow) &&
3792 Ops.Ty->hasSignedIntegerRepresentation() &&
3794 Ops.mayHaveIntegerOverflow()) {
3795 llvm::IntegerType *Ty = cast<llvm::IntegerType>(
Zero->getType());
3797 llvm::Value *IntMin =
3798 Builder.getInt(llvm::APInt::getSignedMinValue(Ty->getBitWidth()));
3799 llvm::Value *NegOne = llvm::Constant::getAllOnesValue(Ty);
3801 llvm::Value *LHSCmp = Builder.CreateICmpNE(Ops.LHS, IntMin);
3802 llvm::Value *RHSCmp = Builder.CreateICmpNE(Ops.RHS, NegOne);
3803 llvm::Value *NotOverflow = Builder.CreateOr(LHSCmp, RHSCmp,
"or");
3805 std::make_pair(NotOverflow, SanitizerKind::SO_SignedIntegerOverflow));
3808 if (Checks.size() > 0)
3809 EmitBinOpCheck(Checks, Ops);
3812Value *ScalarExprEmitter::EmitDiv(
const BinOpInfo &Ops) {
3814 CodeGenFunction::SanitizerScope SanScope(&CGF);
3815 if ((CGF.
SanOpts.
has(SanitizerKind::IntegerDivideByZero) ||
3816 CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow)) &&
3817 Ops.Ty->isIntegerType() &&
3818 (Ops.mayHaveIntegerDivisionByZero() || Ops.mayHaveIntegerOverflow())) {
3819 llvm::Value *
Zero = llvm::Constant::getNullValue(ConvertType(Ops.Ty));
3820 EmitUndefinedBehaviorIntegerDivAndRemCheck(Ops, Zero,
true);
3821 }
else if (CGF.
SanOpts.
has(SanitizerKind::FloatDivideByZero) &&
3822 Ops.Ty->isRealFloatingType() &&
3823 Ops.mayHaveFloatDivisionByZero()) {
3824 llvm::Value *
Zero = llvm::Constant::getNullValue(ConvertType(Ops.Ty));
3825 llvm::Value *NonZero = Builder.CreateFCmpUNE(Ops.RHS, Zero);
3827 std::make_pair(NonZero, SanitizerKind::SO_FloatDivideByZero), Ops);
3831 if (Ops.Ty->isConstantMatrixType()) {
3832 llvm::MatrixBuilder MB(Builder);
3835 auto *BO = cast<BinaryOperator>(Ops.E);
3839 "first operand must be a matrix");
3841 "second operand must be an arithmetic type");
3842 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, Ops.FPFeatures);
3843 return MB.CreateScalarDiv(Ops.LHS, Ops.RHS,
3844 Ops.Ty->hasUnsignedIntegerRepresentation());
3847 if (Ops.LHS->getType()->isFPOrFPVectorTy()) {
3849 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, Ops.FPFeatures);
3850 Val = Builder.CreateFDiv(Ops.LHS, Ops.RHS,
"div");
3854 else if (Ops.isFixedPointOp())
3855 return EmitFixedPointBinOp(Ops);
3856 else if (Ops.Ty->hasUnsignedIntegerRepresentation())
3857 return Builder.CreateUDiv(Ops.LHS, Ops.RHS,
"div");
3859 return Builder.CreateSDiv(Ops.LHS, Ops.RHS,
"div");
3862Value *ScalarExprEmitter::EmitRem(
const BinOpInfo &Ops) {
3864 if ((CGF.
SanOpts.
has(SanitizerKind::IntegerDivideByZero) ||
3865 CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow)) &&
3866 Ops.Ty->isIntegerType() &&
3867 (Ops.mayHaveIntegerDivisionByZero() || Ops.mayHaveIntegerOverflow())) {
3868 CodeGenFunction::SanitizerScope SanScope(&CGF);
3869 llvm::Value *
Zero = llvm::Constant::getNullValue(ConvertType(Ops.Ty));
3870 EmitUndefinedBehaviorIntegerDivAndRemCheck(Ops, Zero,
false);
3873 if (Ops.Ty->hasUnsignedIntegerRepresentation())
3874 return Builder.CreateURem(Ops.LHS, Ops.RHS,
"rem");
3876 return Builder.CreateSRem(Ops.LHS, Ops.RHS,
"rem");
3879Value *ScalarExprEmitter::EmitOverflowCheckedBinOp(
const BinOpInfo &Ops) {
3884 bool isSigned = Ops.Ty->isSignedIntegerOrEnumerationType();
3885 switch (Ops.Opcode) {
3889 IID = isSigned ? llvm::Intrinsic::sadd_with_overflow :
3890 llvm::Intrinsic::uadd_with_overflow;
3891 OverflowKind = SanitizerHandler::AddOverflow;
3896 IID = isSigned ? llvm::Intrinsic::ssub_with_overflow :
3897 llvm::Intrinsic::usub_with_overflow;
3898 OverflowKind = SanitizerHandler::SubOverflow;
3903 IID = isSigned ? llvm::Intrinsic::smul_with_overflow :
3904 llvm::Intrinsic::umul_with_overflow;
3905 OverflowKind = SanitizerHandler::MulOverflow;
3908 llvm_unreachable(
"Unsupported operation for overflow detection");
3914 CodeGenFunction::SanitizerScope SanScope(&CGF);
3919 Value *resultAndOverflow = Builder.CreateCall(intrinsic, {Ops.LHS, Ops.RHS});
3920 Value *result = Builder.CreateExtractValue(resultAndOverflow, 0);
3921 Value *overflow = Builder.CreateExtractValue(resultAndOverflow, 1);
3924 const std::string *handlerName =
3926 if (handlerName->empty()) {
3929 if (!isSigned || CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow)) {
3930 llvm::Value *NotOverflow = Builder.CreateNot(overflow);
3932 isSigned ? SanitizerKind::SO_SignedIntegerOverflow
3933 : SanitizerKind::SO_UnsignedIntegerOverflow;
3934 EmitBinOpCheck(std::make_pair(NotOverflow, Ordinal), Ops);
3936 CGF.
EmitTrapCheck(Builder.CreateNot(overflow), OverflowKind);
3941 llvm::BasicBlock *initialBB = Builder.GetInsertBlock();
3942 llvm::BasicBlock *continueBB =
3946 Builder.CreateCondBr(overflow, overflowBB, continueBB);
3950 Builder.SetInsertPoint(overflowBB);
3953 llvm::Type *Int8Ty = CGF.
Int8Ty;
3954 llvm::Type *argTypes[] = { CGF.
Int64Ty, CGF.
Int64Ty, Int8Ty, Int8Ty };
3955 llvm::FunctionType *handlerTy =
3956 llvm::FunctionType::get(CGF.
Int64Ty, argTypes,
true);
3957 llvm::FunctionCallee handler =
3962 llvm::Value *lhs = Builder.CreateSExt(Ops.LHS, CGF.
Int64Ty);
3963 llvm::Value *rhs = Builder.CreateSExt(Ops.RHS, CGF.
Int64Ty);
3967 llvm::Value *handlerArgs[] = {
3970 Builder.getInt8(OpID),
3971 Builder.getInt8(cast<llvm::IntegerType>(opTy)->getBitWidth())
3973 llvm::Value *handlerResult =
3977 handlerResult = Builder.CreateTrunc(handlerResult, opTy);
3978 Builder.CreateBr(continueBB);
3980 Builder.SetInsertPoint(continueBB);
3981 llvm::PHINode *phi = Builder.CreatePHI(opTy, 2);
3982 phi->addIncoming(result, initialBB);
3983 phi->addIncoming(handlerResult, overflowBB);
3990 const BinOpInfo &op,
3991 bool isSubtraction) {
3996 Value *pointer = op.LHS;
3997 Expr *pointerOperand =
expr->getLHS();
3998 Value *index = op.RHS;
3999 Expr *indexOperand =
expr->getRHS();
4002 if (!isSubtraction && !pointer->
getType()->isPointerTy()) {
4003 std::swap(pointer, index);
4004 std::swap(pointerOperand, indexOperand);
4009 unsigned width = cast<llvm::IntegerType>(index->
getType())->getBitWidth();
4011 auto PtrTy = cast<llvm::PointerType>(pointer->
getType());
4036 if (width != DL.getIndexTypeSizeInBits(PtrTy)) {
4039 index = CGF.
Builder.CreateIntCast(index, DL.getIndexType(PtrTy), isSigned,
4045 index = CGF.
Builder.CreateNeg(index,
"idx.neg");
4047 if (CGF.
SanOpts.
has(SanitizerKind::ArrayBounds))
4057 llvm::Value *objectSize
4060 index = CGF.
Builder.CreateMul(index, objectSize);
4079 index = CGF.
Builder.CreateMul(index, numElements,
"vla.index");
4082 index = CGF.
Builder.CreateNSWMul(index, numElements,
"vla.index");
4084 elemTy, pointer, index, isSigned, isSubtraction, op.E->getExprLoc(),
4103 elemTy, pointer, index, isSigned, isSubtraction, op.E->getExprLoc(),
4114 bool negMul,
bool negAdd) {
4115 Value *MulOp0 = MulOp->getOperand(0);
4116 Value *MulOp1 = MulOp->getOperand(1);
4118 MulOp0 = Builder.CreateFNeg(MulOp0,
"neg");
4120 Addend = Builder.CreateFNeg(Addend,
"neg");
4122 Value *FMulAdd =
nullptr;
4123 if (Builder.getIsFPConstrained()) {
4124 assert(isa<llvm::ConstrainedFPIntrinsic>(MulOp) &&
4125 "Only constrained operation should be created when Builder is in FP "
4126 "constrained mode");
4127 FMulAdd = Builder.CreateConstrainedFPCall(
4128 CGF.
CGM.
getIntrinsic(llvm::Intrinsic::experimental_constrained_fmuladd,
4130 {MulOp0, MulOp1, Addend});
4132 FMulAdd = Builder.CreateCall(
4134 {MulOp0, MulOp1, Addend});
4136 MulOp->eraseFromParent();
4151 assert((op.Opcode == BO_Add || op.Opcode == BO_AddAssign ||
4152 op.Opcode == BO_Sub || op.Opcode == BO_SubAssign) &&
4153 "Only fadd/fsub can be the root of an fmuladd.");
4156 if (!op.FPFeatures.allowFPContractWithinStatement())
4159 Value *LHS = op.LHS;
4160 Value *RHS = op.RHS;
4164 bool NegLHS =
false;
4165 if (
auto *LHSUnOp = dyn_cast<llvm::UnaryOperator>(LHS)) {
4166 if (LHSUnOp->getOpcode() == llvm::Instruction::FNeg &&
4167 LHSUnOp->use_empty() && LHSUnOp->getOperand(0)->hasOneUse()) {
4168 LHS = LHSUnOp->getOperand(0);
4173 bool NegRHS =
false;
4174 if (
auto *RHSUnOp = dyn_cast<llvm::UnaryOperator>(RHS)) {
4175 if (RHSUnOp->getOpcode() == llvm::Instruction::FNeg &&
4176 RHSUnOp->use_empty() && RHSUnOp->getOperand(0)->hasOneUse()) {
4177 RHS = RHSUnOp->getOperand(0);
4185 if (
auto *LHSBinOp = dyn_cast<llvm::BinaryOperator>(LHS)) {
4186 if (LHSBinOp->getOpcode() == llvm::Instruction::FMul &&
4187 (LHSBinOp->use_empty() || NegLHS)) {
4190 cast<llvm::Instruction>(op.LHS)->eraseFromParent();
4191 return buildFMulAdd(LHSBinOp, op.RHS, CGF, Builder, NegLHS, isSub);
4194 if (
auto *RHSBinOp = dyn_cast<llvm::BinaryOperator>(RHS)) {
4195 if (RHSBinOp->getOpcode() == llvm::Instruction::FMul &&
4196 (RHSBinOp->use_empty() || NegRHS)) {
4199 cast<llvm::Instruction>(op.RHS)->eraseFromParent();
4200 return buildFMulAdd(RHSBinOp, op.LHS, CGF, Builder, isSub ^ NegRHS,
false);
4204 if (
auto *LHSBinOp = dyn_cast<llvm::CallBase>(LHS)) {
4205 if (LHSBinOp->getIntrinsicID() ==
4206 llvm::Intrinsic::experimental_constrained_fmul &&
4207 (LHSBinOp->use_empty() || NegLHS)) {
4210 cast<llvm::Instruction>(op.LHS)->eraseFromParent();
4211 return buildFMulAdd(LHSBinOp, op.RHS, CGF, Builder, NegLHS, isSub);
4214 if (
auto *RHSBinOp = dyn_cast<llvm::CallBase>(RHS)) {
4215 if (RHSBinOp->getIntrinsicID() ==
4216 llvm::Intrinsic::experimental_constrained_fmul &&
4217 (RHSBinOp->use_empty() || NegRHS)) {
4220 cast<llvm::Instruction>(op.RHS)->eraseFromParent();
4221 return buildFMulAdd(RHSBinOp, op.LHS, CGF, Builder, isSub ^ NegRHS,
false);
4228Value *ScalarExprEmitter::EmitAdd(
const BinOpInfo &op) {
4229 if (op.LHS->getType()->isPointerTy() ||
4230 op.RHS->getType()->isPointerTy())
4233 if (op.Ty->isSignedIntegerOrEnumerationType()) {
4234 switch (CGF.
getLangOpts().getSignedOverflowBehavior()) {
4236 if (!CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow))
4237 return Builder.CreateAdd(op.LHS, op.RHS,
"add");
4240 if (!CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow))
4241 return Builder.CreateNSWAdd(op.LHS, op.RHS,
"add");
4244 if (CanElideOverflowCheck(CGF.
getContext(), op))
4245 return Builder.CreateNSWAdd(op.LHS, op.RHS,
"add");
4246 return EmitOverflowCheckedBinOp(op);
4251 if (op.LHS->getType()->isFPOrFPVectorTy()) {
4252 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, op.FPFeatures);
4258 if (op.Ty->isConstantMatrixType()) {
4259 llvm::MatrixBuilder MB(Builder);
4260 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, op.FPFeatures);
4261 return MB.CreateAdd(op.LHS, op.RHS);
4264 if (op.Ty->isUnsignedIntegerType() &&
4265 CGF.
SanOpts.
has(SanitizerKind::UnsignedIntegerOverflow) &&
4266 !CanElideOverflowCheck(CGF.
getContext(), op))
4267 return EmitOverflowCheckedBinOp(op);
4269 if (op.LHS->getType()->isFPOrFPVectorTy()) {
4270 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, op.FPFeatures);
4271 return Builder.CreateFAdd(op.LHS, op.RHS,
"add");
4274 if (op.isFixedPointOp())
4275 return EmitFixedPointBinOp(op);
4277 return Builder.CreateAdd(op.LHS, op.RHS,
"add");
4282Value *ScalarExprEmitter::EmitFixedPointBinOp(
const BinOpInfo &op) {
4284 using llvm::ConstantInt;
4292 if (
const auto *BinOp = dyn_cast<BinaryOperator>(op.E)) {
4293 RHSTy = BinOp->getRHS()->getType();
4294 if (
const auto *CAO = dyn_cast<CompoundAssignOperator>(BinOp)) {
4299 LHSTy = CAO->getComputationLHSType();
4300 ResultTy = CAO->getComputationResultType();
4302 LHSTy = BinOp->getLHS()->getType();
4303 }
else if (
const auto *UnOp = dyn_cast<UnaryOperator>(op.E)) {
4304 LHSTy = UnOp->getSubExpr()->getType();
4305 RHSTy = UnOp->getSubExpr()->getType();
4308 Value *LHS = op.LHS;
4309 Value *RHS = op.RHS;
4314 auto CommonFixedSema = LHSFixedSema.getCommonSemantics(RHSFixedSema);
4318 llvm::FixedPointBuilder<CGBuilderTy> FPBuilder(Builder);
4319 switch (op.Opcode) {
4322 Result = FPBuilder.CreateAdd(LHS, LHSFixedSema, RHS, RHSFixedSema);
4326 Result = FPBuilder.CreateSub(LHS, LHSFixedSema, RHS, RHSFixedSema);
4330 Result = FPBuilder.CreateMul(LHS, LHSFixedSema, RHS, RHSFixedSema);
4334 Result = FPBuilder.CreateDiv(LHS, LHSFixedSema, RHS, RHSFixedSema);
4338 Result = FPBuilder.CreateShl(LHS, LHSFixedSema, RHS);
4342 Result = FPBuilder.CreateShr(LHS, LHSFixedSema, RHS);
4345 return FPBuilder.CreateLT(LHS, LHSFixedSema, RHS, RHSFixedSema);
4347 return FPBuilder.CreateGT(LHS, LHSFixedSema, RHS, RHSFixedSema);
4349 return FPBuilder.CreateLE(LHS, LHSFixedSema, RHS, RHSFixedSema);
4351 return FPBuilder.CreateGE(LHS, LHSFixedSema, RHS, RHSFixedSema);
4356 return FPBuilder.CreateEQ(LHS, LHSFixedSema, RHS, RHSFixedSema);
4358 return FPBuilder.CreateNE(LHS, LHSFixedSema, RHS, RHSFixedSema);
4362 llvm_unreachable(
"Found unimplemented fixed point binary operation");
4375 llvm_unreachable(
"Found unsupported binary operation for fixed point types.");
4381 return FPBuilder.CreateFixedToFixed(
Result, IsShift ? LHSFixedSema
4386Value *ScalarExprEmitter::EmitSub(
const BinOpInfo &op) {
4388 if (!op.LHS->getType()->isPointerTy()) {
4389 if (op.Ty->isSignedIntegerOrEnumerationType()) {
4390 switch (CGF.
getLangOpts().getSignedOverflowBehavior()) {
4392 if (!CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow))
4393 return Builder.CreateSub(op.LHS, op.RHS,
"sub");
4396 if (!CGF.
SanOpts.
has(SanitizerKind::SignedIntegerOverflow))
4397 return Builder.CreateNSWSub(op.LHS, op.RHS,
"sub");
4400 if (CanElideOverflowCheck(CGF.
getContext(), op))
4401 return Builder.CreateNSWSub(op.LHS, op.RHS,
"sub");
4402 return EmitOverflowCheckedBinOp(op);
4407 if (op.LHS->getType()->isFPOrFPVectorTy()) {
4408 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, op.FPFeatures);
4414 if (op.Ty->isConstantMatrixType()) {
4415 llvm::MatrixBuilder MB(Builder);
4416 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, op.FPFeatures);
4417 return MB.CreateSub(op.LHS, op.RHS);
4420 if (op.Ty->isUnsignedIntegerType() &&
4421 CGF.
SanOpts.
has(SanitizerKind::UnsignedIntegerOverflow) &&
4422 !CanElideOverflowCheck(CGF.
getContext(), op))
4423 return EmitOverflowCheckedBinOp(op);
4425 if (op.LHS->getType()->isFPOrFPVectorTy()) {
4426 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, op.FPFeatures);
4427 return Builder.CreateFSub(op.LHS, op.RHS,
"sub");
4430 if (op.isFixedPointOp())
4431 return EmitFixedPointBinOp(op);
4433 return Builder.CreateSub(op.LHS, op.RHS,
"sub");
4438 if (!op.RHS->getType()->isPointerTy())
4445 = Builder.CreatePtrToInt(op.LHS, CGF.
PtrDiffTy,
"sub.ptr.lhs.cast");
4447 = Builder.CreatePtrToInt(op.RHS, CGF.
PtrDiffTy,
"sub.ptr.rhs.cast");
4448 Value *diffInChars = Builder.CreateSub(LHS, RHS,
"sub.ptr.sub");
4452 QualType elementType =
expr->getLHS()->getType()->getPointeeType();
4454 llvm::Value *divisor =
nullptr;
4460 elementType = VlaSize.Type;
4461 divisor = VlaSize.NumElts;
4465 if (!eltSize.
isOne())
4481 if (elementSize.
isOne())
4490 return Builder.CreateExactSDiv(diffInChars, divisor,
"sub.ptr.div");
4493Value *ScalarExprEmitter::GetMaximumShiftAmount(
Value *LHS,
Value *RHS,
4495 llvm::IntegerType *Ty;
4496 if (llvm::VectorType *VT = dyn_cast<llvm::VectorType>(LHS->
getType()))
4497 Ty = cast<llvm::IntegerType>(VT->getElementType());
4499 Ty = cast<llvm::IntegerType>(LHS->
getType());
4504 llvm::Type *RHSTy = RHS->
getType();
4505 llvm::APInt RHSMax =
4506 RHSIsSigned ? llvm::APInt::getSignedMaxValue(RHSTy->getScalarSizeInBits())
4507 :
llvm::
APInt::getMaxValue(RHSTy->getScalarSizeInBits());
4508 if (RHSMax.ult(Ty->getBitWidth()))
4509 return llvm::ConstantInt::get(RHSTy, RHSMax);
4510 return llvm::ConstantInt::get(RHSTy, Ty->getBitWidth() - 1);
4514 const Twine &Name) {
4515 llvm::IntegerType *Ty;
4516 if (
auto *VT = dyn_cast<llvm::VectorType>(LHS->
getType()))
4517 Ty = cast<llvm::IntegerType>(VT->getElementType());
4519 Ty = cast<llvm::IntegerType>(LHS->
getType());
4521 if (llvm::isPowerOf2_64(Ty->getBitWidth()))
4522 return Builder.CreateAnd(RHS, GetMaximumShiftAmount(LHS, RHS,
false), Name);
4524 return Builder.CreateURem(
4525 RHS, llvm::ConstantInt::get(RHS->
getType(), Ty->getBitWidth()), Name);
4528Value *ScalarExprEmitter::EmitShl(
const BinOpInfo &Ops) {
4530 if (Ops.isFixedPointOp())
4531 return EmitFixedPointBinOp(Ops);
4535 Value *RHS = Ops.RHS;
4536 if (Ops.LHS->getType() != RHS->
getType())
4537 RHS = Builder.CreateIntCast(RHS, Ops.LHS->getType(),
false,
"sh_prom");
4539 bool SanitizeSignedBase = CGF.
SanOpts.
has(SanitizerKind::ShiftBase) &&
4540 Ops.Ty->hasSignedIntegerRepresentation() &&
4543 bool SanitizeUnsignedBase =
4544 CGF.
SanOpts.
has(SanitizerKind::UnsignedShiftBase) &&
4545 Ops.Ty->hasUnsignedIntegerRepresentation();
4546 bool SanitizeBase = SanitizeSignedBase || SanitizeUnsignedBase;
4547 bool SanitizeExponent = CGF.
SanOpts.
has(SanitizerKind::ShiftExponent);
4550 RHS = ConstrainShiftValue(Ops.LHS, RHS,
"shl.mask");
4551 else if ((SanitizeBase || SanitizeExponent) &&
4552 isa<llvm::IntegerType>(Ops.LHS->getType())) {
4553 CodeGenFunction::SanitizerScope SanScope(&CGF);
4555 bool RHSIsSigned = Ops.rhsHasSignedIntegerRepresentation();
4556 llvm::Value *WidthMinusOne =
4557 GetMaximumShiftAmount(Ops.LHS, Ops.RHS, RHSIsSigned);
4558 llvm::Value *ValidExponent = Builder.CreateICmpULE(Ops.RHS, WidthMinusOne);
4560 if (SanitizeExponent) {
4562 std::make_pair(ValidExponent, SanitizerKind::SO_ShiftExponent));
4569 llvm::BasicBlock *Orig = Builder.GetInsertBlock();
4572 Builder.CreateCondBr(ValidExponent, CheckShiftBase, Cont);
4573 llvm::Value *PromotedWidthMinusOne =
4574 (RHS == Ops.RHS) ? WidthMinusOne
4575 : GetMaximumShiftAmount(Ops.LHS, RHS, RHSIsSigned);
4577 llvm::Value *BitsShiftedOff = Builder.CreateLShr(
4578 Ops.LHS, Builder.CreateSub(PromotedWidthMinusOne, RHS,
"shl.zeros",
4581 if (SanitizeUnsignedBase || CGF.
getLangOpts().CPlusPlus) {
4587 llvm::Value *One = llvm::ConstantInt::get(BitsShiftedOff->getType(), 1);
4588 BitsShiftedOff = Builder.CreateLShr(BitsShiftedOff, One);
4590 llvm::Value *
Zero = llvm::ConstantInt::get(BitsShiftedOff->getType(), 0);
4591 llvm::Value *ValidBase = Builder.CreateICmpEQ(BitsShiftedOff, Zero);
4593 llvm::PHINode *BaseCheck = Builder.CreatePHI(ValidBase->getType(), 2);
4594 BaseCheck->addIncoming(Builder.getTrue(), Orig);
4595 BaseCheck->addIncoming(ValidBase, CheckShiftBase);
4596 Checks.push_back(std::make_pair(
4597 BaseCheck, SanitizeSignedBase ? SanitizerKind::SO_ShiftBase
4598 : SanitizerKind::SO_UnsignedShiftBase));
4601 assert(!Checks.empty());
4602 EmitBinOpCheck(Checks, Ops);
4605 return Builder.CreateShl(Ops.LHS, RHS,
"shl");
4608Value *ScalarExprEmitter::EmitShr(
const BinOpInfo &Ops) {
4610 if (Ops.isFixedPointOp())
4611 return EmitFixedPointBinOp(Ops);
4615 Value *RHS = Ops.RHS;
4616 if (Ops.LHS->getType() != RHS->
getType())
4617 RHS = Builder.CreateIntCast(RHS, Ops.LHS->getType(),
false,
"sh_prom");
4621 RHS = ConstrainShiftValue(Ops.LHS, RHS,
"shr.mask");
4622 else if (CGF.
SanOpts.
has(SanitizerKind::ShiftExponent) &&
4623 isa<llvm::IntegerType>(Ops.LHS->getType())) {
4624 CodeGenFunction::SanitizerScope SanScope(&CGF);
4625 bool RHSIsSigned = Ops.rhsHasSignedIntegerRepresentation();
4626 llvm::Value *Valid = Builder.CreateICmpULE(
4627 Ops.RHS, GetMaximumShiftAmount(Ops.LHS, Ops.RHS, RHSIsSigned));
4628 EmitBinOpCheck(std::make_pair(Valid, SanitizerKind::SO_ShiftExponent), Ops);
4631 if (Ops.Ty->hasUnsignedIntegerRepresentation())
4632 return Builder.CreateLShr(Ops.LHS, RHS,
"shr");
4633 return Builder.CreateAShr(Ops.LHS, RHS,
"shr");
4641 default: llvm_unreachable(
"unexpected element type");
4642 case BuiltinType::Char_U:
4643 case BuiltinType::UChar:
4644 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequb_p :
4645 llvm::Intrinsic::ppc_altivec_vcmpgtub_p;
4646 case BuiltinType::Char_S:
4647 case BuiltinType::SChar:
4648 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequb_p :
4649 llvm::Intrinsic::ppc_altivec_vcmpgtsb_p;
4650 case BuiltinType::UShort:
4651 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequh_p :
4652 llvm::Intrinsic::ppc_altivec_vcmpgtuh_p;
4653 case BuiltinType::Short:
4654 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequh_p :
4655 llvm::Intrinsic::ppc_altivec_vcmpgtsh_p;
4656 case BuiltinType::UInt:
4657 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequw_p :
4658 llvm::Intrinsic::ppc_altivec_vcmpgtuw_p;
4659 case BuiltinType::Int:
4660 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequw_p :
4661 llvm::Intrinsic::ppc_altivec_vcmpgtsw_p;
4662 case BuiltinType::ULong:
4663 case BuiltinType::ULongLong:
4664 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequd_p :
4665 llvm::Intrinsic::ppc_altivec_vcmpgtud_p;
4666 case BuiltinType::Long:
4667 case BuiltinType::LongLong:
4668 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequd_p :
4669 llvm::Intrinsic::ppc_altivec_vcmpgtsd_p;
4670 case BuiltinType::Float:
4671 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpeqfp_p :
4672 llvm::Intrinsic::ppc_altivec_vcmpgtfp_p;
4673 case BuiltinType::Double:
4674 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_vsx_xvcmpeqdp_p :
4675 llvm::Intrinsic::ppc_vsx_xvcmpgtdp_p;
4676 case BuiltinType::UInt128:
4677 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequq_p
4678 : llvm::Intrinsic::ppc_altivec_vcmpgtuq_p;
4679 case BuiltinType::Int128:
4680 return (IT ==
VCMPEQ) ? llvm::Intrinsic::ppc_altivec_vcmpequq_p
4681 : llvm::Intrinsic::ppc_altivec_vcmpgtsq_p;
4686 llvm::CmpInst::Predicate UICmpOpc,
4687 llvm::CmpInst::Predicate SICmpOpc,
4688 llvm::CmpInst::Predicate FCmpOpc,
4690 TestAndClearIgnoreResultAssign();
4695 assert(
E->getOpcode() == BO_EQ ||
4696 E->getOpcode() == BO_NE);
4700 CGF, LHS, RHS, MPT,
E->getOpcode() == BO_NE);
4702 BinOpInfo BOInfo = EmitBinOps(
E);
4703 Value *LHS = BOInfo.LHS;
4704 Value *RHS = BOInfo.RHS;
4710 enum { CR6_EQ=0, CR6_EQ_REV, CR6_LT, CR6_LT_REV } CR6;
4712 llvm::Intrinsic::ID
ID = llvm::Intrinsic::not_intrinsic;
4715 Value *FirstVecArg = LHS,
4716 *SecondVecArg = RHS;
4721 switch(
E->getOpcode()) {
4722 default: llvm_unreachable(
"is not a comparison operation");
4734 std::swap(FirstVecArg, SecondVecArg);
4741 if (ElementKind == BuiltinType::Float) {
4743 ID = llvm::Intrinsic::ppc_altivec_vcmpgefp_p;
4744 std::swap(FirstVecArg, SecondVecArg);
4752 if (ElementKind == BuiltinType::Float) {
4754 ID = llvm::Intrinsic::ppc_altivec_vcmpgefp_p;
4759 std::swap(FirstVecArg, SecondVecArg);
4764 Value *CR6Param = Builder.getInt32(CR6);
4766 Result = Builder.CreateCall(F, {CR6Param, FirstVecArg, SecondVecArg});
4773 llvm::IntegerType *ResultTy = cast<llvm::IntegerType>(
Result->getType());
4774 if (ResultTy->getBitWidth() > 1 &&
4776 Result = Builder.CreateTrunc(
Result, Builder.getInt1Ty());
4781 if (BOInfo.isFixedPointOp()) {
4782 Result = EmitFixedPointBinOp(BOInfo);
4783 }
else if (LHS->
getType()->isFPOrFPVectorTy()) {
4784 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(CGF, BOInfo.FPFeatures);
4786 Result = Builder.CreateFCmp(FCmpOpc, LHS, RHS,
"cmp");
4788 Result = Builder.CreateFCmpS(FCmpOpc, LHS, RHS,
"cmp");
4790 Result = Builder.CreateICmp(SICmpOpc, LHS, RHS,
"cmp");
4795 !isa<llvm::ConstantPointerNull>(LHS) &&
4796 !isa<llvm::ConstantPointerNull>(RHS)) {
4805 LHS = Builder.CreateStripInvariantGroup(LHS);
4807 RHS = Builder.CreateStripInvariantGroup(RHS);
4810 Result = Builder.CreateICmp(UICmpOpc, LHS, RHS,
"cmp");
4816 return Builder.CreateSExt(
Result, ConvertType(
E->
getType()),
"sext");
4824 CETy = CTy->getElementType();
4826 LHS.first = Visit(
E->getLHS());
4827 LHS.second = llvm::Constant::getNullValue(LHS.first->getType());
4833 CTy->getElementType()) &&
4834 "The element types must always match.");
4837 RHS.first = Visit(
E->getRHS());
4838 RHS.second = llvm::Constant::getNullValue(RHS.first->getType());
4840 "The element types must always match.");
4843 Value *ResultR, *ResultI;
4847 ResultR = Builder.CreateFCmp(FCmpOpc, LHS.first, RHS.first,
"cmp.r");
4848 ResultI = Builder.CreateFCmp(FCmpOpc, LHS.second, RHS.second,
"cmp.i");
4852 ResultR = Builder.CreateICmp(UICmpOpc, LHS.first, RHS.first,
"cmp.r");
4853 ResultI = Builder.CreateICmp(UICmpOpc, LHS.second, RHS.second,
"cmp.i");
4856 if (
E->getOpcode() == BO_EQ) {
4857 Result = Builder.CreateAnd(ResultR, ResultI,
"and.ri");
4859 assert(
E->getOpcode() == BO_NE &&
4860 "Complex comparison other than == or != ?");
4861 Result = Builder.CreateOr(ResultR, ResultI,
"or.ri");
4873 if (
auto *ICE = dyn_cast<ImplicitCastExpr>(
E->getRHS())) {
4875 if (Kind == CK_IntegralCast || Kind == CK_LValueToRValue) {
4876 *SrcType = ICE->getSubExpr()->getType();
4888 bool Ignore = TestAndClearIgnoreResultAssign();
4907 RHS = Visit(
E->getRHS());
4923 RHS = Visit(
E->getRHS());
4966 Value *LHS = Visit(
E->getLHS());
4967 Value *RHS = Visit(
E->getRHS());
4969 if (LHS->
getType()->isFPOrFPVectorTy()) {
4970 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(
4972 LHS = Builder.CreateFCmp(llvm::CmpInst::FCMP_UNE, LHS, Zero,
"cmp");
4973 RHS = Builder.CreateFCmp(llvm::CmpInst::FCMP_UNE, RHS, Zero,
"cmp");
4975 LHS = Builder.CreateICmp(llvm::CmpInst::ICMP_NE, LHS, Zero,
"cmp");
4976 RHS = Builder.CreateICmp(llvm::CmpInst::ICMP_NE, RHS, Zero,
"cmp");
4978 Value *
And = Builder.CreateAnd(LHS, RHS);
4979 return Builder.CreateSExt(
And, ConvertType(
E->
getType()),
"sext");
4983 llvm::Type *ResTy = ConvertType(
E->
getType());
5004 if (InstrumentRegions &&
5009 Builder.CreateCondBr(RHSCond, RHSBlockCnt, FBlock);
5023 return Builder.CreateZExtOrBitCast(RHSCond, ResTy,
"land.ext");
5029 return llvm::Constant::getNullValue(ResTy);
5042 CodeGenFunction::ConditionalEvaluation eval(CGF);
5051 llvm::PHINode *PN = llvm::PHINode::Create(llvm::Type::getInt1Ty(VMContext), 2,
5053 for (llvm::pred_iterator PI = pred_begin(ContBlock), PE = pred_end(ContBlock);
5055 PN->addIncoming(llvm::ConstantInt::getFalse(VMContext), *PI);
5064 RHSBlock = Builder.GetInsertBlock();
5069 if (InstrumentRegions &&
5073 Builder.CreateCondBr(RHSCond, RHSBlockCnt, ContBlock);
5077 PN->addIncoming(RHSCond, RHSBlockCnt);
5087 PN->addIncoming(RHSCond, RHSBlock);
5097 PN->setDebugLoc(Builder.getCurrentDebugLocation());
5101 return Builder.CreateZExtOrBitCast(PN, ResTy,
"land.ext");
5109 Value *LHS = Visit(
E->getLHS());
5110 Value *RHS = Visit(
E->getRHS());
5112 if (LHS->
getType()->isFPOrFPVectorTy()) {
5113 CodeGenFunction::CGFPOptionsRAII FPOptsRAII(
5115 LHS = Builder.CreateFCmp(llvm::CmpInst::FCMP_UNE, LHS, Zero,
"cmp");
5116 RHS = Builder.CreateFCmp(llvm::CmpInst::FCMP_UNE, RHS, Zero,
"cmp");
5118 LHS = Builder.CreateICmp(llvm::CmpInst::ICMP_NE, LHS, Zero,
"cmp");
5119 RHS = Builder.CreateICmp(llvm::CmpInst::ICMP_NE, RHS, Zero,
"cmp");
5121 Value *
Or = Builder.CreateOr(LHS, RHS);
5122 return Builder.CreateSExt(
Or, ConvertType(
E->
getType()),
"sext");
5126 llvm::Type *ResTy = ConvertType(
E->
getType());
5147 if (InstrumentRegions &&
5152 Builder.CreateCondBr(RHSCond, FBlock, RHSBlockCnt);
5166 return Builder.CreateZExtOrBitCast(RHSCond, ResTy,
"lor.ext");
5172 return llvm::ConstantInt::get(ResTy, 1);
5185 CodeGenFunction::ConditionalEvaluation eval(CGF);
5195 llvm::PHINode *PN = llvm::PHINode::Create(llvm::Type::getInt1Ty(VMContext), 2,
5197 for (llvm::pred_iterator PI = pred_begin(ContBlock), PE = pred_end(ContBlock);
5199 PN->addIncoming(llvm::ConstantInt::getTrue(VMContext), *PI);
5211 RHSBlock = Builder.GetInsertBlock();
5216 if (InstrumentRegions &&
5220 Builder.CreateCondBr(RHSCond, ContBlock, RHSBlockCnt);
5224 PN->addIncoming(RHSCond, RHSBlockCnt);
5230 PN->addIncoming(RHSCond, RHSBlock);
5238 return Builder.CreateZExtOrBitCast(PN, ResTy,
"lor.ext");
5244 return Visit(
E->getRHS());
5269Value *ScalarExprEmitter::
5271 TestAndClearIgnoreResultAssign();
5274 CodeGenFunction::OpaqueValueMapping binding(CGF,
E);
5276 Expr *condExpr =
E->getCond();
5277 Expr *lhsExpr =
E->getTrueExpr();
5278 Expr *rhsExpr =
E->getFalseExpr();
5284 Expr *live = lhsExpr, *dead = rhsExpr;
5285 if (!CondExprBool) std::swap(live, dead);
5316 llvm::Value *LHS = Visit(lhsExpr);
5317 llvm::Value *RHS = Visit(rhsExpr);
5319 llvm::Type *condType = ConvertType(condExpr->
getType());
5320 auto *vecTy = cast<llvm::FixedVectorType>(condType);
5322 unsigned numElem = vecTy->getNumElements();
5323 llvm::Type *elemType = vecTy->getElementType();
5325 llvm::Value *zeroVec = llvm::Constant::getNullValue(vecTy);
5326 llvm::Value *TestMSB = Builder.CreateICmpSLT(CondV, zeroVec);
5327 llvm::Value *tmp = Builder.CreateSExt(
5328 TestMSB, llvm::FixedVectorType::get(elemType, numElem),
"sext");
5329 llvm::Value *tmp2 = Builder.CreateNot(tmp);
5332 llvm::Value *RHSTmp = RHS;
5333 llvm::Value *LHSTmp = LHS;
5334 bool wasCast =
false;
5335 llvm::VectorType *rhsVTy = cast<llvm::VectorType>(RHS->getType());
5336 if (rhsVTy->getElementType()->isFloatingPointTy()) {
5337 RHSTmp = Builder.CreateBitCast(RHS, tmp2->getType());
5338 LHSTmp = Builder.CreateBitCast(LHS, tmp->getType());
5342 llvm::Value *tmp3 = Builder.CreateAnd(RHSTmp, tmp2);
5343 llvm::Value *tmp4 = Builder.CreateAnd(LHSTmp, tmp);
5344 llvm::Value *tmp5 = Builder.CreateOr(tmp3, tmp4,
"cond");
5346 tmp5 = Builder.CreateBitCast(tmp5, RHS->getType());
5356 llvm::Value *LHS = Visit(lhsExpr);
5357 llvm::Value *RHS = Visit(rhsExpr);
5359 llvm::Type *CondType = ConvertType(condExpr->
getType());
5360 auto *VecTy = cast<llvm::VectorType>(CondType);
5361 llvm::Value *ZeroVec = llvm::Constant::getNullValue(VecTy);
5363 CondV = Builder.CreateICmpNE(CondV, ZeroVec,
"vector_cond");
5364 return Builder.CreateSelect(CondV, LHS, RHS,
"vector_select");
5373 llvm::Value *StepV = Builder.CreateZExtOrBitCast(CondV, CGF.
Int64Ty);
5382 llvm::Value *LHS = Visit(lhsExpr);
5383 llvm::Value *RHS = Visit(rhsExpr);
5386 assert(!RHS &&
"LHS and RHS types must match");
5389 return Builder.CreateSelect(CondV, LHS, RHS,
"cond");
5400 CodeGenFunction::ConditionalEvaluation eval(CGF);
5418 Value *LHS = Visit(lhsExpr);
5421 LHSBlock = Builder.GetInsertBlock();
5422 Builder.CreateBr(ContBlock);
5436 Value *RHS = Visit(rhsExpr);
5439 RHSBlock = Builder.GetInsertBlock();
5449 llvm::PHINode *PN = Builder.CreatePHI(LHS->
getType(), 2,
"cond");
5450 PN->addIncoming(LHS, LHSBlock);
5451 PN->addIncoming(RHS, RHSBlock);
5462 return Visit(
E->getChosenSubExpr());
5472Value *ScalarExprEmitter::VisitBlockExpr(
const BlockExpr *block) {
5478 Value *Src,
unsigned NumElementsDst) {
5479 static constexpr int Mask[] = {0, 1, 2, -1};
5480 return Builder.CreateShuffleVector(Src,
llvm::ArrayRef(Mask, NumElementsDst));
5500 const llvm::DataLayout &DL,
5501 Value *Src, llvm::Type *DstTy,
5502 StringRef Name =
"") {
5506 if (!SrcTy->isPointerTy() && !DstTy->isPointerTy())
5507 return Builder.CreateBitCast(Src, DstTy, Name);
5510 if (SrcTy->isPointerTy() && DstTy->isPointerTy())
5511 return Builder.CreatePointerBitCastOrAddrSpaceCast(Src, DstTy, Name);
5514 if (SrcTy->isPointerTy() && !DstTy->isPointerTy()) {
5516 if (!DstTy->isIntegerTy())
5517 Src = Builder.CreatePtrToInt(Src, DL.getIntPtrType(SrcTy));
5519 return Builder.CreateBitOrPointerCast(Src, DstTy, Name);
5523 if (!SrcTy->isIntegerTy())
5524 Src = Builder.CreateBitCast(Src, DL.getIntPtrType(DstTy));
5526 return Builder.CreateIntToPtr(Src, DstTy, Name);
5531 llvm::Type *DstTy = ConvertType(
E->
getType());
5533 llvm::Type *SrcTy = Src->
getType();
5534 unsigned NumElementsSrc =
5535 isa<llvm::VectorType>(SrcTy)
5536 ? cast<llvm::FixedVectorType>(SrcTy)->getNumElements()
5538 unsigned NumElementsDst =
5539 isa<llvm::VectorType>(DstTy)
5540 ? cast<llvm::FixedVectorType>(DstTy)->getNumElements()
5549 if (NumElementsSrc == 3 && NumElementsDst != 3) {
5554 Src->setName(
"astype");
5561 if (NumElementsSrc != 3 && NumElementsDst == 3) {
5562 auto *Vec4Ty = llvm::FixedVectorType::get(
5563 cast<llvm::VectorType>(DstTy)->getElementType(), 4);
5568 Src->setName(
"astype");
5573 Src, DstTy,
"astype");
5588 "Invalid scalar expression to emit");
5590 return ScalarExprEmitter(*
this, IgnoreResultAssign)
5591 .Visit(
const_cast<Expr *
>(
E));
5600 "Invalid scalar expression to emit");
5601 return ScalarExprEmitter(*this).EmitScalarConversion(Src, SrcTy, DstTy,
Loc);
5611 "Invalid complex -> scalar conversion");
5612 return ScalarExprEmitter(*
this)
5613 .EmitComplexToScalarConversion(Src, SrcTy, DstTy,
Loc);
5620 if (!PromotionType.
isNull())
5621 return ScalarExprEmitter(*this).EmitPromoted(
E, PromotionType);
5623 return ScalarExprEmitter(*this).Visit(
const_cast<Expr *
>(
E));
5629 bool isInc,
bool isPre) {
5630 return ScalarExprEmitter(*this).EmitScalarPrePostIncDec(
E, LV, isInc, isPre);
5637 Expr *BaseExpr =
E->getBase();
5640 llvm::Type *BaseTy =
5655 ScalarExprEmitter
Scalar(*
this);
5657 switch (
E->getOpcode()) {
5658#define COMPOUND_OP(Op) \
5659 case BO_##Op##Assign: \
5660 return Scalar.EmitCompoundAssignLValue(E, &ScalarExprEmitter::Emit##Op, \
5697 llvm_unreachable(
"Not valid compound assignment operators");
5700 llvm_unreachable(
"Unhandled compound assignment operator");
5715 llvm::LLVMContext &VMContext,
5721 llvm::Value *TotalOffset =
nullptr;
5724 if (isa<llvm::Constant>(GEPVal)) {
5727 Value *BasePtr_int =
5728 Builder.CreatePtrToInt(BasePtr, DL.getIntPtrType(BasePtr->
getType()));
5730 Builder.CreatePtrToInt(GEPVal, DL.getIntPtrType(GEPVal->
getType()));
5731 TotalOffset = Builder.CreateSub(GEPVal_int, BasePtr_int);
5732 return {TotalOffset, Builder.getFalse()};
5735 auto *GEP = cast<llvm::GEPOperator>(GEPVal);
5736 assert(GEP->getPointerOperand() == BasePtr &&
5737 "BasePtr must be the base of the GEP.");
5738 assert(GEP->isInBounds() &&
"Expected inbounds GEP");
5740 auto *IntPtrTy = DL.getIntPtrType(GEP->getPointerOperandType());
5743 auto *Zero = llvm::ConstantInt::getNullValue(IntPtrTy);
5744 auto *SAddIntrinsic =
5745 CGM.
getIntrinsic(llvm::Intrinsic::sadd_with_overflow, IntPtrTy);
5746 auto *SMulIntrinsic =
5747 CGM.
getIntrinsic(llvm::Intrinsic::smul_with_overflow, IntPtrTy);
5750 llvm::Value *OffsetOverflows = Builder.getFalse();
5754 llvm::Value *RHS) -> llvm::Value * {
5755 assert((Opcode == BO_Add || Opcode == BO_Mul) &&
"Can't eval binop");
5758 if (
auto *LHSCI = dyn_cast<llvm::ConstantInt>(LHS)) {
5759 if (
auto *RHSCI = dyn_cast<llvm::ConstantInt>(RHS)) {
5761 bool HasOverflow = mayHaveIntegerOverflow(LHSCI, RHSCI, Opcode,
5764 OffsetOverflows = Builder.getTrue();
5765 return llvm::ConstantInt::get(VMContext, N);
5770 auto *ResultAndOverflow = Builder.CreateCall(
5771 (Opcode == BO_Add) ? SAddIntrinsic : SMulIntrinsic, {LHS, RHS});
5772 OffsetOverflows = Builder.CreateOr(
5773 Builder.CreateExtractValue(ResultAndOverflow, 1), OffsetOverflows);
5774 return Builder.CreateExtractValue(ResultAndOverflow, 0);
5778 for (
auto GTI = llvm::gep_type_begin(GEP), GTE = llvm::gep_type_end(GEP);
5779 GTI != GTE; ++GTI) {
5780 llvm::Value *LocalOffset;
5781 auto *Index = GTI.getOperand();
5783 if (
auto *STy = GTI.getStructTypeOrNull()) {
5786 unsigned FieldNo = cast<llvm::ConstantInt>(Index)->getZExtValue();
5787 LocalOffset = llvm::ConstantInt::get(
5788 IntPtrTy, DL.getStructLayout(STy)->getElementOffset(FieldNo));
5793 llvm::ConstantInt::get(IntPtrTy, GTI.getSequentialElementStride(DL));
5794 auto *IndexS = Builder.CreateIntCast(Index, IntPtrTy,
true);
5795 LocalOffset = eval(BO_Mul, ElementSize, IndexS);
5800 if (!TotalOffset || TotalOffset == Zero)
5801 TotalOffset = LocalOffset;
5803 TotalOffset = eval(BO_Add, TotalOffset, LocalOffset);
5806 return {TotalOffset, OffsetOverflows};
5812 bool SignedIndices,
bool IsSubtraction,
5814 llvm::Type *PtrTy = Ptr->
getType();
5816 llvm::GEPNoWrapFlags NWFlags = llvm::GEPNoWrapFlags::inBounds();
5818 NWFlags |= llvm::GEPNoWrapFlags::noUnsignedWrap();
5823 if (!
SanOpts.
has(SanitizerKind::PointerOverflow))
5827 bool PerformNullCheck = !NullPointerIsDefined(
5828 Builder.GetInsertBlock()->getParent(), PtrTy->getPointerAddressSpace());
5831 bool PerformOverflowCheck =
5832 !isa<llvm::Constant>(GEPVal) && PtrTy->getPointerAddressSpace() == 0;
5834 if (!(PerformNullCheck || PerformOverflowCheck))
5839 SanitizerScope SanScope(
this);
5840 llvm::Type *
IntPtrTy = DL.getIntPtrType(PtrTy);
5845 assert((!isa<llvm::Constant>(EvaluatedGEP.
TotalOffset) ||
5847 "If the offset got constant-folded, we don't expect that there was an "
5850 auto *
Zero = llvm::ConstantInt::getNullValue(
IntPtrTy);
5865 if (PerformNullCheck) {
5873 auto *BaseIsNotNullptr =
Builder.CreateIsNotNull(Ptr);
5874 auto *ResultIsNotNullptr =
Builder.CreateIsNotNull(ComputedGEP);
5875 auto *Valid =
Builder.CreateICmpEQ(BaseIsNotNullptr, ResultIsNotNullptr);
5876 Checks.emplace_back(Valid, SanitizerKind::SO_PointerOverflow);
5879 if (PerformOverflowCheck) {
5884 llvm::Value *ValidGEP;
5886 if (SignedIndices) {
5892 auto *PosOrZeroValid =
Builder.CreateICmpUGE(ComputedGEP, IntPtr);
5893 auto *PosOrZeroOffset =
5895 llvm::Value *NegValid =
Builder.CreateICmpULT(ComputedGEP, IntPtr);
5897 Builder.CreateSelect(PosOrZeroOffset, PosOrZeroValid, NegValid);
5903 ValidGEP =
Builder.CreateICmpUGE(ComputedGEP, IntPtr);
5909 ValidGEP =
Builder.CreateICmpULE(ComputedGEP, IntPtr);
5911 ValidGEP =
Builder.CreateAnd(ValidGEP, NoOffsetOverflow);
5912 Checks.emplace_back(ValidGEP, SanitizerKind::SO_PointerOverflow);
5915 assert(!Checks.empty() &&
"Should have produced some checks.");
5919 llvm::Value *DynamicArgs[] = {IntPtr, ComputedGEP};
5920 EmitCheck(Checks, SanitizerHandler::PointerOverflow, StaticArgs, DynamicArgs);
5928 const Twine &Name) {
5929 if (!
SanOpts.
has(SanitizerKind::PointerOverflow)) {
5930 llvm::GEPNoWrapFlags NWFlags = llvm::GEPNoWrapFlags::inBounds();
5932 NWFlags |= llvm::GEPNoWrapFlags::noUnsignedWrap();
5934 return Builder.
CreateGEP(Addr, IdxList, elementType, Align, Name, NWFlags);
5940 elementType, Align);
Defines the clang::ASTContext interface.
ASTImporterLookupTable & LT
static llvm::Value * EmitCompare(CGBuilderTy &Builder, CodeGenFunction &CGF, const BinaryOperator *E, llvm::Value *LHS, llvm::Value *RHS, CompareKind Kind, const char *NameSuffix="")
CodeGenFunction::ComplexPairTy ComplexPairTy
#define VISITCOMP(CODE, UI, SI, FP, SIG)
static int getAsInt32(llvm::ConstantInt *C, llvm::Type *I32Ty)
static Value * emitPointerArithmetic(CodeGenFunction &CGF, const BinOpInfo &op, bool isSubtraction)
Emit pointer + index arithmetic.
static llvm::Value * EmitIsNegativeTestHelper(Value *V, QualType VType, const char *Name, CGBuilderTy &Builder)
static Value * createCastsForTypeOfSameSize(CGBuilderTy &Builder, const llvm::DataLayout &DL, Value *Src, llvm::Type *DstTy, StringRef Name="")
static bool matchesPostDecrInWhile(const UnaryOperator *UO, bool isInc, bool isPre, ASTContext &Ctx)
For the purposes of overflow pattern exclusion, does this match the "while(i--)" pattern?
static llvm::Intrinsic::ID GetIntrinsic(IntrinsicType IT, BuiltinType::Kind ElemKind)
static GEPOffsetAndOverflow EmitGEPOffsetInBytes(Value *BasePtr, Value *GEPVal, llvm::LLVMContext &VMContext, CodeGenModule &CGM, CGBuilderTy &Builder)
Evaluate given GEPVal, which is either an inbounds GEP, or a constant, and compute the total offset i...
static bool isCheapEnoughToEvaluateUnconditionally(const Expr *E, CodeGenFunction &CGF)
isCheapEnoughToEvaluateUnconditionally - Return true if the specified expression is cheap enough and ...
static bool PromotionIsPotentiallyEligibleForImplicitIntegerConversionCheck(QualType SrcType, QualType DstType)
static std::pair< ScalarExprEmitter::ImplicitConversionCheckKind, std::pair< llvm::Value *, SanitizerKind::SanitizerOrdinal > > EmitBitfieldTruncationCheckHelper(Value *Src, QualType SrcType, Value *Dst, QualType DstType, CGBuilderTy &Builder)
static Value * buildFMulAdd(llvm::Instruction *MulOp, Value *Addend, const CodeGenFunction &CGF, CGBuilderTy &Builder, bool negMul, bool negAdd)
static std::pair< ScalarExprEmitter::ImplicitConversionCheckKind, std::pair< llvm::Value *, SanitizerKind::SanitizerOrdinal > > EmitBitfieldSignChangeCheckHelper(Value *Src, QualType SrcType, Value *Dst, QualType DstType, CGBuilderTy &Builder)
static std::pair< ScalarExprEmitter::ImplicitConversionCheckKind, std::pair< llvm::Value *, SanitizerKind::SanitizerOrdinal > > EmitIntegerSignChangeCheckHelper(Value *Src, QualType SrcType, Value *Dst, QualType DstType, CGBuilderTy &Builder)
static int getMaskElt(llvm::ShuffleVectorInst *SVI, unsigned Idx, unsigned Off)
static std::pair< ScalarExprEmitter::ImplicitConversionCheckKind, std::pair< llvm::Value *, SanitizerKind::SanitizerOrdinal > > EmitIntegerTruncationCheckHelper(Value *Src, QualType SrcType, Value *Dst, QualType DstType, CGBuilderTy &Builder)
static Value * ConvertVec3AndVec4(CGBuilderTy &Builder, CodeGenFunction &CGF, Value *Src, unsigned NumElementsDst)
static Value * tryEmitFMulAdd(const BinOpInfo &op, const CodeGenFunction &CGF, CGBuilderTy &Builder, bool isSub=false)
static BinOpInfo createBinOpInfoFromIncDec(const UnaryOperator *E, llvm::Value *InVal, bool IsInc, FPOptions FPFeatures)
static Decl::Kind getKind(const Decl *D)
static QualType getPointeeType(const MemRegion *R)
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 ...
CharUnits getTypeAlignInChars(QualType T) const
Return the ABI-specified alignment of a (complete) type T, in characters.
ParentMapContext & getParentMapContext()
Returns the dynamic AST node parent map context.
unsigned getIntWidth(QualType T) const
const llvm::fltSemantics & getFloatTypeSemantics(QualType T) const
Return the APFloat 'semantics' for the specified scalar floating point type.
const ASTRecordLayout & getASTRecordLayout(const RecordDecl *D) const
Get or compute information about the layout of the specified record (struct/union/class) D,...
CanQualType getCanonicalType(QualType T) const
Return the canonical (structural) type corresponding to the specified potentially non-canonical type ...
QualType getVectorType(QualType VectorType, unsigned NumElts, VectorKind VecKind) const
Return the unique reference to a vector type of the specified element type and size.
QualType getPointerType(QualType T) const
Return the uniqued reference to the type for a pointer to the specified type.
const LangOptions & getLangOpts() const
bool isTypeIgnoredBySanitizer(const SanitizerMask &Mask, const QualType &Ty) const
Check if a type can have its sanitizer instrumentation elided based on its presence within an ignorel...
unsigned getOpenMPDefaultSimdAlign(QualType T) const
Get default simd alignment of the specified complete type in bits.
llvm::FixedPointSemantics getFixedPointSemantics(QualType Ty) const
bool hasSameUnqualifiedType(QualType T1, QualType T2) const
Determine whether the given types are equivalent after cvr-qualifiers have been removed.
const ArrayType * getAsArrayType(QualType T) const
Type Query functions.
uint64_t getTypeSize(QualType T) const
Return the size of the specified (complete) type T, in bits.
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
QualType getPromotedIntegerType(QualType PromotableType) const
Return the type that PromotableType will promote to: C99 6.3.1.1p2, assuming that PromotableType is a...
const VariableArrayType * getAsVariableArrayType(QualType T) const
QualType getComplexType(QualType T) const
Return the uniqued reference to the type for a complex number with the specified element type.
const TargetInfo & getTargetInfo() const
CharUnits toCharUnitsFromBits(int64_t BitSize) const
Convert a size in bits to a size in characters.
unsigned getTargetAddressSpace(LangAS AS) const
bool isPromotableIntegerType(QualType T) const
More type predicates useful for type checking/promotion.
uint64_t getCharWidth() const
Return the size of the character type, in bits.
ASTRecordLayout - This class contains layout information for one RecordDecl, which is a struct/union/...
unsigned getFieldCount() const
getFieldCount - Get the number of fields in the layout.
uint64_t getFieldOffset(unsigned FieldNo) const
getFieldOffset - Get the offset of the given field index, in bits.
CharUnits getBaseClassOffset(const CXXRecordDecl *Base) const
getBaseClassOffset - Get the offset, in chars, for the given base class.
AbstractConditionalOperator - An abstract base class for ConditionalOperator and BinaryConditionalOpe...
AddrLabelExpr - The GNU address of label extension, representing &&label.
Represents the index of the current element of an array being initialized by an ArrayInitLoopExpr.
ArraySubscriptExpr - [C99 6.5.2.1] Array Subscripting.
An Embarcadero array type trait, as used in the implementation of __array_rank and __array_extent.
QualType getElementType() const
AsTypeExpr - Clang builtin function __builtin_astype [OpenCL 6.2.4.2] This AST node provides support ...
AtomicExpr - Variadic atomic builtins: __atomic_exchange, __atomic_fetch_*, __atomic_load,...
A builtin binary operation expression such as "x + y" or "x <= y".
static Opcode getOpForCompoundAssignment(Opcode Opc)
bool isCompoundAssignmentOp() const
bool isShiftAssignOp() const
static bool isNullPointerArithmeticExtension(ASTContext &Ctx, Opcode Opc, const Expr *LHS, const Expr *RHS)
Return true if a binary operator using the specified opcode and operands would match the 'p = (i8*)nu...
BlockExpr - Adaptor class for mixing a BlockDecl with expressions.
This class is used for builtin types like 'int'.
bool isVirtual() const
Determines whether the base class is a virtual base class (or not).
QualType getType() const
Retrieves the type of the base class.
A boolean literal, per ([C++ lex.bool] Boolean literals).
A default argument (C++ [dcl.fct.default]).
A use of a default initializer in a constructor or in aggregate initialization.
Expr * getExpr()
Get the initialization expression that will be used.
Represents a delete expression for memory deallocation and destructor calls, e.g.
A C++ dynamic_cast expression (C++ [expr.dynamic.cast]).
Represents a new-expression for memory allocation and constructor calls, e.g: "new CXXNewExpr(foo)".
Represents a C++11 noexcept expression (C++ [expr.unary.noexcept]).
The null pointer literal (C++11 [lex.nullptr])
Represents a C++ pseudo-destructor (C++ [expr.pseudo]).
Represents a C++ struct/union/class.
A rewritten comparison expression that was originally written using operator syntax.
An expression "T()" which creates an rvalue of a non-class type T.
Represents the this expression in C++.
A C++ throw-expression (C++ [except.throw]).
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
path_iterator path_begin()
CastKind getCastKind() const
bool changesVolatileQualification() const
Return.
CharUnits - This is an opaque type for sizes expressed in character units.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
static CharUnits One()
One - Construct a CharUnits quantity of one.
bool isOne() const
isOne - Test whether the quantity equals one.
ChooseExpr - GNU builtin-in function __builtin_choose_expr.
Represents a 'co_await' expression.
bool hasProfileClangInstr() const
Check if Clang profile instrumenation is on.
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...
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.
A scoped helper to set the current debug location to the specified location or preferred location of ...
static ApplyDebugLocation CreateArtificial(CodeGenFunction &CGF)
Apply TemporaryLocation if it is valid.
static ApplyDebugLocation CreateEmpty(CodeGenFunction &CGF)
Set the IRBuilder to not attach debug locations.
Address CreateGEP(CodeGenFunction &CGF, Address Addr, llvm::Value *Index, const llvm::Twine &Name="")
virtual llvm::Constant * EmitNullMemberPointer(const MemberPointerType *MPT)
Create a null member pointer of the given type.
virtual llvm::Value * EmitMemberPointerIsNotNull(CodeGenFunction &CGF, llvm::Value *MemPtr, const MemberPointerType *MPT)
Determine if a member pointer is non-null. Returns an i1.
virtual llvm::Value * EmitMemberPointerComparison(CodeGenFunction &CGF, llvm::Value *L, llvm::Value *R, const MemberPointerType *MPT, bool Inequality)
Emit a comparison between two member pointers. Returns an i1.
virtual llvm::Value * EmitMemberPointerConversion(CodeGenFunction &CGF, const CastExpr *E, llvm::Value *Src)
Perform a derived-to-base, base-to-derived, or bitcast member pointer conversion.
void EmitPseudoVariable(CGBuilderTy &Builder, llvm::Instruction *Value, QualType Ty)
Emit a pseudo variable and debug info for an intermediate value if it does not correspond to a variab...
void addHeapAllocSiteMetadata(llvm::CallBase *CallSite, QualType AllocatedTy, SourceLocation Loc)
Add heapallocsite metadata for MSAllocator calls.
virtual void checkAndEmitLastprivateConditional(CodeGenFunction &CGF, const Expr *LHS)
Checks if the provided LVal is lastprivate conditional and emits the code to update the value of the ...
CodeGenFunction - This class organizes the per-function state that is used while generating LLVM code...
llvm::Value * EmitFromMemory(llvm::Value *Value, QualType Ty)
EmitFromMemory - Change a scalar value from its memory representation to its value representation.
std::pair< RValue, llvm::Value * > EmitAtomicCompareExchange(LValue Obj, RValue Expected, RValue Desired, SourceLocation Loc, llvm::AtomicOrdering Success=llvm::AtomicOrdering::SequentiallyConsistent, llvm::AtomicOrdering Failure=llvm::AtomicOrdering::SequentiallyConsistent, bool IsWeak=false, AggValueSlot Slot=AggValueSlot::ignored())
llvm::Value * EmitARCExtendBlockObject(const Expr *expr)
static bool ContainsLabel(const Stmt *S, bool IgnoreCaseStmts=false)
ContainsLabel - Return true if the statement contains a label in it.
CurrentSourceLocExprScope CurSourceLocExprScope
Source location information about the default argument or member initializer expression we're evaluat...
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.
bool sanitizePerformTypeCheck() const
Whether any type-checking sanitizers are enabled.
SanitizerSet SanOpts
Sanitizers enabled for this function.
LValue getOrCreateOpaqueLValueMapping(const OpaqueValueExpr *e)
Given an opaque value expression, return its LValue mapping if it exists, otherwise create one.
LValue EmitScalarCompoundAssignWithComplex(const CompoundAssignOperator *E, llvm::Value *&Result)
void EmitStoreThroughLValue(RValue Src, LValue Dst, bool isInit=false)
EmitStoreThroughLValue - Store the specified rvalue into the specified lvalue, where both are guarant...
static bool hasScalarEvaluationKind(QualType T)
LValue EmitCompoundAssignmentLValue(const CompoundAssignOperator *E)
void EmitBoundsCheck(const Expr *E, const Expr *Base, llvm::Value *Index, QualType IndexType, bool Accessed)
Emit a check that Base points into an array object, which we can access at index Index.
void EmitBitfieldConversionCheck(llvm::Value *Src, QualType SrcType, llvm::Value *Dst, QualType DstType, const CGBitFieldInfo &Info, SourceLocation Loc)
Emit a check that an [implicit] conversion of a bitfield.
void EmitTrapCheck(llvm::Value *Checked, SanitizerHandler CheckHandlerID, bool NoMerge=false)
Create a basic block that will call the trap intrinsic, and emit a conditional branch to it,...
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...
VlaSizePair getVLASize(const VariableArrayType *vla)
Returns an LLVM value that corresponds to the size, in non-variably-sized elements,...
LValue EmitLValue(const Expr *E, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
EmitLValue - Emit code to compute a designator that specifies the location of the expression.
void EmitVTablePtrCheckForCast(QualType T, Address Derived, bool MayBeNull, CFITypeCheckKind TCK, SourceLocation Loc)
Derived is the presumed address of an object of type T after a cast.
llvm::Value * EmitARCStoreWeak(Address addr, llvm::Value *value, bool ignored)
llvm::Value * getAsNaturalPointerTo(Address Addr, QualType PointeeType)
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
llvm::Value * EmitObjCProtocolExpr(const ObjCProtocolExpr *E)
llvm::Value * EmitObjCStringLiteral(const ObjCStringLiteral *E)
llvm::Constant * EmitCheckTypeDescriptor(QualType T)
Emit a description of a type in a format suitable for passing to a runtime sanitizer handler.
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....
llvm::Value * EmitObjCBoxedExpr(const ObjCBoxedExpr *E)
ComplexPairTy EmitComplexExpr(const Expr *E, bool IgnoreReal=false, bool IgnoreImag=false)
EmitComplexExpr - Emit the computation of the specified expression of complex type,...
RValue EmitLoadOfLValue(LValue V, SourceLocation Loc)
EmitLoadOfLValue - Given an expression that represents a value lvalue, this method emits the address ...
@ TCK_DowncastPointer
Checking the operand of a static_cast to a derived pointer type.
@ TCK_Store
Checking the destination of a store. Must be suitably sized and aligned.
void SetDivFPAccuracy(llvm::Value *Val)
Set the minimum required accuracy of the given sqrt operation based on CodeGenOpts.
RValue EmitObjCMessageExpr(const ObjCMessageExpr *E, ReturnValueSlot Return=ReturnValueSlot())
void EmitIgnoredExpr(const Expr *E)
EmitIgnoredExpr - Emit an expression in a context which ignores the result.
llvm::Type * ConvertTypeForMem(QualType T)
llvm::Value * EmitObjCDictionaryLiteral(const ObjCDictionaryLiteral *E)
LValue EmitCheckedLValue(const Expr *E, TypeCheckKind TCK)
Same as EmitLValue but additionally we generate checking code to guard against undefined behavior.
void markStmtMaybeUsed(const Stmt *S)
llvm::Value * EmitObjCArrayLiteral(const ObjCArrayLiteral *E)
void EmitStoreThroughBitfieldLValue(RValue Src, LValue Dst, llvm::Value **Result=nullptr)
EmitStoreThroughBitfieldLValue - Store Src into Dst with same constraints as EmitStoreThroughLValue.
llvm::Value * EmitObjCConsumeObject(QualType T, llvm::Value *Ptr)
ConstantEmission tryEmitAsConstant(DeclRefExpr *refExpr)
const TargetInfo & getTarget() const
std::pair< LValue, llvm::Value * > EmitARCStoreAutoreleasing(const BinaryOperator *e)
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...
llvm::Value * EmitMatrixIndexExpr(const Expr *E)
std::pair< LValue, llvm::Value * > EmitARCStoreUnsafeUnretained(const BinaryOperator *e, bool ignored)
Address EmitArrayToPointerDecay(const Expr *Array, LValueBaseInfo *BaseInfo=nullptr, TBAAAccessInfo *TBAAInfo=nullptr)
void maybeResetMCDCCondBitmap(const Expr *E)
Zero-init the MCDC temp value.
void EmitCheck(ArrayRef< std::pair< llvm::Value *, SanitizerKind::SanitizerOrdinal > > 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...
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.
RValue EmitCoyieldExpr(const CoyieldExpr &E, AggValueSlot aggSlot=AggValueSlot::ignored(), bool ignoreResult=false)
static bool ShouldNullCheckClassCastValue(const CastExpr *Cast)
llvm::Value * EmitComplexToScalarConversion(ComplexPairTy Src, QualType SrcTy, QualType DstTy, SourceLocation Loc)
Emit a conversion from the specified complex type to the specified destination type,...
llvm::Constant * EmitCheckSourceLocation(SourceLocation Loc)
Emit a description of a source location in a format suitable for passing to a runtime sanitizer handl...
void ErrorUnsupported(const Stmt *S, const char *Type)
ErrorUnsupported - Print out an error that codegen doesn't support the specified stmt yet.
CGDebugInfo * getDebugInfo()
void EmitBranch(llvm::BasicBlock *Block)
EmitBranch - Emit a branch to the specified basic block from the current insert block,...
llvm::Value * EmitARCReclaimReturnedObject(const Expr *e, bool allowUnsafeClaim)
void EmitCXXDeleteExpr(const CXXDeleteExpr *E)
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="")
Address GetAddressOfDerivedClass(Address Value, const CXXRecordDecl *Derived, CastExpr::path_const_iterator PathBegin, CastExpr::path_const_iterator PathEnd, bool NullCheckValue)
Address EmitCompoundStmt(const CompoundStmt &S, bool GetLast=false, AggValueSlot AVS=AggValueSlot::ignored())
llvm::Value * EmitToMemory(llvm::Value *Value, QualType Ty)
EmitToMemory - Change a scalar value from its value representation to its in-memory representation.
llvm::Value * authPointerToPointerCast(llvm::Value *ResultPtr, QualType SourceType, QualType DestType)
RValue EmitVAArg(VAArgExpr *VE, Address &VAListAddr, AggValueSlot Slot=AggValueSlot::ignored())
Generate code to get an argument from the passed in pointer and update it accordingly.
RValue EmitPseudoObjectRValue(const PseudoObjectExpr *e, AggValueSlot slot=AggValueSlot::ignored())
llvm::Value * EmitObjCSelectorExpr(const ObjCSelectorExpr *E)
llvm::CallInst * EmitNounwindRuntimeCall(llvm::FunctionCallee callee, const Twine &name="")
ASTContext & getContext() const
llvm::Value * EmitBuiltinAvailable(const VersionTuple &Version)
llvm::Value * EmitARCStoreStrong(LValue lvalue, llvm::Value *value, bool resultIgnored)
llvm::Value * EmitLoadOfScalar(Address Addr, bool Volatile, QualType Ty, SourceLocation Loc, AlignmentSource Source=AlignmentSource::Type, bool isNontemporal=false)
EmitLoadOfScalar - Load a scalar value from an address, taking care to appropriately convert from the...
llvm::Value * EmitBlockLiteral(const BlockExpr *)
Emit block literal.
void EmitNullabilityCheck(LValue LHS, llvm::Value *RHS, SourceLocation Loc)
Given an assignment *LHS = RHS, emit a test that checks if RHS is nonnull, if LHS is marked _Nonnull.
void maybeUpdateMCDCCondBitmap(const Expr *E, llvm::Value *Val)
Update the MCDC temp value with the condition's evaluated result.
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,...
ComplexPairTy EmitPromotedValue(ComplexPairTy result, QualType PromotionType)
llvm::Value * EmitPromotedScalarExpr(const Expr *E, QualType PromotionType)
void maybeUpdateMCDCTestVectorBitmap(const Expr *E)
Increment the profiler's counter for the given expression by StepV.
llvm::Type * ConvertType(QualType T)
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::Value * EvaluateExprAsBool(const Expr *E)
EvaluateExprAsBool - Perform the usual unary conversions on the specified expression and compare the ...
RValue EmitCoawaitExpr(const CoawaitExpr &E, AggValueSlot aggSlot=AggValueSlot::ignored(), bool ignoreResult=false)
llvm::Value * EmitWithOriginalRHSBitfieldAssignment(const BinaryOperator *E, llvm::Value **Previous, QualType *SrcType)
Retrieve the implicit cast expression of the rhs in a binary operator expression by passing pointers ...
llvm::Value * EmitCheckedInBoundsGEP(llvm::Type *ElemTy, llvm::Value *Ptr, ArrayRef< llvm::Value * > IdxList, bool SignedIndices, bool IsSubtraction, SourceLocation Loc, const Twine &Name="")
Same as IRBuilder::CreateInBoundsGEP, but additionally emits a check to detect undefined behavior whe...
llvm::Value * EmitDynamicCast(Address V, const CXXDynamicCastExpr *DCE)
llvm::Value * EmitBlockCopyAndAutorelease(llvm::Value *Block, QualType Ty)
uint64_t getProfileCount(const Stmt *S)
Get the profiler's count for the given statement.
llvm::Value * EmitScalarPrePostIncDec(const UnaryOperator *E, LValue LV, bool isInc, bool isPre)
LValue MakeAddrLValue(Address Addr, QualType T, AlignmentSource Source=AlignmentSource::Type)
llvm::Value * LoadCXXThis()
LoadCXXThis - Load the value of 'this'.
llvm::Value * getArrayInitIndex()
Get the index of the current ArrayInitLoopExpr, if any.
LValue EmitObjCIsaExpr(const ObjCIsaExpr *E)
llvm::Value * EmitScalarConversion(llvm::Value *Src, QualType SrcTy, QualType DstTy, SourceLocation Loc)
Emit a conversion from the specified type to the specified destination type, both of which are LLVM s...
std::pair< llvm::Value *, llvm::Value * > ComplexPairTy
RValue getOrCreateOpaqueRValueMapping(const OpaqueValueExpr *e)
Given an opaque value expression, return its RValue mapping if it exists, otherwise create one.
llvm::AtomicRMWInst * emitAtomicRMWInst(llvm::AtomicRMWInst::BinOp Op, Address Addr, llvm::Value *Val, llvm::AtomicOrdering Order=llvm::AtomicOrdering::SequentiallyConsistent, llvm::SyncScope::ID SSID=llvm::SyncScope::System, const AtomicExpr *AE=nullptr)
Emit an atomicrmw instruction, and applying relevant metadata when applicable.
llvm::Value * emitScalarConstant(const ConstantEmission &Constant, Expr *E)
RValue EmitCallExpr(const CallExpr *E, ReturnValueSlot ReturnValue=ReturnValueSlot(), llvm::CallBase **CallOrInvoke=nullptr)
void EnsureInsertPoint()
EnsureInsertPoint - Ensure that an insertion point is defined so that emitted IR has a place to go.
llvm::LLVMContext & getLLVMContext()
llvm::Value * EmitScalarExpr(const Expr *E, bool IgnoreResultAssign=false)
EmitScalarExpr - Emit the computation of the specified expression of LLVM scalar type,...
void incrementProfileCounter(const Stmt *S, llvm::Value *StepV=nullptr)
Increment the profiler's counter for the given statement by StepV.
llvm::Value * EmitARCRetainScalarExpr(const Expr *expr)
llvm::Value * EmitCXXNewExpr(const CXXNewExpr *E)
llvm::Type * convertTypeForLoadStore(QualType ASTTy, llvm::Type *LLVMTy=nullptr)
void EmitStoreOfScalar(llvm::Value *Value, Address Addr, bool Volatile, QualType Ty, AlignmentSource Source=AlignmentSource::Type, bool isInit=false, bool isNontemporal=false)
EmitStoreOfScalar - Store a scalar value to an address, taking care to appropriately convert from the...
RValue EmitAtomicExpr(AtomicExpr *E)
This class organizes the cross-function state that is used while generating LLVM code.
void EmitExplicitCastExprType(const ExplicitCastExpr *E, CodeGenFunction *CGF=nullptr)
Emit type info if type of an expression is a variably modified type.
llvm::FunctionCallee CreateRuntimeFunction(llvm::FunctionType *Ty, StringRef Name, llvm::AttributeList ExtraAttrs=llvm::AttributeList(), bool Local=false, bool AssumeConvergent=false)
Create or return a runtime function declaration with the specified type and name.
llvm::Constant * getNullPointer(llvm::PointerType *T, QualType QT)
Get target specific null pointer.
CodeGenTypes & getTypes()
const TargetInfo & getTarget() const
llvm::Constant * getMemberPointerConstant(const UnaryOperator *e)
const llvm::DataLayout & getDataLayout() const
CGCXXABI & getCXXABI() const
CGOpenMPRuntime & getOpenMPRuntime()
Return a reference to the configured OpenMP runtime.
const TargetCodeGenInfo & getTargetCodeGenInfo()
const CodeGenOptions & getCodeGenOpts() const
llvm::Function * getIntrinsic(unsigned IID, ArrayRef< llvm::Type * > Tys={})
llvm::Value * createOpenCLIntToSamplerConversion(const Expr *E, CodeGenFunction &CGF)
llvm::Constant * EmitNullConstant(QualType T)
Return the result of value-initializing the given type, i.e.
LangAS GetGlobalConstantAddressSpace() const
Return the AST address space of constant literal, which is used to emit the constant literal as globa...
llvm::ConstantInt * getSize(CharUnits numChars)
Emit the given number of characters as a value of type size_t.
llvm::Type * ConvertType(QualType T)
ConvertType - Convert type T into a llvm::Type.
llvm::Constant * emitAbstract(const Expr *E, QualType T)
Emit the result of the given expression as an abstract constant, asserting that it succeeded.
LValue - This represents an lvalue references.
bool isVolatileQualified() const
void setTBAAInfo(TBAAAccessInfo Info)
Address getAddress() const
const CGBitFieldInfo & getBitFieldInfo() const
RValue - This trivial value class is used to represent the result of an expression that is evaluated.
static RValue get(llvm::Value *V)
llvm::Value * getScalarVal() const
getScalarVal() - Return the Value* of this scalar value.
An abstract representation of an aligned address.
Address performAddrSpaceCast(CodeGen::CodeGenFunction &CGF, Address Addr, LangAS SrcAddr, LangAS DestAddr, llvm::Type *DestTy, bool IsNonNull=false) const
Complex values, per C99 6.2.5p11.
CompoundAssignOperator - For compound assignments (e.g.
CompoundLiteralExpr - [C99 6.5.2.5].
Represents the specialization of a concept - evaluates to a prvalue of type bool.
ConstantExpr - An expression that occurs in a constant context and optionally the result of evaluatin...
Represents a concrete matrix type with constant number of rows and columns.
unsigned getNumRows() const
Returns the number of rows in the matrix.
ConvertVectorExpr - Clang builtin function __builtin_convertvector This AST node provides support for...
Represents a 'co_yield' expression.
const Expr * getDefaultExpr() const
specific_decl_iterator - Iterates over a subrange of declarations stored in a DeclContext,...
A reference to a declared variable, function, enum, etc.
Represents a reference to #emded data.
ExplicitCastExpr - An explicit cast written in the source code.
Represents an expression – generally a full-expression – that introduces cleanups to be run at the en...
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,...
@ SE_AllowSideEffects
Allow any unmodeled side effect.
llvm::APSInt EvaluateKnownConstInt(const ASTContext &Ctx, SmallVectorImpl< PartialDiagnosticAt > *Diag=nullptr) const
EvaluateKnownConstInt - Call EvaluateAsRValue and return the folded 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.
bool isEvaluatable(const ASTContext &Ctx, SideEffectsKind AllowSideEffects=SE_NoSideEffects) const
isEvaluatable - Call EvaluateAsRValue to see if this expression can be constant folded without side-e...
bool EvaluateAsRValue(EvalResult &Result, const ASTContext &Ctx, bool InConstantContext=false) const
EvaluateAsRValue - Return true if this is a constant which we can fold to an rvalue using any crazy t...
bool isIntegerConstantExpr(const ASTContext &Ctx, SourceLocation *Loc=nullptr) const
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...
bool refersToBitField() const
Returns true if this expression is a gl-value that potentially refers to a bit-field.
An expression trait intrinsic.
ExtVectorType - Extended vector type.
Represents a member of a struct/union/class.
GNUNullExpr - Implements the GNU __null extension, which is a name for a null pointer constant that h...
Represents a C11 generic selection.
ImplicitCastExpr - Allows us to explicitly represent implicit type conversions, which have no direct ...
Represents an implicitly-generated value initialization of an object of a given type.
Describes an C or C++ initializer list.
@ PostDecrInWhile
while (count–)
bool isSignedOverflowDefined() const
bool isOverflowPatternExcluded(OverflowPatternExclusionKind Kind) const
std::string OverflowHandler
The name of the handler function to be called when -ftrapv is specified.
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
MatrixSubscriptExpr - Matrix subscript expression for the MatrixType extension.
Represents a matrix type, as defined in the Matrix Types clang extensions.
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
A pointer to member type per C++ 8.3.3 - Pointers to members.
ObjCArrayLiteral - used for objective-c array containers; as in: @["Hello", NSApp,...
A runtime availability query.
ObjCBoolLiteralExpr - Objective-C Boolean Literal.
ObjCBoxedExpr - used for generalized expression boxing.
ObjCDictionaryLiteral - AST node to represent objective-c dictionary literals; as in:"name" : NSUserN...
ObjCIsaExpr - Represent X->isa and X.isa when X is an ObjC 'id' type.
ObjCIvarRefExpr - A reference to an ObjC instance variable.
An expression that sends a message to the given Objective-C object or class.
Represents a pointer to an Objective C object.
const ObjCObjectType * getObjectType() const
Gets the type pointed to by this ObjC pointer.
ObjCProtocolExpr used for protocol expression in Objective-C.
ObjCSelectorExpr used for @selector in Objective-C.
ObjCStringLiteral, used for Objective-C string literals i.e.
OffsetOfExpr - [C99 7.17] - This represents an expression of the form offsetof(record-type,...
Helper class for OffsetOfExpr.
unsigned getArrayExprIndex() const
For an array element node, returns the index into the array of expressions.
FieldDecl * getField() const
For a field offsetof node, returns the field.
@ Array
An index into an array.
@ Identifier
A field in a dependent type, known only by its name.
@ Base
An implicit indirection through a C++ base class, when the field found is in a base class.
Kind getKind() const
Determine what kind of offsetof node this is.
CXXBaseSpecifier * getBase() const
For a base class node, returns the base specifier.
OpaqueValueExpr - An expression referring to an opaque object of a fixed type and value class.
This expression type represents an asterisk in an OpenACC Size-Expr, used in the 'tile' and 'gang' cl...
ParenExpr - This represents a parenthesized expression, e.g.
const Expr * getSubExpr() const
DynTypedNodeList getParents(const NodeT &Node)
Returns the parents of the given node (within the traversal scope).
PointerType - C99 6.7.5.1 - Pointer Declarators.
QualType getPointeeType() const
PseudoObjectExpr - An expression which accesses a pseudo-object l-value.
A (possibly-)qualified type.
bool mayBeDynamicClass() const
Returns true if it is a class and it might be dynamic.
bool isNull() const
Return true if this QualType doesn't point to a type yet.
const Type * getTypePtr() const
Retrieves a pointer to the underlying (unqualified) type.
LangAS getAddressSpace() const
Return the address space of this type.
Qualifiers::ObjCLifetime getObjCLifetime() const
Returns lifetime attribute of this type.
QualType getNonReferenceType() const
If Type is a reference type (e.g., const int&), returns the type that the reference refers to ("const...
QualType getCanonicalType() const
bool UseExcessPrecision(const ASTContext &Ctx)
bool mayBeNotDynamicClass() const
Returns true if it is not a class or if the class might not be dynamic.
@ OCL_Strong
Assigning into this object requires the old value to be released and the new value to be retained.
@ OCL_ExplicitNone
This object can be modified without requiring retains or releases.
@ OCL_None
There is no lifetime qualification on this type.
@ OCL_Weak
Reading or writing from this object requires a barrier call.
@ OCL_Autoreleasing
Assigning into this object requires a lifetime extension.
Represents a struct/union/class.
field_iterator field_end() const
field_iterator field_begin() const
A helper class that allows the use of isa/cast/dyncast to detect TagType objects of structs/unions/cl...
C++2a [expr.prim.req]: A requires-expression provides a concise way to express requirements on templa...
static constexpr SanitizerMask bitPosToMask(const unsigned Pos)
Create a mask with a bit enabled at position Pos.
Scope - A scope is a transient data structure that is used while parsing the program.
Sema - This implements semantic analysis and AST building for C.
ShuffleVectorExpr - clang-specific builtin-in function __builtin_shufflevector.
Represents an expression that computes the length of a parameter pack.
Represents a function call to one of __builtin_LINE(), __builtin_COLUMN(), __builtin_FUNCTION(),...
APValue EvaluateInContext(const ASTContext &Ctx, const Expr *DefaultExpr) const
Return the result of evaluating this SourceLocExpr in the specified (and possibly null) default argum...
SourceLocation getLocation() const
Encodes a location in the source.
StmtExpr - This is the GNU Statement Expression extension: ({int X=4; X;}).
RetTy Visit(PTR(Stmt) S, ParamTys... P)
StmtVisitor - This class implements a simple visitor for Stmt subclasses.
Stmt - This represents one statement.
SourceLocation getBeginLoc() const LLVM_READONLY
Represents a reference to a non-type template parameter that has been substituted with a template arg...
virtual bool useFP16ConversionIntrinsics() const
Check whether llvm intrinsics such as llvm.convert.to.fp16 should be used to convert to and from __fp...
VersionTuple getPlatformMinVersion() const
Retrieve the minimum desired version of the platform, to which the program should be compiled.
const llvm::fltSemantics & getHalfFormat() const
const llvm::fltSemantics & getBFloat16Format() const
const llvm::fltSemantics & getLongDoubleFormat() const
const llvm::fltSemantics & getFloat128Format() const
const llvm::fltSemantics & getIbm128Format() const
A type trait used in the implementation of various C++11 and Library TR1 trait templates.
bool isBooleanType() const
bool isSignedIntegerOrEnumerationType() const
Determines whether this is an integer type that is signed or an enumeration types whose underlying ty...
bool isUnsignedIntegerOrEnumerationType() const
Determines whether this is an integer type that is unsigned or an enumeration types whose underlying ...
bool isArithmeticType() const
bool isIntegerType() const
isIntegerType() does not include complex integers (a GCC extension).
const T * castAs() const
Member-template castAs<specific type>.
bool isReferenceType() const
const CXXRecordDecl * getPointeeCXXRecordDecl() const
If this is a pointer or reference to a RecordType, return the CXXRecordDecl that the type refers to.
bool isSveVLSBuiltinType() const
Determines if this is a sizeless type supported by the 'arm_sve_vector_bits' type attribute,...
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool isExtVectorType() const
bool isExtVectorBoolType() const
bool isOCLIntelSubgroupAVCType() const
bool isBuiltinType() const
Helper methods to distinguish type categories.
bool isAnyComplexType() const
bool isFixedPointType() const
Return true if this is a fixed point type according to ISO/IEC JTC1 SC22 WG14 N1169.
bool hasSignedIntegerRepresentation() const
Determine whether this type has an signed integer representation of some sort, e.g....
bool isMatrixType() const
bool isFunctionType() const
bool isVectorType() const
bool isRealFloatingType() const
Floating point categories.
bool isFloatingType() const
bool isUnsignedIntegerType() const
Return true if this is an integer type that is unsigned, according to C99 6.2.5p6 [which returns true...
const T * getAs() const
Member-template getAs<specific type>'.
bool isNullPtrType() const
UnaryExprOrTypeTraitExpr - expression with either a type or (unevaluated) expression operand.
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
bool canOverflow() const
Returns true if the unary operator can cause an overflow.
Represents a call to the builtin function __builtin_va_arg.
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
Represents a C array with a specified size that is not an integer-constant-expression.
Represents a GCC generic vector type.
VectorKind getVectorKind() const
QualType getElementType() const
WhileStmt - This represents a 'while' stmt.
Defines the clang::TargetInfo interface.
const AstTypeMatcher< PointerType > pointerType
Matches pointer types, but does not match Objective-C object pointer types.
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const internal::ArgumentAdaptingMatcherFunc< internal::HasMatcher > has
Matches AST nodes that have child AST nodes that match the provided matcher.
const internal::VariadicDynCastAllOfMatcher< Stmt, Expr > expr
Matches expressions.
bool LE(InterpState &S, CodePtr OpPC)
bool Zero(InterpState &S, CodePtr OpPC)
bool Load(InterpState &S, CodePtr OpPC)
bool GE(InterpState &S, CodePtr OpPC)
The JSON file list parser is used to communicate input to InstallAPI.
@ Result
The result type of a method or function.
CastKind
CastKind - The kind of operation required for a conversion.
const FunctionProtoType * T
@ Generic
not a target-specific vector type
Diagnostic wrappers for TextAPI types for error reporting.
cl::opt< bool > EnableSingleByteCoverage
llvm::Value * TotalOffset
llvm::Value * OffsetOverflows
Structure with information about how a bitfield should be accessed.
unsigned Size
The total size of the bit-field, in bits.
llvm::IntegerType * Int64Ty
llvm::IntegerType * Int8Ty
i8, i16, i32, and i64
llvm::IntegerType * SizeTy
llvm::IntegerType * Int32Ty
llvm::IntegerType * IntPtrTy
llvm::IntegerType * PtrDiffTy
CharUnits getPointerAlign() const
static TBAAAccessInfo getMayAliasInfo()
EvalResult is a struct with detailed info about an evaluated expression.
APValue Val
Val - This is the value the expression can be folded to.
bool has(SanitizerMask K) const
Check if a certain (single) sanitizer is enabled.
bool hasOneOf(SanitizerMask K) const
Check if one or more sanitizers are enabled.