40#include "llvm/ADT/ArrayRef.h"
41#include "llvm/ADT/DenseMap.h"
42#include "llvm/ADT/STLExtras.h"
43#include "llvm/ADT/SmallVector.h"
44#include "llvm/ADT/StringExtras.h"
72 bool HasLeadingEmptyMacro) {
101 Diag(
decl->getLocation(), diag::err_non_variable_decl_in_for);
102 decl->setInvalidDecl();
108 var->setInit(
nullptr);
121 var->setType(
type.withConst());
122 var->setARCPseudoStrong(
true);
135 enum { Equality, Inequality, Relational, ThreeWay } Kind;
138 if (!Op->isComparisonOp())
141 if (Op->getOpcode() == BO_EQ)
143 else if (Op->getOpcode() == BO_NE)
145 else if (Op->getOpcode() == BO_Cmp)
148 assert(Op->isRelationalOp());
151 Loc = Op->getOperatorLoc();
152 CanAssign = Op->getLHS()->IgnoreParenImpCasts()->isLValue();
154 switch (Op->getOperator()) {
158 case OO_ExclaimEqual:
163 case OO_GreaterEqual:
174 Loc = Op->getOperatorLoc();
175 CanAssign = Op->getArg(0)->IgnoreParenImpCasts()->isLValue();
186 S.
Diag(Loc, diag::warn_unused_comparison)
192 if (Kind == Inequality)
193 S.
Diag(Loc, diag::note_inequality_comparison_to_or_assign)
195 else if (Kind == Equality)
196 S.
Diag(Loc, diag::note_equality_comparison_to_assign)
208 StringRef Msg = A->getMessage();
212 return S.
Diag(Loc, diag::warn_unused_return_type)
213 << IsCtor << A << OffendingDecl <<
false << R1 << R2;
215 return S.
Diag(Loc, diag::warn_unused_constructor)
216 << A <<
false << R1 << R2;
217 return S.
Diag(Loc, diag::warn_unused_result) << A <<
false << R1 << R2;
221 return S.
Diag(Loc, diag::warn_unused_return_type)
222 << IsCtor << A << OffendingDecl <<
true << Msg << R1 << R2;
224 return S.
Diag(Loc, diag::warn_unused_constructor)
225 << A <<
true << Msg << R1 << R2;
226 return S.
Diag(Loc, diag::warn_unused_result) << A <<
true << Msg << R1 << R2;
235void DiagnoseUnused(
Sema &S,
const Expr *E, std::optional<unsigned> DiagID) {
236 bool NoDiscardOnly = !DiagID.has_value();
252 const Expr *WarnExpr;
258 if (!NoDiscardOnly) {
279 if (
const FullExpr *Temps = dyn_cast<FullExpr>(E))
280 E = Temps->getSubExpr();
282 E = TempExpr->getSubExpr();
288 if (
const auto *Cast = dyn_cast<CastExpr>(E))
289 if (
Cast->getCastKind() == CK_NoOp ||
290 Cast->getCastKind() == CK_ConstructorConversion ||
291 Cast->getCastKind() == CK_IntegralCast)
292 E =
Cast->getSubExpr()->IgnoreImpCasts();
294 if (
const CallExpr *CE = dyn_cast<CallExpr>(E)) {
298 auto [OffendingDecl, A] = CE->getUnusedResultAttr(S.
Context);
307 if (
const Decl *FD = CE->getCalleeDecl()) {
310 if (FD->hasAttr<PureAttr>()) {
311 S.
Diag(Loc, diag::warn_unused_call) << R1 << R2 <<
"pure";
314 if (FD->hasAttr<ConstAttr>()) {
315 S.
Diag(Loc, diag::warn_unused_call) << R1 << R2 <<
"const";
319 }
else if (
const auto *CE = dyn_cast<CXXConstructExpr>(E)) {
320 auto [OffendingDecl, A] = CE->getUnusedResultAttr(S.
Context);
324 }
else if (
const auto *ILE = dyn_cast<InitListExpr>(E)) {
325 if (
const TagDecl *TD = ILE->getType()->getAsTagDecl()) {
331 }
else if (ShouldSuppress)
336 if (S.
getLangOpts().ObjCAutoRefCount && ME->isDelegateInitCall()) {
337 S.
Diag(Loc, diag::err_arc_unused_init_message) << R1;
341 auto [OffendingDecl, A] = ME->getUnusedResultAttr(S.
Context);
346 const Expr *Source = POE->getSyntacticForm();
349 POE->getNumSemanticExprs() == 1 &&
351 return DiagnoseUnused(S, POE->getSemanticExpr(0), DiagID);
353 DiagID = diag::warn_unused_container_subscript_expr;
355 DiagID = diag::warn_unused_property_expr;
357 = dyn_cast<CXXFunctionalCastExpr>(E)) {
358 const Expr *E = FC->getSubExpr();
360 E = TE->getSubExpr();
364 if (
const CXXRecordDecl *RD = CE->getType()->getAsCXXRecordDecl())
365 if (!RD->getAttr<WarnUnusedAttr>())
381 S.
Diag(Loc, diag::warn_unused_voidptr)
391 S.
Diag(Loc, diag::warn_unused_volatile) << R1 << R2;
398 if (DiagID == diag::warn_unused_comma_left_operand && S.
isSFINAEContext())
402 S.
PDiag(*DiagID) << R1 << R2);
407 if (
const LabelStmt *Label = dyn_cast_if_present<LabelStmt>(S))
408 S = Label->getSubStmt();
410 const Expr *E = dyn_cast_if_present<Expr>(S);
414 DiagnoseUnused(*
this, E, DiagID);
439 const unsigned NumElts = Elts.size();
444 const unsigned MixedDeclsCodeID =
getLangOpts().C99
445 ? diag::warn_mixed_decls_code
446 : diag::ext_mixed_decls_code;
469 for (
unsigned i = 0; i != NumElts - 1; ++i)
505 auto CheckAndFinish = [&](
Expr *E) {
512 llvm::APSInt TempVal;
529 return CheckAndFinish(Val.
get());
536 assert((LHSVal.
isInvalid() || LHSVal.
get()) &&
"missing LHS value");
539 "missing RHS value");
542 Diag(CaseLoc, diag::err_case_not_in_switch);
553 Diag(CaseLoc, diag::err_acc_branch_in_out_compute_construct)
559 CaseLoc, DotDotDotLoc, ColonLoc);
572 Diag(DefaultLoc, diag::err_default_not_in_switch);
578 Diag(DefaultLoc, diag::err_acc_branch_in_out_compute_construct)
593 Diag(IdentLoc, diag::err_redefinition_of_label) << TheDecl->
getDeclName();
600 !
Context.getSourceManager().isInSystemHeader(IdentLoc))
601 Diag(IdentLoc, diag::warn_reserved_extern_symbol)
602 << TheDecl << static_cast<int>(Status);
606 if (
getCurScope()->isInOpenACCComputeConstructScope())
637 for (
const auto *A : Attrs) {
638 if (A->getKind() == attr::MustTail) {
653 if (!SemanticAttrs.empty())
663 Expr *E = R->getRetValue();
669 if (!checkMustTailAttr(St, MTA))
675 auto IgnoreImplicitAsWritten = [](
Expr *E) ->
Expr * {
682 R->setRetValue(IgnoreImplicitAsWritten(E));
686bool Sema::checkMustTailAttr(
const Stmt *St,
const Attr &MTA) {
688 "musttail cannot be checked from a dependent context");
691 auto IgnoreParenImplicitAsWritten = [](
const Expr *E) ->
const Expr * {
698 const auto *CE = dyn_cast_or_null<CallExpr>(IgnoreParenImplicitAsWritten(E));
705 if (
const FunctionDecl *CalleeDecl = CE->getDirectCallee();
706 CalleeDecl && CalleeDecl->hasAttr<NotTailCalledAttr>()) {
707 Diag(St->
getBeginLoc(), diag::err_musttail_mismatch) <<
true << CalleeDecl;
708 Diag(CalleeDecl->getLocation(), diag::note_musttail_disabled_by_not_tail_called);
712 if (
const auto *EWC = dyn_cast<ExprWithCleanups>(E)) {
713 if (EWC->cleanupsHaveSideEffects()) {
714 Diag(St->
getBeginLoc(), diag::err_musttail_needs_trivial_args) << &MTA;
725 ft_non_static_member,
726 ft_pointer_to_member,
727 } MemberType = ft_non_member;
730 const FunctionProtoType *
Func;
731 const CXXMethodDecl *
Method =
nullptr;
732 } CallerType, CalleeType;
734 auto GetMethodType = [
this, St, MTA](
const CXXMethodDecl *CMD, FuncType &
Type,
735 bool IsCallee) ->
bool {
738 << IsCallee << isa<CXXDestructorDecl>(CMD);
740 Diag(CMD->getBeginLoc(), diag::note_musttail_structors_forbidden)
746 Type.MemberType = FuncType::ft_static_member;
748 Type.This = CMD->getFunctionObjectParameterType();
749 Type.MemberType = FuncType::ft_non_static_member;
751 Type.Func = CMD->getType()->castAs<FunctionProtoType>();
755 const auto *CallerDecl = dyn_cast<FunctionDecl>(
CurContext);
766 Diag(St->
getBeginLoc(), diag::err_musttail_forbidden_from_this_context)
767 << &MTA << ContextType;
769 }
else if (
const auto *CMD = dyn_cast<CXXMethodDecl>(
CurContext)) {
771 if (!GetMethodType(CMD, CallerType,
false))
775 CallerType.Func = CallerDecl->getType()->getAs<FunctionProtoType>();
778 const Expr *CalleeExpr = CE->getCallee()->IgnoreParens();
779 const auto *CalleeBinOp = dyn_cast<BinaryOperator>(CalleeExpr);
780 SourceLocation CalleeLoc = CE->getCalleeDecl()
781 ? CE->getCalleeDecl()->getBeginLoc()
785 if (
const CXXMethodDecl *CMD =
786 dyn_cast_or_null<CXXMethodDecl>(CE->getCalleeDecl())) {
788 if (!GetMethodType(CMD, CalleeType,
true))
790 }
else if (CalleeBinOp && CalleeBinOp->isPtrMemOp()) {
793 CalleeBinOp->getRHS()->getType()->castAs<MemberPointerType>();
795 Context.getCanonicalTagType(MPT->getMostRecentCXXRecordDecl());
796 CalleeType.Func = MPT->getPointeeType()->castAs<FunctionProtoType>();
797 CalleeType.MemberType = FuncType::ft_pointer_to_member;
810 if (!CalleeType.Func || !CallerType.Func) {
812 if (!CalleeType.Func && CE->getDirectCallee()) {
813 Diag(CE->getDirectCallee()->getBeginLoc(),
814 diag::note_musttail_fix_non_prototype);
816 if (!CallerType.Func)
817 Diag(CallerDecl->getBeginLoc(), diag::note_musttail_fix_non_prototype);
828 if (CallerType.Func->getCallConv() != CalleeType.Func->getCallConv()) {
829 if (
const auto *ND = dyn_cast_or_null<NamedDecl>(CE->getCalleeDecl()))
831 <<
true << ND->getDeclName();
833 Diag(St->
getBeginLoc(), diag::err_musttail_callconv_mismatch) <<
false;
834 Diag(CalleeLoc, diag::note_musttail_callconv_mismatch)
841 if (CalleeType.Func->isVariadic() || CallerType.Func->isVariadic()) {
846 const auto *CalleeDecl = CE->getCalleeDecl();
847 if (CalleeDecl && CalleeDecl->hasAttr<CXX11NoReturnAttr>()) {
853 if (CallerType.This.isNull() != CalleeType.This.isNull()) {
854 if (
const auto *ND = dyn_cast_or_null<NamedDecl>(CE->getCalleeDecl())) {
856 << CallerType.MemberType << CalleeType.MemberType <<
true
857 << ND->getDeclName();
858 Diag(CalleeLoc, diag::note_musttail_callee_defined_here)
859 << ND->getDeclName();
862 << CallerType.MemberType << CalleeType.MemberType <<
false;
867 auto CheckTypesMatch = [
this](FuncType CallerType, FuncType CalleeType,
868 PartialDiagnostic &PD) ->
bool {
876 auto DoTypesMatch = [
this, &PD](QualType A, QualType B,
877 unsigned Select) ->
bool {
878 if (!
Context.hasSimilarType(A, B)) {
885 if (!CallerType.This.isNull() &&
889 if (!DoTypesMatch(CallerType.Func->getReturnType(),
893 if (CallerType.Func->getNumParams() != CalleeType.Func->getNumParams()) {
895 << CalleeType.Func->getNumParams();
899 ArrayRef<QualType> CalleeParams = CalleeType.Func->getParamTypes();
900 ArrayRef<QualType> CallerParams = CallerType.Func->getParamTypes();
901 size_t N = CallerType.Func->getNumParams();
902 for (
size_t I = 0; I < N; I++) {
903 if (!DoTypesMatch(CalleeParams[I], CallerParams[I],
905 PD << static_cast<int>(I) + 1;
913 PartialDiagnostic PD =
PDiag(diag::note_musttail_mismatch);
914 if (!CheckTypesMatch(CallerType, CalleeType, PD)) {
915 if (
const auto *ND = dyn_cast_or_null<NamedDecl>(CE->getCalleeDecl()))
917 <<
true << ND->getDeclName();
928 for (
auto ArgExpr : CE->arguments()) {
930 Context, ArgExpr->getType(),
false);
939 typedef EvaluatedExprVisitor<CommaVisitor> Inherited;
942 CommaVisitor(Sema &SemaRef) : Inherited(SemaRef.Context), SemaRef(SemaRef) {}
943 void VisitBinaryOperator(BinaryOperator *E) {
946 EvaluatedExprVisitor<CommaVisitor>::VisitBinaryOperator(E);
957 if (
Cond.isInvalid())
960 bool ConstevalOrNegatedConsteval =
965 assert((CondExpr || ConstevalOrNegatedConsteval) &&
966 "If statement: missing condition");
970 CommaVisitor(*this).Visit(CondExpr);
972 if (!ConstevalOrNegatedConsteval && !elseStmt)
975 if (ConstevalOrNegatedConsteval ||
977 auto DiagnoseLikelihood = [&](
const Stmt *S) {
979 Diags.Report(A->getLocation(),
980 diag::warn_attribute_has_no_effect_on_compile_time_if)
981 << A << ConstevalOrNegatedConsteval << A->getRange();
983 diag::note_attribute_has_no_effect_on_compile_time_if_here)
984 << ConstevalOrNegatedConsteval
991 DiagnoseLikelihood(thenStmt);
992 DiagnoseLikelihood(elseStmt);
994 std::tuple<bool, const Attr *, const Attr *> LHC =
996 if (std::get<0>(LHC)) {
997 const Attr *ThenAttr = std::get<1>(LHC);
998 const Attr *ElseAttr = std::get<2>(LHC);
1000 diag::warn_attributes_likelihood_ifstmt_conflict)
1001 << ThenAttr << ThenAttr->
getRange();
1003 << ElseAttr << ElseAttr->
getRange();
1007 if (ConstevalOrNegatedConsteval) {
1013 if (FD && FD->isImmediateFunction())
1017 Diags.Report(IfLoc, diag::warn_consteval_if_always_true) << Immediate;
1029 return BuildIfStmt(IfLoc, StatementKind, LParenLoc, InitStmt,
Cond, RParenLoc,
1030 thenStmt, ElseLoc, elseStmt);
1039 if (
Cond.isInvalid())
1047 Cond.get().first,
Cond.get().second, LParenLoc,
1048 RParenLoc, thenStmt, ElseLoc, elseStmt);
1052 struct CaseCompareFunctor {
1053 bool operator()(
const std::pair<llvm::APSInt, CaseStmt*> &LHS,
1054 const llvm::APSInt &RHS) {
1055 return LHS.first < RHS;
1057 bool operator()(
const std::pair<llvm::APSInt, CaseStmt*> &LHS,
1058 const std::pair<llvm::APSInt, CaseStmt*> &RHS) {
1059 return LHS.first < RHS.first;
1061 bool operator()(
const llvm::APSInt &LHS,
1062 const std::pair<llvm::APSInt, CaseStmt*> &RHS) {
1063 return LHS < RHS.first;
1071 const std::pair<llvm::APSInt, CaseStmt*>& rhs) {
1072 if (lhs.first < rhs.first)
1075 if (lhs.first == rhs.first &&
1076 lhs.second->getCaseLoc() < rhs.second->getCaseLoc())
1083static bool CmpEnumVals(
const std::pair<llvm::APSInt, EnumConstantDecl*>& lhs,
1084 const std::pair<llvm::APSInt, EnumConstantDecl*>& rhs)
1086 return lhs.first < rhs.first;
1091static bool EqEnumVals(
const std::pair<llvm::APSInt, EnumConstantDecl*>& lhs,
1092 const std::pair<llvm::APSInt, EnumConstantDecl*>& rhs)
1094 return lhs.first == rhs.first;
1100 if (
const auto *FE = dyn_cast<FullExpr>(E))
1101 E = FE->getSubExpr();
1102 while (
const auto *ImpCast = dyn_cast<ImplicitCastExpr>(E)) {
1103 if (ImpCast->getCastKind() != CK_IntegralCast)
break;
1104 E = ImpCast->getSubExpr();
1120 return S.
Diag(Loc, diag::err_typecheck_statement_requires_integer) << T;
1125 return S.
Diag(Loc, diag::err_switch_incomplete_class_type)
1126 << T <<
Cond->getSourceRange();
1131 return S.
Diag(Loc, diag::err_switch_explicit_conversion) << T << ConvTy;
1142 return S.
Diag(Loc, diag::err_switch_multiple_conversions) << T;
1153 llvm_unreachable(
"conversion functions are permitted");
1155 } SwitchDiagnoser(
Cond);
1165 if (!
Cond->isTypeDependent() &&
1166 !
Cond->getType()->isIntegralOrEnumerationType())
1177 Expr *CondExpr =
Cond.get().second;
1178 assert((
Cond.isInvalid() || CondExpr) &&
"switch with no condition");
1191 Diag(SwitchLoc, diag::warn_bool_switch_condition)
1199 LParenLoc, RParenLoc);
1205static void AdjustAPSInt(llvm::APSInt &Val,
unsigned BitWidth,
bool IsSigned) {
1206 Val = Val.extOrTrunc(BitWidth);
1207 Val.setIsSigned(IsSigned);
1213 unsigned UnpromotedWidth,
bool UnpromotedSign) {
1221 if (UnpromotedWidth < Val.getBitWidth()) {
1222 llvm::APSInt ConvVal(Val);
1223 AdjustAPSInt(ConvVal, UnpromotedWidth, UnpromotedSign);
1224 AdjustAPSInt(ConvVal, Val.getBitWidth(), Val.isSigned());
1229 S.
Diag(Loc, diag::warn_case_value_overflow) <<
toString(Val, 10)
1240 const Expr *CaseExpr,
1241 EnumValsTy::iterator &EI,
1242 EnumValsTy::iterator &EIEnd,
1243 const llvm::APSInt &Val) {
1249 if (
const VarDecl *VD = dyn_cast<VarDecl>(DRE->getDecl())) {
1258 if (ED->
hasAttr<FlagEnumAttr>())
1261 while (EI != EIEnd && EI->first < Val)
1264 if (EI != EIEnd && EI->first == Val)
1275 const EnumType *CondEnumType = CondType->
getAsCanonical<EnumType>();
1276 const EnumType *CaseEnumType = CaseType->
getAsCanonical<EnumType>();
1277 if (!CondEnumType || !CaseEnumType)
1281 if (!CondEnumType->getDecl()->getIdentifier() &&
1282 !CondEnumType->getDecl()->getTypedefNameForAnonDecl())
1284 if (!CaseEnumType->getDecl()->getIdentifier() &&
1285 !CaseEnumType->getDecl()->getTypedefNameForAnonDecl())
1291 S.
Diag(Case->
getExprLoc(), diag::warn_comparison_of_mixed_enum_types_switch)
1292 << CondType << CaseType <<
Cond->getSourceRange()
1301 assert(SS ==
getCurFunction()->SwitchStack.back().getPointer() &&
1302 "switch stack missing push/pop!");
1317 SS->
setBody(BodyStmt, SwitchLoc);
1331 const Expr *CondExprBeforePromotion = CondExpr;
1337 bool HasDependentValue
1339 unsigned CondWidth = HasDependentValue ? 0 :
Context.getIntWidth(CondType);
1346 unsigned CondWidthBeforePromotion
1347 = HasDependentValue ? 0 :
Context.getIntWidth(CondTypeBeforePromotion);
1348 bool CondIsSignedBeforePromotion
1355 CaseValsTy CaseVals;
1358 typedef std::vector<std::pair<llvm::APSInt, CaseStmt*> > CaseRangesTy;
1359 CaseRangesTy CaseRanges;
1363 bool CaseListIsErroneous =
false;
1369 SC = SC->getNextSwitchCase()) {
1371 if (
DefaultStmt *DS = dyn_cast<DefaultStmt>(SC)) {
1372 if (TheDefaultStmt) {
1373 Diag(DS->getDefaultLoc(), diag::err_multiple_default_labels_defined);
1380 CaseListIsErroneous =
true;
1382 TheDefaultStmt = DS;
1390 HasDependentValue =
true;
1396 const Expr *LoBeforePromotion = Lo;
1403 CondIsSignedBeforePromotion);
1415 HasDependentValue =
true;
1418 CaseRanges.push_back(std::make_pair(LoVal, CS));
1420 CaseVals.push_back(std::make_pair(LoVal, CS));
1424 if (!HasDependentValue) {
1427 llvm::APSInt ConstantCondValue;
1428 bool HasConstantCond =
false;
1429 if (!TheDefaultStmt) {
1434 ConstantCondValue =
Result.Val.getInt();
1435 assert(!HasConstantCond ||
1436 (ConstantCondValue.getBitWidth() == CondWidth &&
1437 ConstantCondValue.isSigned() == CondIsSigned));
1438 Diag(SwitchLoc, diag::warn_switch_default);
1440 bool ShouldCheckConstantCond = HasConstantCond;
1445 if (!CaseVals.empty()) {
1446 for (
unsigned i = 0, e = CaseVals.size(); i != e; ++i) {
1447 if (ShouldCheckConstantCond &&
1448 CaseVals[i].first == ConstantCondValue)
1449 ShouldCheckConstantCond =
false;
1451 if (i != 0 && CaseVals[i].first == CaseVals[i-1].first) {
1454 StringRef PrevString, CurrString;
1457 if (
DeclRefExpr *DeclRef = dyn_cast<DeclRefExpr>(PrevCase)) {
1458 PrevString = DeclRef->getDecl()->getName();
1460 if (
DeclRefExpr *DeclRef = dyn_cast<DeclRefExpr>(CurrCase)) {
1461 CurrString = DeclRef->getDecl()->getName();
1464 CaseVals[i-1].first.toString(CaseValStr);
1466 if (PrevString == CurrString)
1467 Diag(CaseVals[i].second->getLHS()->getBeginLoc(),
1468 diag::err_duplicate_case)
1469 << (PrevString.empty() ? CaseValStr.str() : PrevString);
1471 Diag(CaseVals[i].second->getLHS()->getBeginLoc(),
1472 diag::err_duplicate_case_differing_expr)
1473 << (PrevString.empty() ? CaseValStr.str() : PrevString)
1474 << (CurrString.empty() ? CaseValStr.str() : CurrString)
1477 Diag(CaseVals[i - 1].second->getLHS()->getBeginLoc(),
1478 diag::note_duplicate_case_prev);
1481 CaseListIsErroneous =
true;
1488 if (!CaseRanges.empty()) {
1491 llvm::stable_sort(CaseRanges);
1494 std::vector<llvm::APSInt> HiVals;
1495 for (
unsigned i = 0, e = CaseRanges.size(); i != e; ++i) {
1496 llvm::APSInt &LoVal = CaseRanges[i].first;
1497 CaseStmt *CR = CaseRanges[i].second;
1500 const Expr *HiBeforePromotion = Hi;
1507 CondWidthBeforePromotion, CondIsSignedBeforePromotion);
1513 if (LoVal > HiVal) {
1516 CaseRanges.erase(CaseRanges.begin()+i);
1522 if (ShouldCheckConstantCond &&
1523 LoVal <= ConstantCondValue &&
1524 ConstantCondValue <= HiVal)
1525 ShouldCheckConstantCond =
false;
1527 HiVals.push_back(HiVal);
1533 for (
unsigned i = 0, e = CaseRanges.size(); i != e; ++i) {
1534 llvm::APSInt &CRLo = CaseRanges[i].first;
1535 llvm::APSInt &CRHi = HiVals[i];
1536 CaseStmt *CR = CaseRanges[i].second;
1541 llvm::APSInt OverlapVal(32);
1545 CaseValsTy::iterator I =
1546 llvm::lower_bound(CaseVals, CRLo, CaseCompareFunctor());
1547 if (I != CaseVals.end() && I->first < CRHi) {
1548 OverlapVal = I->first;
1549 OverlapStmt = I->second;
1553 I = std::upper_bound(I, CaseVals.end(), CRHi, CaseCompareFunctor());
1554 if (I != CaseVals.begin() && (I-1)->first >= CRLo) {
1555 OverlapVal = (I-1)->first;
1556 OverlapStmt = (I-1)->second;
1561 if (i && CRLo <= HiVals[i-1]) {
1562 OverlapVal = HiVals[i-1];
1563 OverlapStmt = CaseRanges[i-1].second;
1571 diag::note_duplicate_case_prev);
1574 CaseListIsErroneous =
true;
1580 if (!CaseListIsErroneous && !CaseListIsIncomplete &&
1581 ShouldCheckConstantCond) {
1584 Diag(CondExpr->
getExprLoc(), diag::warn_missing_case_for_condition)
1595 if (!CaseListIsErroneous && !CaseListIsIncomplete && !HasConstantCond &&
1598 if (!ED->isCompleteDefinition() || ED->enumerators().empty())
1605 for (
auto *EDI : ED->enumerators()) {
1606 llvm::APSInt Val = EDI->getInitVal();
1608 EnumVals.push_back(std::make_pair(Val, EDI));
1611 auto EI = EnumVals.begin(), EIEnd = llvm::unique(EnumVals,
EqEnumVals);
1614 for (CaseValsTy::const_iterator CI = CaseVals.begin();
1615 CI != CaseVals.end(); CI++) {
1616 Expr *CaseExpr = CI->second->getLHS();
1620 << CondTypeBeforePromotion;
1624 EI = EnumVals.begin();
1625 for (CaseRangesTy::const_iterator RI = CaseRanges.begin();
1626 RI != CaseRanges.end(); RI++) {
1627 Expr *CaseExpr = RI->second->getLHS();
1631 << CondTypeBeforePromotion;
1634 RI->second->getRHS()->EvaluateKnownConstInt(
Context);
1637 CaseExpr = RI->second->getRHS();
1641 << CondTypeBeforePromotion;
1645 auto CI = CaseVals.begin();
1646 auto RI = CaseRanges.begin();
1647 bool hasCasesNotInSwitch =
false;
1651 for (EI = EnumVals.begin(); EI != EIEnd; EI++) {
1653 switch (EI->second->getAvailability()) {
1670 if (EI->second->hasAttr<UnusedAttr>())
1674 while (CI != CaseVals.end() && CI->first < EI->first)
1677 if (CI != CaseVals.end() && CI->first == EI->first)
1681 for (; RI != CaseRanges.end(); RI++) {
1683 RI->second->getRHS()->EvaluateKnownConstInt(
Context);
1685 if (EI->first <= Hi)
1689 if (RI == CaseRanges.end() || EI->first < RI->first) {
1690 hasCasesNotInSwitch =
true;
1691 UnhandledNames.push_back(EI->second->getDeclName());
1695 if (TheDefaultStmt && UnhandledNames.empty() && ED->isClosedNonFlag())
1699 if (!UnhandledNames.empty()) {
1701 ? diag::warn_def_missing_case
1702 : diag::warn_missing_case)
1705 for (
size_t I = 0, E = std::min(UnhandledNames.size(), (
size_t)3);
1707 DB << UnhandledNames[I];
1710 if (!hasCasesNotInSwitch)
1718 diag::warn_empty_switch_body);
1722 if (CaseListIsErroneous)
1736 Context.hasSameUnqualifiedType(SrcType, DstType))
1743 if (!ED->isClosed())
1746 if (
Diags.isIgnored(diag::warn_not_in_enum_assignment, SrcExpr->
getExprLoc()))
1754 unsigned DstWidth =
Context.getIntWidth(DstType);
1758 if (ED->hasAttr<FlagEnumAttr>()) {
1767 auto &Values = It->second;
1770 Values.reserve(std::distance(ED->enumerator_begin(), ED->enumerator_end()));
1772 for (
auto *EC : ED->enumerators()) {
1773 Values.push_back(EC->getInitVal());
1781 Values.erase(llvm::unique(Values), Values.end());
1784 if (llvm::binary_search(Values, *RHSVal))
1794 if (
Cond.isInvalid())
1797 auto CondVal =
Cond.get();
1799 if (CondVal.second &&
1800 !
Diags.isIgnored(diag::warn_comma_operator, CondVal.second->getExprLoc()))
1801 CommaVisitor(*this).Visit(CondVal.second);
1816 WhileLoc, LParenLoc, RParenLoc);
1823 assert(
Cond &&
"ActOnDoStmt(): missing expression");
1855 DeclSetVector &Decls;
1861 DeclExtractor(
Sema &S, DeclSetVector &Decls,
1863 Inherited(S.Context),
1868 bool isSimple() {
return Simple; }
1871 void VisitMemberExpr(MemberExpr* E) {
1877 void VisitStmt(Stmt *S) { Simple =
false; }
1879 void VisitBinaryOperator(BinaryOperator *E) {
1888 void VisitUnaryOperator(UnaryOperator *E) {
1896 void VisitConditionalOperator(ConditionalOperator *E) {
1902 void VisitParenExpr(ParenExpr *E) {
1906 void VisitBinaryConditionalOperator(BinaryConditionalOperator *E) {
1911 void VisitIntegerLiteral(IntegerLiteral *E) { }
1912 void VisitFloatingLiteral(FloatingLiteral *E) { }
1913 void VisitCXXBoolLiteralExpr(CXXBoolLiteralExpr *E) { }
1914 void VisitCharacterLiteral(CharacterLiteral *E) { }
1915 void VisitGNUNullExpr(GNUNullExpr *E) { }
1916 void VisitImaginaryLiteral(ImaginaryLiteral *E) { }
1918 void VisitDeclRefExpr(DeclRefExpr *E) {
1919 VarDecl *VD = dyn_cast<VarDecl>(E->
getDecl());
1936 DeclSetVector &Decls;
1940 typedef EvaluatedExprVisitor<DeclMatcher> Inherited;
1942 DeclMatcher(Sema &S, DeclSetVector &Decls, Stmt *Statement) :
1943 Inherited(S.Context), Decls(Decls), FoundDecl(
false) {
1944 if (!Statement)
return;
1949 void VisitReturnStmt(ReturnStmt *S) {
1953 void VisitBreakStmt(BreakStmt *S) {
1957 void VisitGotoStmt(GotoStmt *S) {
1968 void CheckLValueToRValueCast(Expr *E) {
1975 if (ConditionalOperator *CO = dyn_cast<ConditionalOperator>(E)) {
1976 Visit(CO->getCond());
1977 CheckLValueToRValueCast(CO->getTrueExpr());
1978 CheckLValueToRValueCast(CO->getFalseExpr());
1982 if (BinaryConditionalOperator *BCO =
1983 dyn_cast<BinaryConditionalOperator>(E)) {
1984 CheckLValueToRValueCast(BCO->getOpaqueValue()->getSourceExpr());
1985 CheckLValueToRValueCast(BCO->getFalseExpr());
1992 void VisitDeclRefExpr(DeclRefExpr *E) {
1993 if (
const auto *VD = dyn_cast<VarDecl>(E->
getDecl())) {
1994 if (Decls.count(VD))
1996 }
else if (
const auto *MD = dyn_cast<CXXMethodDecl>(E->
getDecl());
2007 for (
const auto &
Capture : MD->getParent()->captures()) {
2008 if (!
Capture.capturesVariable())
2015 const auto *VD = dyn_cast<VarDecl>(
Capture.getCapturedVar());
2016 if (VD && Decls.count(VD))
2022 void VisitPseudoObjectExpr(PseudoObjectExpr *POE) {
2026 if (
auto *OVE = dyn_cast<OpaqueValueExpr>(S))
2028 Visit(OVE->getSourceExpr());
2034 bool FoundDeclInUse() {
return FoundDecl; }
2038 void CheckForLoopConditionalStatement(
Sema &S,
Expr *Second,
2041 if (!Second)
return;
2048 DeclSetVector Decls;
2050 DeclExtractor DE(S, Decls, Ranges);
2054 if (!DE.isSimple())
return;
2057 if (Decls.size() == 0)
return;
2060 for (
auto *VD : Decls)
2064 if (DeclMatcher(S, Decls, Second).FoundDeclInUse() ||
2065 DeclMatcher(S, Decls, Third).FoundDeclInUse() ||
2066 DeclMatcher(S, Decls, Body).FoundDeclInUse())
2070 if (Decls.size() > 4) {
2073 PDiag << (unsigned)Decls.size();
2074 for (
auto *VD : Decls)
2078 for (
auto Range : Ranges)
2081 S.
Diag(Ranges.begin()->getBegin(), PDiag);
2086 bool ProcessIterationStmt(
Sema &S,
Stmt* Statement,
bool &Increment,
2088 if (
auto Cleanups = dyn_cast<ExprWithCleanups>(Statement))
2089 if (!Cleanups->cleanupsHaveSideEffects())
2092 if (
UnaryOperator *UO = dyn_cast<UnaryOperator>(Statement)) {
2093 switch (UO->getOpcode()) {
2094 default:
return false;
2104 DRE = dyn_cast<DeclRefExpr>(UO->getSubExpr());
2112 default:
return false;
2120 DRE = dyn_cast<DeclRefExpr>(
Call->getArg(0));
2130 SourceLocation BreakLoc;
2131 SourceLocation ContinueLoc;
2132 bool InSwitch =
false;
2135 BreakContinueFinder(Sema &S,
const Stmt* Body) :
2136 Inherited(S.Context) {
2140 typedef ConstEvaluatedExprVisitor<BreakContinueFinder> Inherited;
2142 void VisitContinueStmt(
const ContinueStmt* E) {
2146 void VisitBreakStmt(
const BreakStmt* E) {
2151 void VisitSwitchStmt(
const SwitchStmt* S) {
2161 if (
const Stmt *Body = S->
getBody())
2166 void VisitForStmt(
const ForStmt *S) {
2173 void VisitWhileStmt(
const WhileStmt *) {
2178 void VisitDoStmt(
const DoStmt *) {
2183 void VisitCXXForRangeStmt(
const CXXForRangeStmt *S) {
2196 void VisitObjCForCollectionStmt(
const ObjCForCollectionStmt *S) {
2205 bool ContinueFound() {
return ContinueLoc.
isValid(); }
2206 bool BreakFound() {
return BreakLoc.
isValid(); }
2207 SourceLocation GetContinueLoc() {
return ContinueLoc; }
2208 SourceLocation GetBreakLoc() {
return BreakLoc; }
2217 void CheckForRedundantIteration(
Sema &S,
Expr *Third,
Stmt *Body) {
2219 if (!Body || !Third)
return;
2225 if (!LastStmt)
return;
2231 bool LoopIncrement, LastIncrement;
2234 if (!ProcessIterationStmt(S, Third, LoopIncrement, LoopDRE))
return;
2235 if (!ProcessIterationStmt(S, LastStmt, LastIncrement, LastDRE))
return;
2239 if (LoopIncrement != LastIncrement ||
2242 if (BreakContinueFinder(S, Body).ContinueFound())
return;
2245 << LastDRE->
getDecl() << LastIncrement;
2264 const Decl *NonVarSeen =
nullptr;
2265 bool VarDeclSeen =
false;
2266 for (
auto *DI : DS->decls()) {
2267 if (
VarDecl *VD = dyn_cast<VarDecl>(DI)) {
2270 Diag(DI->getLocation(),
2272 ? diag::warn_c17_non_local_variable_decl_in_for
2273 : diag::ext_c23_non_local_variable_decl_in_for);
2274 }
else if (!NonVarSeen) {
2288 if (NonVarSeen && !VarDeclSeen)
2291 : diag::ext_c23_non_variable_decl_in_for);
2295 if (!Second.
get().first)
2296 CheckForLoopConditionalStatement(*
this, Second.
get().second, third.
get(),
2298 CheckForRedundantIteration(*
this, third.
get(), Body);
2300 if (Second.
get().second &&
2301 !
Diags.isIgnored(diag::warn_comma_operator,
2302 Second.
get().second->getExprLoc()))
2303 CommaVisitor(*this).Visit(Second.
get().second);
2311 Body, ForLoc, LParenLoc, RParenLoc);
2331 if (
Decl->getType()->isUndeducedType()) {
2344 SemaRef.
Diag(Loc, DiagID) <<
Init->getType();
2348 Decl->getTypeSourceInfo()->getTypeLoc(),
Init, InitType, Info);
2351 SemaRef.
Diag(Loc, DiagID) <<
Init->getType();
2358 Decl->setType(InitType);
2376enum BeginEndFunction {
2385void NoteForRangeBeginEndFunction(
Sema &SemaRef,
Expr *E,
2386 BeginEndFunction BEF) {
2387 CallExpr *CE = dyn_cast<CallExpr>(E);
2395 std::string Description;
2396 bool IsTemplate =
false;
2403 SemaRef.
Diag(Loc, diag::note_for_range_begin_end)
2404 << BEF << IsTemplate << Description << E->
getType();
2416 Decl->setCXXForRangeImplicitVar(
true);
2439 return Diag(InitStmt->
getBeginLoc(), diag::err_objc_for_range_init_stmt)
2445 assert(DS &&
"first part of for range not a decl stmt");
2472 const auto DepthStr = std::to_string(S->
getDepth() / 2);
2474 VarDecl *RangeVar = BuildForRangeVarDecl(*
this, RangeLoc,
2475 Context.getAutoRRefDeductType(),
2476 std::string(
"__range") + DepthStr);
2478 diag::err_for_range_deduction_failure)) {
2493 ForLoc, CoawaitLoc, InitStmt, ColonLoc, RangeDecl.
get(),
2495 nullptr,
nullptr, DS, RParenLoc, Kind,
2496 LifetimeExtendTemps);
2497 if (R.isInvalid()) {
2519 ExprResult *EndExpr, BeginEndFunction *BEF) {
2529 auto BuildBegin = [&] {
2533 BeginMemberLookup, CandidateSet,
2534 BeginRange, BeginExpr);
2539 << ColonLoc << BEF_begin << BeginRange->
getType();
2552 diag::err_for_range_iter_deduction_failure)) {
2553 NoteForRangeBeginEndFunction(SemaRef, BeginExpr->
get(), *BEF);
2559 auto BuildEnd = [&] {
2563 EndMemberLookup, CandidateSet,
2568 << ColonLoc << BEF_end << EndRange->
getType();
2572 diag::err_for_range_iter_deduction_failure)) {
2573 NoteForRangeBeginEndFunction(SemaRef, EndExpr->
get(), *BEF);
2593 if (BeginMemberLookup.
empty() != EndMemberLookup.
empty()) {
2598 auto BuildNonmember = [&](
2608 switch (BuildFound()) {
2615 SemaRef.
PDiag(diag::err_for_range_invalid)
2616 << BeginRange->
getType() << BEFFound),
2623 diag::note_for_range_member_begin_end_ignored)
2624 << BeginRange->
getType() << BEFFound;
2628 llvm_unreachable(
"unexpected ForRangeStatus");
2630 if (BeginMemberLookup.
empty())
2631 return BuildNonmember(BEF_end, EndMemberLookup, BuildEnd, BuildBegin);
2632 return BuildNonmember(BEF_begin, BeginMemberLookup, BuildBegin, BuildEnd);
2664 AdjustedRange = SemaRef.
BuildUnaryOp(S, RangeLoc, UO_Deref, Range);
2669 S, ForLoc, CoawaitLoc, InitStmt, LoopVarDecl, ColonLoc,
2678 SemaRef.
Diag(RangeLoc, diag::err_for_range_dereference)
2681 S, ForLoc, CoawaitLoc, InitStmt, LoopVarDecl, ColonLoc,
2720 if (
auto *DD = dyn_cast<DecompositionDecl>(LoopVar))
2721 for (
auto *Binding : DD->bindings()) {
2722 if (!Binding->isParameterPack())
2723 Binding->setType(
Context.DependentTy);
2727 }
else if (!BeginDeclStmt.
get()) {
2746 QualType RangeType = Range->getType();
2749 diag::err_for_range_incomplete_type))
2754 const auto DepthStr = std::to_string(S->
getDepth() / 2);
2755 VarDecl *BeginVar = BuildForRangeVarDecl(*
this, ColonLoc, AutoType,
2756 std::string(
"__begin") + DepthStr);
2757 VarDecl *EndVar = BuildForRangeVarDecl(*
this, ColonLoc, AutoType,
2758 std::string(
"__end") + DepthStr);
2769 BeginExpr = BeginRangeRef;
2776 diag::err_for_range_iter_deduction_failure)) {
2777 NoteForRangeBeginEndFunction(*
this, BeginExpr.
get(), BEF_begin);
2785 Context, CAT->getSize(),
Context.getPointerDiffType(), RangeLoc);
2787 dyn_cast<VariableArrayType>(UnqAT)) {
2814 VAT->desugar(), RangeLoc))
2824 Context.getTrivialTypeSourceInfo(
2825 VAT->getElementType(), RangeLoc))
2833 SizeOfVLAExprR.
get(), SizeOfEachElementExprR.
get());
2840 llvm_unreachable(
"Unexpected array type in for-range");
2844 EndExpr =
ActOnBinOp(S, ColonLoc, tok::plus, EndRangeRef.
get(),
2849 diag::err_for_range_iter_deduction_failure)) {
2850 NoteForRangeBeginEndFunction(*
this, EndExpr.
get(), BEF_end);
2856 BeginEndFunction BEFFailure;
2858 *
this, BeginRangeRef.
get(), EndRangeRef.
get(), RangeType, BeginVar,
2859 EndVar, ColonLoc, CoawaitLoc, &CandidateSet, &BeginExpr, &EndExpr,
2863 BEFFailure == BEF_begin) {
2866 if (
DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(Range)) {
2868 QualType ArrayTy = PVD->getOriginalType();
2869 QualType PointerTy = PVD->getType();
2871 Diag(Range->getBeginLoc(), diag::err_range_on_array_parameter)
2872 << RangeLoc << PVD << ArrayTy << PointerTy;
2873 Diag(PVD->getLocation(), diag::note_declared_at);
2882 CoawaitLoc, InitStmt,
2883 LoopVarDecl, ColonLoc,
2892 Expr *Range = BEFFailure ? EndRangeRef.
get() : BeginRangeRef.
get();
2895 PDiag(diag::err_for_range_invalid)
2896 << RangeLoc << Range->getType()
2906 "invalid range expression in for loop");
2911 if (!
Context.hasSameType(BeginType, EndType)) {
2913 ? diag::warn_for_range_begin_end_types_differ
2914 : diag::ext_for_range_begin_end_types_differ)
2915 << BeginType << EndType;
2916 NoteForRangeBeginEndFunction(*
this, BeginExpr.
get(), BEF_begin);
2917 NoteForRangeBeginEndFunction(*
this, EndExpr.
get(), BEF_end);
2937 NotEqExpr =
ActOnBinOp(S, ColonLoc, tok::exclaimequal,
2938 BeginRef.
get(), EndRef.
get());
2945 Diag(RangeLoc, diag::note_for_range_invalid_iterator)
2946 << RangeLoc << diag::InvalidRangeForIterator::OpNotEq
2948 NoteForRangeBeginEndFunction(*
this, BeginExpr.
get(), BEF_begin);
2949 if (!
Context.hasSameType(BeginType, EndType))
2950 NoteForRangeBeginEndFunction(*
this, EndExpr.
get(), BEF_end);
2961 if (!IncrExpr.isInvalid() && CoawaitLoc.
isValid())
2966 if (!IncrExpr.isInvalid())
2968 if (IncrExpr.isInvalid()) {
2969 Diag(RangeLoc, diag::note_for_range_invalid_iterator)
2970 << RangeLoc << diag::InvalidRangeForIterator::OpAdvance
2972 NoteForRangeBeginEndFunction(*
this, BeginExpr.
get(), BEF_begin);
2984 Diag(RangeLoc, diag::note_for_range_invalid_iterator)
2985 << RangeLoc << diag::InvalidRangeForIterator::OpDeref
2987 NoteForRangeBeginEndFunction(*
this, BeginExpr.
get(), BEF_begin);
2997 NoteForRangeBeginEndFunction(*
this, BeginExpr.
get(), BEF_begin);
3014 for (
auto *MTE : LifetimeExtendTemps)
3019 InitStmt, RangeDS, cast_or_null<DeclStmt>(BeginDeclStmt.
get()),
3020 cast_or_null<DeclStmt>(EndDeclStmt.
get()), NotEqExpr.
get(),
3021 IncrExpr.get(), LoopVarDS,
nullptr, ForLoc, CoawaitLoc,
3022 ColonLoc, RParenLoc);
3041 if (
auto Cleanups = dyn_cast<ExprWithCleanups>(InitExpr))
3042 if (!Cleanups->cleanupsHaveSideEffects())
3043 InitExpr = Cleanups->getSubExpr();
3046 dyn_cast<MaterializeTemporaryExpr>(InitExpr);
3077 ReferenceReturnType = ReturnType;
3080 if (!ReferenceReturnType.
isNull()) {
3085 diag::warn_for_range_const_ref_binds_temp_built_from_ref)
3086 << VD << VariableType << ReferenceReturnType;
3099 SemaRef.
Diag(VD->
getLocation(), diag::warn_for_range_ref_binds_ret_temp)
3100 << VD << RangeInitType;
3113 return RD->hasAttr<TrivialABIAttr>();
3130 if (!CE->getConstructor()->isCopyConstructor())
3132 }
else if (
const CastExpr *CE = dyn_cast<CastExpr>(InitExpr)) {
3133 if (CE->getCastKind() != CK_LValueToRValue)
3151 << VD << VariableType;
3173 diag::warn_for_range_const_ref_binds_temp_built_from_ref, Loc) &&
3174 SemaRef.
Diags.
isIgnored(diag::warn_for_range_ref_binds_ret_temp, Loc) &&
3197 ForStmt->getRangeInit()->getType());
3214 diag::warn_empty_range_based_for_body);
3228 if (
getCurScope()->isInOpenACCComputeConstructScope())
3262 if (
getCurScope()->isInOpenACCComputeConstructScope())
3269 const Scope &DestScope,
3270 unsigned DeferJumpKind) {
3273 S.
Diag(Loc, diag::warn_jump_out_of_seh_finally);
3282 if (DestScope.
Contains(*Parent) || &DestScope == Parent)
3283 S.
Diag(Loc, diag::err_jump_out_of_defer_stmt) << DeferJumpKind;
3292 assert(
Target &&
"not a named break/continue?");
3302 S.
Diag(KWLoc, diag::err_acc_branch_in_out_compute_construct)
3315 if (IsContinue && !
Found->isContinueScope()) {
3316 S.
Diag(LabelLoc, diag::err_continue_switch);
3322 S.
Diag(LabelLoc, diag::err_break_continue_label_not_found) << IsContinue;
3341 return StmtError(
Diag(ContinueLoc, diag::err_continue_not_in_loop));
3349 Diag(ContinueLoc, diag::err_acc_branch_in_out_compute_construct)
3353 diag::DeferJumpKind::Continue);
3373 return StmtError(
Diag(BreakLoc, diag::err_break_not_in_loop_or_switch));
3377 return StmtError(
Diag(BreakLoc, diag::err_omp_loop_cannot_use_stmt)
3391 Diag(BreakLoc, diag::err_acc_branch_in_out_compute_construct)
3395 diag::DeferJumpKind::Break);
3406 const auto *DR = dyn_cast<DeclRefExpr>(E->
IgnoreParens());
3407 if (!DR || DR->refersToEnclosingVariableOrCapture())
3409 const auto *VD = dyn_cast<VarDecl>(DR->getDecl());
3432 if (VD->
getKind() == Decl::ParmVar)
3434 else if (VD->
getKind() != Decl::Var)
3447 if (VD->
hasAttr<BlocksAttr>())
3483 auto invalidNRVO = [&] {
3492 if ((ReturnType->
getTypeClass() == Type::TypeClass::Auto &&
3495 return invalidNRVO();
3501 return invalidNRVO();
3507 !
Context.hasSameUnqualifiedType(ReturnType, VDType))
3521 const auto *Step = llvm::find_if(
Seq.steps(), [](
const auto &Step) {
3522 return Step.Kind == InitializationSequence::SK_ConstructorInitialization ||
3523 Step.Kind == InitializationSequence::SK_UserConversion;
3525 if (Step !=
Seq.step_end()) {
3526 const auto *FD = Step->Function.Function;
3537 bool SupressSimplerImplicitMoves) {
3543 Expr *InitExpr = &AsRvalue;
3545 Value->getBeginLoc());
3547 auto Res =
Seq.getFailedOverloadResult();
3558 return Seq.Perform(*
this, Entity, Kind,
Value);
3578 bool SupressSimplerImplicitMoves) {
3586 bool HasDeducedReturnType =
3596 RetValExp = ER.
get();
3602 if (HasDeducedReturnType) {
3615 assert(AT &&
"lost auto type from lambda return type");
3631 RetValExp =
Result.get();
3646 Diag(ReturnLoc, diag::err_lambda_return_init_list)
3648 RetValExp =
nullptr;
3661 if (
auto *CurBlock = dyn_cast<BlockScopeInfo>(CurCap)) {
3663 Diag(ReturnLoc, diag::err_noreturn_has_return_expr)
3664 << diag::FalloffFunctionKind::Block;
3667 }
else if (
auto *CurRegion = dyn_cast<CapturedRegionScopeInfo>(CurCap)) {
3668 Diag(ReturnLoc, diag::err_return_in_captured_stmt) << CurRegion->getRegionName();
3671 assert(CurLambda &&
"unknown kind of captured scope");
3675 Diag(ReturnLoc, diag::err_noreturn_has_return_expr)
3676 << diag::FalloffFunctionKind::Lambda;
3688 if (isa_and_nonnull<InitListExpr>(RetValExp)) {
3689 Diag(ReturnLoc, diag::err_return_block_has_expr)
3690 << (CurLambda !=
nullptr);
3691 RetValExp =
nullptr;
3696 Diag(ReturnLoc, diag::ext_return_has_void_expr) <<
"literal" << 2;
3698 Diag(ReturnLoc, diag::err_return_block_has_expr)
3699 << (CurLambda !=
nullptr);
3700 RetValExp =
nullptr;
3703 }
else if (!RetValExp) {
3704 return StmtError(
Diag(ReturnLoc, diag::err_block_return_missing_expr));
3717 Entity, NRInfo, RetValExp, SupressSimplerImplicitMoves);
3722 RetValExp = Res.
get();
3723 CheckReturnValExpr(RetValExp, FnRetType, ReturnLoc);
3731 RetValExp = ER.
get();
3745 if (
auto *CurBlock = dyn_cast<BlockScopeInfo>(CurCap);
3748 CurBlock->TheDecl->setInvalidDecl();
3768 LocalTypedefNameReferencer(
Sema &S) : S(S) {}
3769 bool VisitRecordType(RecordType *RT)
override;
3774bool LocalTypedefNameReferencer::VisitRecordType(RecordType *RT) {
3775 auto *
R = dyn_cast<CXXRecordDecl>(RT->getDecl());
3776 if (!R || !
R->isLocalClass() || !
R->isLocalClass()->isExternallyVisible() ||
3777 R->isDependentType())
3779 for (
auto *TmpD :
R->decls())
3780 if (
auto *T = dyn_cast<TypedefNameDecl>(TmpD))
3781 if (T->getAccess() !=
AS_private ||
R->hasFriends())
3796 Expr *RetExpr,
const AutoType *AT) {
3803 if (isa_and_nonnull<InitListExpr>(RetExpr)) {
3808 : diag::err_auto_fn_return_init_list)
3818 assert(AT->isDeduced() &&
"should have deduced to dependent type");
3832 Diag(ReturnLoc, diag::err_auto_fn_return_void_but_not_auto)
3848 if (FindResult.Expression)
3849 TemplateSpecLoc = FindResult.Expression->getNameLoc();
3853 OrigResultType, RetExpr,
Deduced, Info,
false,
3869 Diag(ReturnLoc, diag::err_typecheck_missing_return_type_incompatible)
3872 Diag(ReturnLoc, diag::err_auto_fn_different_deductions)
3873 << (AT->isDecltypeAuto() ? 1 : 0) << Info.
SecondArg
3887 LocalTypedefNameReferencer(*this).TraverseType(RetExpr->
getType());
3904 diag::warn_qual_return_type,
3916 if (
getCurScope()->isInOpenACCComputeConstructScope())
3918 Diag(ReturnLoc, diag::err_acc_branch_in_out_compute_construct)
3925 "first coroutine location not set");
3926 Diag(ReturnLoc, diag::err_return_in_coroutine);
3931 CheckInvalidBuiltinCountedByRef(RetVal.
get(),
3936 if (R.isInvalid() ||
ExprEvalContexts.back().isDiscardedStatementContext())
3942 CurScope->updateNRVOCandidate(VD);
3945 diag::DeferJumpKind::Return);
3960 [[maybe_unused]]
Scope *CurScope) {
3983 bool AllowRecovery) {
3991 bool SupressSimplerImplicitMoves =
3999 SupressSimplerImplicitMoves);
4003 const AttrVec *Attrs =
nullptr;
4004 bool isObjCMethod =
false;
4011 Diag(ReturnLoc, diag::warn_noreturn_function_has_return_expr) << FD;
4012 if (FD->
isMain() && RetValExp)
4014 Diag(ReturnLoc, diag::warn_main_returns_bool_literal)
4016 if (FD->
hasAttr<CmseNSEntryAttr>() && RetValExp) {
4018 if (RT->getDecl()->isOrContainsUnion())
4023 FnRetType = MD->getReturnType();
4024 isObjCMethod =
true;
4026 Attrs = &MD->getAttrs();
4027 if (MD->hasRelatedResultType() && MD->getClassInterface()) {
4031 RelatedRetType =
Context.getObjCInterfaceType(MD->getClassInterface());
4032 RelatedRetType =
Context.getObjCObjectPointerType(RelatedRetType);
4038 const auto *ATy = dyn_cast<ArrayType>(RetValExp->
getType());
4039 if (ATy && ATy->getElementType().isWebAssemblyReferenceType()) {
4040 Diag(ReturnLoc, diag::err_wasm_table_art) << 1;
4054 RetValExp = ER.
get();
4080 AT->isDeduced() ? FnRetType :
QualType());
4081 if (Recovery.isInvalid())
4083 RetValExp = Recovery.get();
4099 if (
auto *ILE = dyn_cast<InitListExpr>(RetValExp)) {
4104 int FunctionKind = 0;
4112 Diag(ReturnLoc, diag::err_return_init_list)
4116 RetValExp = AllowRecovery
4118 ILE->getRBraceLoc(), ILE->inits())
4123 unsigned D = diag::ext_return_has_expr;
4128 D = diag::err_ctor_dtor_returns_void;
4130 D = diag::ext_return_has_void_expr;
4137 RetValExp =
Result.get();
4142 if (D == diag::err_ctor_dtor_returns_void) {
4144 Diag(ReturnLoc, D) << CurDecl << isa<CXXDestructorDecl>(CurDecl)
4148 else if (D != diag::ext_return_has_void_expr ||
4152 int FunctionKind = 0;
4170 RetValExp = ER.
get();
4176 }
else if (!RetValExp && !HasDependentReturnType) {
4183 Diag(ReturnLoc, diag::err_constexpr_return_missing_expr)
4189 unsigned DiagID =
getLangOpts().C99 ? diag::ext_return_missing_expr
4190 : diag::warn_return_missing_expr;
4194 "Not in a FunctionDecl or ObjCMethodDecl?");
4195 bool IsMethod = FD ==
nullptr;
4198 Diag(ReturnLoc, DiagID) << ND << IsMethod;
4204 assert(RetValExp || HasDependentReturnType);
4205 QualType RetType = RelatedRetType.
isNull() ? FnRetType : RelatedRetType;
4218 Entity, NRInfo, RetValExp, SupressSimplerImplicitMoves);
4221 RetValExp->
getEndLoc(), RetValExp, RetType);
4237 if (!RelatedRetType.
isNull()) {
4248 CheckReturnValExpr(RetValExp, FnRetType, ReturnLoc, isObjCMethod, Attrs,
4257 RetValExp = ER.
get();
4264 if (
Result->getNRVOCandidate())
4275 Stmt *HandlerBlock) {
4278 CXXCatchStmt(CatchLoc, cast_or_null<VarDecl>(ExDecl), HandlerBlock);
4282class CatchHandlerType {
4284 LLVM_PREFERRED_TYPE(
bool)
4285 unsigned IsPointer : 1;
4287 friend struct llvm::DenseMapInfo<CatchHandlerType>;
4305 CatchHandlerType(
QualType QT,
bool IsPointer)
4306 : QT(QT), IsPointer(IsPointer) {}
4308 QualType underlying()
const {
return QT; }
4309 bool isPointer()
const {
return IsPointer; }
4311 friend bool operator==(
const CatchHandlerType &LHS,
4312 const CatchHandlerType &RHS) {
4314 if (LHS.IsPointer != RHS.IsPointer)
4317 return LHS.QT == RHS.QT;
4323template <>
struct DenseMapInfo<CatchHandlerType> {
4325 return DenseMapInfo<QualType>::getHashValue(
Base.underlying());
4329 const CatchHandlerType &RHS) {
4336class CatchTypePublicBases {
4337 const llvm::DenseMap<QualType, CXXCatchStmt *> &TypesToCheck;
4339 CXXCatchStmt *FoundHandler;
4340 QualType FoundHandlerType;
4341 QualType TestAgainstType;
4344 CatchTypePublicBases(
const llvm::DenseMap<QualType, CXXCatchStmt *> &T,
4346 : TypesToCheck(T), FoundHandler(
nullptr), TestAgainstType(QT) {}
4348 CXXCatchStmt *getFoundHandler()
const {
return FoundHandler; }
4349 QualType getFoundHandlerType()
const {
return FoundHandlerType; }
4351 bool operator()(
const CXXBaseSpecifier *S, CXXBasePath &) {
4354 const auto &M = TypesToCheck;
4355 auto I = M.find(Check);
4367 if (I->second->getCaughtType()->isPointerType() ==
4369 FoundHandler = I->second;
4370 FoundHandlerType = Check;
4382 const llvm::Triple &T =
Context.getTargetInfo().getTriple();
4383 const bool IsOpenMPGPUTarget =
4390 if (IsOpenMPGPUTarget)
4391 targetDiag(TryLoc, diag::warn_try_not_valid_on_target) << T.str();
4399 Diag(TryLoc, diag::err_omp_simd_region_cannot_use_stmt) <<
"try";
4405 Diag(TryLoc, diag::err_mixing_cxx_try_seh_try) << 0;
4409 const unsigned NumHandlers = Handlers.size();
4410 assert(!Handlers.empty() &&
4411 "The parser shouldn't call this if there are no handlers.");
4413 llvm::DenseMap<QualType, CXXCatchStmt *> HandledBaseTypes;
4414 llvm::DenseMap<CatchHandlerType, CXXCatchStmt *> HandledTypes;
4415 for (
unsigned i = 0; i < NumHandlers; ++i) {
4422 if (i < NumHandlers - 1)
4436 QualType Underlying = HandlerCHT.underlying();
4438 if (!RD->hasDefinition())
4447 CatchTypePublicBases CTPB(HandledBaseTypes,
4449 if (RD->lookupInBases(CTPB, Paths)) {
4454 diag::warn_exception_caught_by_earlier_handler)
4457 diag::note_previous_exception_handler)
4469 auto R = HandledTypes.insert(
4474 diag::warn_exception_caught_by_earlier_handler)
4477 diag::note_previous_exception_handler)
4489 const llvm::Triple &T =
Context.getTargetInfo().getTriple();
4490 const bool IsOpenMPGPUTarget =
4502 targetDiag(Loc, diag::err_exceptions_disabled) << (IsTry ?
"try" :
"throw");
4507 assert(TryBlock && Handler);
4534 Diag(TryLoc, diag::err_seh_try_outside_functions);
4537 if (!
Context.getTargetInfo().isSEHTrySupported())
4538 Diag(TryLoc, diag::err_seh_try_unsupported);
4545 assert(FilterExpr &&
Block);
4549 Diag(FilterExpr->
getExprLoc(), diag::err_filter_expression_integral)
4571 Scope *SEHTryParent = CurScope;
4573 SEHTryParent = SEHTryParent->
getParent();
4575 return StmtError(
Diag(Loc, diag::err_ms___leave_not_in___try));
4577 diag::DeferJumpKind::SEHLeave);
4589 QualifierLoc, NameInfo,
4607 unsigned NumParams) {
4625 assert(NumParams > 0 &&
"CapturedStmt requires context parameter");
4669 CaptureInits.push_back(
Init.get());
4674static std::optional<int>
4691 unsigned NumParams) {
4693 Diag(Loc, diag::err_sme_openmp_captured_region) << *ErrorIndex;
4726 unsigned OpenMPCaptureLevel) {
4728 Diag(Loc, diag::err_sme_openmp_captured_region) << *ErrorIndex;
4735 bool ContextIsFound =
false;
4736 unsigned ParamNum = 0;
4739 I != E; ++I, ++ParamNum) {
4740 if (I->second.isNull()) {
4741 assert(!ContextIsFound &&
4742 "null type has been found already for '__context' parameter");
4753 ContextIsFound =
true;
4763 assert(ContextIsFound &&
"no null type for '__context' parameter");
4764 if (!ContextIsFound) {
4795 Record->setInvalidDecl();
4821 Captures, CaptureInits, CD, RD);
Defines the clang::ASTContext interface.
This file provides some common utility functions for processing Lambda related AST Constructs.
Defines the clang::Expr interface and subclasses for C++ expressions.
Result
Implement __builtin_bit_cast and related operations.
llvm::MachO::Target Target
llvm::MachO::Record Record
Defines the clang::Preprocessor interface.
static std::string toString(const clang::SanitizerSet &Sanitizers)
Produce a string containing comma-separated names of sanitizers in Sanitizers set.
This file declares semantic analysis for CUDA constructs.
This file declares semantic analysis for HLSL constructs.
This file declares semantic analysis for Objective-C.
This file declares semantic analysis for OpenMP constructs and clauses.
static bool CmpEnumVals(const std::pair< llvm::APSInt, EnumConstantDecl * > &lhs, const std::pair< llvm::APSInt, EnumConstantDecl * > &rhs)
CmpEnumVals - Comparison predicate for sorting enumeration values.
static bool FinishForRangeVarDecl(Sema &SemaRef, VarDecl *Decl, Expr *Init, SourceLocation Loc, int DiagID)
Finish building a variable declaration for a for-range statement.
static bool CmpCaseVals(const std::pair< llvm::APSInt, CaseStmt * > &lhs, const std::pair< llvm::APSInt, CaseStmt * > &rhs)
CmpCaseVals - Comparison predicate for sorting case values.
SmallVector< std::pair< llvm::APSInt, EnumConstantDecl * >, 64 > EnumValsTy
static bool ShouldDiagnoseSwitchCaseNotInEnum(const Sema &S, const EnumDecl *ED, const Expr *CaseExpr, EnumValsTy::iterator &EI, EnumValsTy::iterator &EIEnd, const llvm::APSInt &Val)
Returns true if we should emit a diagnostic about this case expression not being a part of the enum u...
static bool DiagnoseUnusedComparison(Sema &S, const Expr *E)
Diagnose unused comparisons, both builtin and overloaded operators.
static Scope * FindLabeledBreakContinueScope(Sema &S, Scope *CurScope, SourceLocation KWLoc, LabelDecl *Target, SourceLocation LabelLoc, bool IsContinue)
static bool EqEnumVals(const std::pair< llvm::APSInt, EnumConstantDecl * > &lhs, const std::pair< llvm::APSInt, EnumConstantDecl * > &rhs)
EqEnumVals - Comparison preficate for uniqing enumeration values.
static std::optional< int > isOpenMPCapturedRegionInArmSMEFunction(Sema const &S, CapturedRegionKind Kind)
static bool hasDeducedReturnType(FunctionDecl *FD)
Determine whether the declared return type of the specified function contains 'auto'.
static bool ObjCEnumerationCollection(Expr *Collection)
static void DiagnoseForRangeConstVariableCopies(Sema &SemaRef, const VarDecl *VD)
static StmtResult RebuildForRangeWithDereference(Sema &SemaRef, Scope *S, SourceLocation ForLoc, SourceLocation CoawaitLoc, Stmt *InitStmt, Stmt *LoopVarDecl, SourceLocation ColonLoc, Expr *Range, SourceLocation RangeLoc, SourceLocation RParenLoc)
Speculatively attempt to dereference an invalid range expression.
static void checkEnumTypesInSwitchStmt(Sema &S, const Expr *Cond, const Expr *Case)
static void DiagnoseForRangeReferenceVariableCopies(Sema &SemaRef, const VarDecl *VD, QualType RangeInitType)
static void DiagnoseForRangeVariableCopies(Sema &SemaRef, const CXXForRangeStmt *ForStmt)
DiagnoseForRangeVariableCopies - Diagnose three cases and fixes for them.
static bool CheckSimplerImplicitMovesMSVCWorkaround(const Sema &S, const Expr *E)
static bool VerifyInitializationSequenceCXX98(const Sema &S, const InitializationSequence &Seq)
Verify that the initialization sequence that was picked for the first overload resolution is permissi...
static QualType GetTypeBeforeIntegralPromotion(const Expr *&E)
GetTypeBeforeIntegralPromotion - Returns the pre-promotion type of potentially integral-promoted expr...
static Sema::ForRangeStatus BuildNonArrayForRange(Sema &SemaRef, Expr *BeginRange, Expr *EndRange, QualType RangeType, VarDecl *BeginVar, VarDecl *EndVar, SourceLocation ColonLoc, SourceLocation CoawaitLoc, OverloadCandidateSet *CandidateSet, ExprResult *BeginExpr, ExprResult *EndExpr, BeginEndFunction *BEF)
Create the initialization, compare, and increment steps for the range-based for loop expression.
static bool hasTrivialABIAttr(QualType VariableType)
Determines whether the VariableType's declaration is a record with the clang::trivial_abi attribute.
static void CheckJumpOutOfSEHFinallyOrDefer(Sema &S, SourceLocation Loc, const Scope &DestScope, unsigned DeferJumpKind)
static void AdjustAPSInt(llvm::APSInt &Val, unsigned BitWidth, bool IsSigned)
static bool buildCapturedStmtCaptureList(Sema &S, CapturedRegionScopeInfo *RSI, SmallVectorImpl< CapturedStmt::Capture > &Captures, SmallVectorImpl< Expr * > &CaptureInits)
static bool DiagnoseNoDiscard(Sema &S, const NamedDecl *OffendingDecl, const WarnUnusedResultAttr *A, SourceLocation Loc, SourceRange R1, SourceRange R2, bool IsCtor)
static void checkCaseValue(Sema &S, SourceLocation Loc, const llvm::APSInt &Val, unsigned UnpromotedWidth, bool UnpromotedSign)
Check the specified case value is in range for the given unpromoted switch type.
Defines the Objective-C statement AST node classes.
Defines the clang::TypeLoc interface and its subclasses.
Allows QualTypes to be sorted and hence used in maps and sets.
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
QualType getLValueReferenceType(QualType T, bool SpelledAsLValue=true) const
Return the uniqued reference to the type for an lvalue reference to the specified type.
TypeSourceInfo * getTrivialTypeSourceInfo(QualType T, SourceLocation Loc=SourceLocation()) const
Allocate a TypeSourceInfo where all locations have been initialized to a given location,...
uint64_t getTypeSize(QualType T) const
Return the size of the specified (complete) type T, in bits.
CanQualType getCanonicalTagType(const TagDecl *TD) const
static bool hasSameUnqualifiedType(QualType T1, QualType T2)
Determine whether the given types are equivalent after cvr-qualifiers have been removed.
Represents an array type, per C99 6.7.5.2 - Array Declarators.
Attr - This represents one attribute.
SourceLocation getLocation() const
SourceRange getRange() const
static AttributedStmt * Create(const ASTContext &C, SourceLocation Loc, ArrayRef< const Attr * > Attrs, Stmt *SubStmt)
Expr * getFalseExpr() const
getFalseExpr - Return the subexpression which will be evaluated if the condition evaluates to false; ...
OpaqueValueExpr * getOpaqueValue() const
getOpaqueValue - Return the opaque value placeholder.
A builtin binary operation expression such as "x + y" or "x <= y".
SourceLocation getExprLoc() const
BreakStmt - This represents a break.
CStyleCastExpr - An explicit cast in C (C99 6.5.4) or a C-style cast in C++ (C++ [expr....
BasePaths - Represents the set of paths from a derived class to one of its (direct or indirect) bases...
void setOrigin(const CXXRecordDecl *Rec)
bool isAmbiguous(CanQualType BaseType) const
Determine whether the path from the most-derived type to the given base type is ambiguous (i....
QualType getType() const
Retrieves the type of the base class.
AccessSpecifier getAccessSpecifier() const
Returns the access specifier for this base specifier.
Represents binding an expression to a temporary.
CXXCatchStmt - This represents a C++ catch block.
SourceLocation getBeginLoc() const LLVM_READONLY
VarDecl * getExceptionDecl() const
QualType getCaughtType() const
Represents a call to a C++ constructor.
Represents a C++ conversion function within a class.
CXXForRangeStmt - This represents C++0x [stmt.ranged]'s ranged for statement, represented as 'for (ra...
DeclStmt * getBeginStmt()
DeclStmt * getRangeStmt()
Represents an explicit C++ type conversion that uses "functional" notation (C++ [expr....
Represents a call to a member function that may be written either with member call syntax (e....
A call to an overloaded operator written using operator syntax.
Represents a C++ struct/union/class.
static CXXRecordDecl * Create(const ASTContext &C, TagKind TK, DeclContext *DC, SourceLocation StartLoc, SourceLocation IdLoc, IdentifierInfo *Id, CXXRecordDecl *PrevDecl=nullptr)
An expression "T()" which creates an rvalue of a non-class type T.
Represents a C++ nested-name-specifier or a global scope specifier.
NestedNameSpecifierLoc getWithLocInContext(ASTContext &Context) const
Retrieve a nested-name-specifier with location information, copied into the given AST context.
static CXXTryStmt * Create(const ASTContext &C, SourceLocation tryLoc, CompoundStmt *tryBlock, ArrayRef< Stmt * > handlers)
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
static CanQual< Type > CreateUnsafe(QualType Other)
Represents the body of a CapturedStmt, and serves as its DeclContext.
static DeclContext * castToDeclContext(const CapturedDecl *D)
void setContextParam(unsigned i, ImplicitParamDecl *P)
void setParam(unsigned i, ImplicitParamDecl *P)
static CapturedDecl * Create(ASTContext &C, DeclContext *DC, unsigned NumParams)
Describes the capture of either a variable, or 'this', or variable-length array type.
This captures a statement into a function.
Stmt * getCapturedStmt()
Retrieve the statement being captured.
static CapturedStmt * Create(const ASTContext &Context, Stmt *S, CapturedRegionKind Kind, ArrayRef< Capture > Captures, ArrayRef< Expr * > CaptureInits, CapturedDecl *CD, RecordDecl *RD)
CaseStmt - Represent a case statement.
static CaseStmt * Create(const ASTContext &Ctx, Expr *lhs, Expr *rhs, SourceLocation caseLoc, SourceLocation ellipsisLoc, SourceLocation colonLoc)
Build a case statement.
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
CastKind getCastKind() const
CompoundStmt - This represents a group of statements like { stmt stmt }.
static CompoundStmt * Create(const ASTContext &C, ArrayRef< Stmt * > Stmts, FPOptionsOverride FPFeatures, SourceLocation LB, SourceLocation RB)
Expr * getFalseExpr() const
getFalseExpr - Return the subexpression representing the value of the expression if the condition eva...
Expr * getCond() const
getCond - Return the expression representing the condition for the ?
Expr * getTrueExpr() const
getTrueExpr - Return the subexpression representing the value of the expression if the condition eval...
ConstEvaluatedExprVisitor - This class visits 'const Expr *'s.
Represents the canonical version of C arrays with a specified constant size.
ContinueStmt - This represents a continue.
DeclContext - This is used only as base class of specific decl types that can act as declaration cont...
DeclContext * getParent()
getParent - Returns the containing DeclContext.
bool isFileContext() const
bool isDependentContext() const
Determines whether this context is dependent on a template parameter.
void addDecl(Decl *D)
Add the declaration D into this context.
bool isStdNamespace() const
bool isFunctionOrMethod() const
void addHiddenDecl(Decl *D)
Add the declaration D to this context without modifying any lookup tables.
bool isSingleDecl() const
A reference to a declared variable, function, enum, etc.
SourceLocation getLocation() const
DeclStmt - Adaptor class for mixing declarations with statements and expressions.
bool isSingleDecl() const
isSingleDecl - This method returns true if this DeclStmt refers to a single Decl.
const Decl * getSingleDecl() const
SourceLocation getBeginLoc() const LLVM_READONLY
Decl - This represents one declaration (or definition), e.g.
void setInvalidDecl(bool Invalid=true)
setInvalidDecl - Indicates the Decl had a semantic error.
void markUsed(ASTContext &C)
Mark the declaration used, in the sense of odr-use.
static Decl * castFromDeclContext(const DeclContext *)
bool isInvalidDecl() const
SourceLocation getLocation() const
void setImplicit(bool I=true)
void setLocation(SourceLocation L)
DeclContext * getDeclContext()
SourceLocation getTypeSpecEndLoc() const
SourceLocation getTypeSpecStartLoc() const
SourceLocation getBeginLoc() const LLVM_READONLY
TypeSourceInfo * getTypeSourceInfo() const
SourceLocation getDefaultLoc() const
static DeferStmt * Create(ASTContext &Context, SourceLocation DeferLoc, Stmt *Body)
bool isIgnored(unsigned DiagID, SourceLocation Loc) const
Determine whether the diagnostic is known to be ignored.
DoStmt - This represents a 'do/while' stmt.
EnumDecl * getCanonicalDecl() override
Retrieves the "canonical" declaration of the given declaration.
bool isClosed() const
Returns true if this enum is either annotated with enum_extensibility(closed) or isn't annotated with...
EvaluatedExprVisitor - This class visits 'Expr *'s.
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.
Expr * IgnoreParenCasts() LLVM_READONLY
Skip past any parentheses and casts which might surround this expression until reaching a fixed point...
bool isUnusedResultAWarning(const Expr *&WarnExpr, SourceLocation &Loc, SourceRange &R1, SourceRange &R2, ASTContext &Ctx) const
isUnusedResultAWarning - Return true if this immediate expression should be warned about if the resul...
bool isValueDependent() const
Determines whether the value of this expression depends on.
bool isTypeDependent() const
Determines whether the type of this expression depends on.
llvm::APSInt EvaluateKnownConstInt(const ASTContext &Ctx) const
EvaluateKnownConstInt - Call EvaluateAsRValue and return the folded integer.
Expr * IgnoreParenImpCasts() LLVM_READONLY
Skip past any parentheses and implicit casts which might surround this expression until reaching a fi...
bool containsErrors() const
Whether this expression contains subexpressions which had errors.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
std::optional< llvm::APSInt > getIntegerConstantExpr(const ASTContext &Ctx) const
isIntegerConstantExpr - Return the value if this expression is a valid integer constant expression.
Decl * getReferencedDeclOfCallee()
bool isInstantiationDependent() const
Whether this expression is instantiation-dependent, meaning that it depends in some way on.
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 isKnownToHaveBooleanValue(bool Semantic=true) const
isKnownToHaveBooleanValue - Return true if this is an integer expression that is known to return 0 or...
Represents difference between two FPOptions values.
FPOptionsOverride getChangesFrom(const FPOptions &Base) const
Return difference with the given option set.
Represents a member of a struct/union/class.
static FixItHint CreateReplacement(CharSourceRange RemoveRange, StringRef Code)
Create a code modification hint that replaces the given source range with the given code string.
static FixItHint CreateRemoval(CharSourceRange RemoveRange)
Create a code modification hint that removes the given source range.
static FixItHint CreateInsertion(SourceLocation InsertionLoc, StringRef Code, bool BeforePreviousInsertions=false)
Create a code modification hint that inserts the given code string at a specific location.
ForStmt - This represents a 'for (init;cond;inc)' stmt.
SourceLocation getRParenLoc() const
SourceLocation getBeginLoc() const
FullExpr - Represents a "full-expression" node.
Represents a function declaration or definition.
const ParmVarDecl * getParamDecl(unsigned i) const
bool isFunctionTemplateSpecialization() const
Determine whether this function is a function template specialization.
void setUsesSEHTry(bool UST)
bool isNoReturn() const
Determines whether this function is known to be 'noreturn', through an attribute on its declaration o...
QualType getReturnType() const
FunctionTemplateDecl * getPrimaryTemplate() const
Retrieve the primary template that this function template specialization either specializes or was in...
const TemplateArgumentList * getTemplateSpecializationArgs() const
Retrieve the template arguments used to produce this function template specialization from the primar...
bool isConstexpr() const
Whether this is a (C++11) constexpr function or constexpr constructor.
bool isMain() const
Determines whether this function is "main", which is the entry point into an executable program.
SourceRange getSourceRange() const override LLVM_READONLY
Source range that this declaration covers.
bool isOverloadedOperator() const
Whether this function declaration represents an C++ overloaded operator, e.g., "operator+".
OverloadedOperatorKind getOverloadedOperator() const
getOverloadedOperator - Which C++ overloaded operator this function represents, if any.
QualType getDeclaredReturnType() const
Get the declared return type, which may differ from the actual return type if the return type is dedu...
Represents a prototype with parameter type info, e.g.
Declaration of a template function.
FunctionType - C99 6.7.5.3 - Function Declarators.
static StringRef getNameForCallConv(CallingConv CC)
bool getNoReturnAttr() const
Determine whether this function type includes the GNU noreturn attribute.
QualType getReturnType() const
GotoStmt - This represents a direct goto.
One of these records is kept for each identifier that is lexed.
IdentifierInfo & get(StringRef Name)
Return the identifier token info for the specified named identifier.
static IfStmt * Create(const ASTContext &Ctx, SourceLocation IL, IfStatementKind Kind, Stmt *Init, VarDecl *Var, Expr *Cond, SourceLocation LPL, SourceLocation RPL, Stmt *Then, SourceLocation EL=SourceLocation(), Stmt *Else=nullptr)
Create an IfStmt.
ImplicitCastExpr - Allows us to explicitly represent implicit type conversions, which have no direct ...
static ImplicitCastExpr * Create(const ASTContext &Context, QualType T, CastKind Kind, Expr *Operand, const CXXCastPath *BasePath, ExprValueKind Cat, FPOptionsOverride FPO)
static ImplicitParamDecl * Create(ASTContext &C, DeclContext *DC, SourceLocation IdLoc, const IdentifierInfo *Id, QualType T, ImplicitParamKind ParamKind)
Create implicit parameter.
IndirectGotoStmt - This represents an indirect goto.
static InitializationKind CreateCopy(SourceLocation InitLoc, SourceLocation EqualLoc, bool AllowExplicitConvs=false)
Create a copy initialization.
Describes the sequence of initializations required to initialize a given object or reference with a s...
Describes an entity that is being initialized.
static InitializedEntity InitializeResult(SourceLocation ReturnLoc, QualType Type)
Create the initialization entity for the result of a function.
static InitializedEntity InitializeRelatedResult(ObjCMethodDecl *MD, QualType Type)
Create the initialization entity for a related result.
unsigned allocateManglingNumber() const
static InitializedEntity InitializeVariable(VarDecl *Var)
Create the initialization entity for a variable.
static InitializedEntity InitializeParameter(ASTContext &Context, ParmVarDecl *Parm)
Create the initialization entity for a parameter.
static IntegerLiteral * Create(const ASTContext &C, const llvm::APInt &V, QualType type, SourceLocation l)
Returns a new integer literal with value 'V' and type 'type'.
Represents the declaration of a label.
void setLocStart(SourceLocation L)
LabelStmt * getStmt() const
void setStmt(LabelStmt *T)
bool isMSAsmLabel() const
LabelStmt - Represents a label, which has a substatement.
Represents the results of name lookup.
bool empty() const
Return true if no decls were found.
SourceLocation getKwLoc() const
Representation of a Microsoft __if_exists or __if_not_exists statement with a dependent name.
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
Expr * getSubExpr() const
Retrieve the temporary-generating subexpression whose value will be materialized into a glvalue.
MemberExpr - [C99 6.5.2.3] Structure and Union Members.
This represents a decl that may have a name.
DeclarationName getDeclName() const
Get the actual, stored name of the declaration, which may be a special name.
ReservedIdentifierStatus isReserved(const LangOptions &LangOpts) const
Determine if the declaration obeys the reserved identifier rules of the given language.
A C++ nested-name-specifier augmented with source location information.
NullStmt - This is the null statement ";": C99 6.8.3p3.
An expression that sends a message to the given Objective-C object or class.
ObjCMethodDecl - Represents an instance or class method declaration.
Represents a pointer to an Objective C object.
Expr * getSourceExpr() const
The source expression of an opaque value expression is the expression which originally generated the ...
OverloadCandidateSet - A set of overload candidates, used in C++ overload resolution (C++ 13....
@ CSK_Normal
Normal lookup.
void NoteCandidates(PartialDiagnosticAt PA, Sema &S, OverloadCandidateDisplayKind OCD, ArrayRef< Expr * > Args, StringRef Opc="", SourceLocation Loc=SourceLocation(), llvm::function_ref< bool(OverloadCandidate &)> Filter=[](OverloadCandidate &) { return true;})
When overload resolution fails, prints diagnostic messages containing the candidates in the candidate...
static FindResult find(Expr *E)
Finds the overloaded expression in the given expression E of OverloadTy.
const Expr * getSubExpr() const
Represents a parameter to a function.
ParsedAttributes - A collection of parsed attributes.
Wrapper for source info for pointers.
SourceLocation getStarLoc() const
IdentifierTable & getIdentifierTable()
PseudoObjectExpr - An expression which accesses a pseudo-object l-value.
ArrayRef< Expr * > semantics()
A (possibly-)qualified type.
bool isVolatileQualified() const
Determine whether this type is volatile-qualified.
QualType withConst() const
bool isTriviallyCopyConstructibleType(const ASTContext &Context) const
Return true if this is a trivially copyable type.
bool isNull() const
Return true if this QualType doesn't point to a type yet.
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
QualType getUnqualifiedType() const
Retrieve the unqualified variant of the given type, removing as little sugar as possible.
unsigned getLocalCVRQualifiers() const
Retrieve the set of CVR (const-volatile-restrict) qualifiers local to this particular QualType instan...
bool isConstQualified() const
Determine whether this type is const-qualified.
@ OCL_Strong
Assigning into this object requires the old value to be released and the new value to be retained.
Represents a struct/union/class.
static RecordDecl * Create(const ASTContext &C, TagKind TK, DeclContext *DC, SourceLocation StartLoc, SourceLocation IdLoc, IdentifierInfo *Id, RecordDecl *PrevDecl=nullptr)
virtual void completeDefinition()
Note that the definition of this type is now complete.
void setCapturedRecord()
Mark the record as a record for captured variables in CapturedStmt construct.
ReturnStmt - This represents a return, optionally of an expression: return; return 4;.
static ReturnStmt * Create(const ASTContext &Ctx, SourceLocation RL, Expr *E, const VarDecl *NRVOCandidate)
Create a return statement.
static SEHExceptStmt * Create(const ASTContext &C, SourceLocation ExceptLoc, Expr *FilterExpr, Stmt *Block)
static SEHFinallyStmt * Create(const ASTContext &C, SourceLocation FinallyLoc, Stmt *Block)
Represents a __leave statement.
static SEHTryStmt * Create(const ASTContext &C, bool isCXXTry, SourceLocation TryLoc, Stmt *TryBlock, Stmt *Handler)
Scope - A scope is a transient data structure that is used while parsing the program.
unsigned getDepth() const
Returns the depth of this scope. The translation-unit has scope depth 0.
bool Contains(const Scope &rhs) const
Returns if rhs has a higher scope depth than this.
LabelDecl * getPrecedingLabel() const
Get the label that precedes this scope.
Scope * getContinueParent()
getContinueParent - Return the closest scope that a continue statement would be affected by.
bool isLoopScope() const
Return true if this scope is a loop.
bool isSEHTryScope() const
Determine whether this scope is a SEH '__try' block.
Scope * getBreakParent()
getBreakParent - Return the closest scope that a break statement would be affected by.
const Scope * getParent() const
getParent - Return the scope that this is nested in.
bool isBreakOrContinueScope() const
Determine whether this is a scope which can have 'break' or 'continue' statements embedded into it.
bool isFunctionScope() const
isFunctionScope() - Return true if this scope is a function scope.
bool isOpenACCComputeConstructScope() const
Determine whether this scope is the statement associated with an OpenACC Compute construct directive.
bool isOpenMPLoopScope() const
Determine whether this scope is a loop having OpenMP loop directive attached.
@ SwitchScope
This is a scope that corresponds to a switch statement.
A generic diagnostic builder for errors which may or may not be deferred.
PartialDiagnostic PDiag(unsigned DiagID=0)
Build a partial diagnostic.
SemaDiagnosticBuilder Diag(SourceLocation Loc, unsigned DiagID)
Emit a diagnostic.
CUDAFunctionTarget CurrentTarget()
Gets the CUDA target for the current context.
SemaDiagnosticBuilder DiagIfDeviceCode(SourceLocation Loc, unsigned DiagID)
Creates a SemaDiagnosticBuilder that emits the diagnostic if the current context is "used as device c...
void propagateContextualMatrixLayout(Expr *E, QualType DestType)
StmtResult ActOnObjCForCollectionStmt(SourceLocation ForColLoc, Stmt *First, Expr *collection, SourceLocation RParenLoc)
StmtResult FinishObjCForCollectionStmt(Stmt *ForCollection, Stmt *Body)
FinishObjCForCollectionStmt - Attach the body to a objective-C foreach statement.
bool inferObjCARCLifetime(ValueDecl *decl)
void ActOnOpenMPLoopInitialization(SourceLocation ForLoc, Stmt *Init)
Check if the current region is an OpenMP loop region and if it is, mark loop control variable,...
void setOpenMPCaptureKind(FieldDecl *FD, const ValueDecl *D, unsigned Level)
Sets OpenMP capture kind (OMPC_private, OMPC_firstprivate, OMPC_map etc.) for FD based on DSA for the...
std::pair< VarDecl *, Expr * > get() const
RAII class used to determine whether SFINAE has trapped any errors that occur during template argumen...
Sema - This implements semantic analysis and AST building for C.
ParsedType CreateParsedType(QualType T, TypeSourceInfo *TInfo)
Package the given type and TSI into a ParsedType.
SmallVector< Scope *, 2 > CurrentSEHFinally
Stack of active SEH __finally scopes. Can be empty.
LocalInstantiationScope * CurrentInstantiationScope
The current instantiation scope used to store local variables.
Scope * getCurScope() const
Retrieve the parser's current scope.
StmtResult ActOnBreakStmt(SourceLocation BreakLoc, Scope *CurScope, LabelDecl *Label, SourceLocation LabelLoc)
void ProcessStmtAttributes(Stmt *Stmt, const ParsedAttributes &InAttrs, SmallVectorImpl< const Attr * > &OutAttrs)
Process the attributes before creating an attributed statement.
ExprResult ActOnUnaryOp(Scope *S, SourceLocation OpLoc, tok::TokenKind Op, Expr *Input, bool IsAfterAmp=false)
Unary Operators. 'Tok' is the token for the operator.
StmtResult BuildMSDependentExistsStmt(SourceLocation KeywordLoc, bool IsIfExists, NestedNameSpecifierLoc QualifierLoc, DeclarationNameInfo NameInfo, Stmt *Nested)
ExprResult IgnoredValueConversions(Expr *E)
IgnoredValueConversions - Given that an expression's result is syntactically ignored,...
@ LookupMemberName
Member name lookup, which finds the names of class/struct/union members.
bool ActOnCoroutineBodyStart(Scope *S, SourceLocation KwLoc, StringRef Keyword)
StmtResult BuildAttributedStmt(SourceLocation AttrsLoc, ArrayRef< const Attr * > Attrs, Stmt *SubStmt)
StmtResult ActOnSEHLeaveStmt(SourceLocation Loc, Scope *CurScope)
llvm::DenseMap< const EnumDecl *, llvm::SmallVector< llvm::APSInt > > AssignEnumCache
A cache of enumerator values for enums checked by -Wassign-enum.
StmtResult ActOnForEachLValueExpr(Expr *E)
In an Objective C collection iteration statement: for (x in y) x can be an arbitrary l-value expressi...
void ActOnForEachDeclStmt(DeclGroupPtrTy Decl)
void PushExpressionEvaluationContext(ExpressionEvaluationContext NewContext, Decl *LambdaContextDecl=nullptr, ExpressionEvaluationContextRecord::ExpressionKind Type=ExpressionEvaluationContextRecord::EK_Other)
ExprResult CheckBooleanCondition(SourceLocation Loc, Expr *E, bool IsConstexpr=false)
CheckBooleanCondition - Diagnose problems involving the use of the given expression as a boolean cond...
@ Switch
An integral condition for a 'switch' statement.
SmallVector< sema::FunctionScopeInfo *, 4 > FunctionScopes
Stack containing information about each of the nested function, block, and method scopes that are cur...
bool checkAndRewriteMustTailAttr(Stmt *St, const Attr &MTA)
Check whether the given statement can have musttail applied to it, issuing a diagnostic and returning...
StmtResult ActOnContinueStmt(SourceLocation ContinueLoc, Scope *CurScope, LabelDecl *Label, SourceLocation LabelLoc)
StmtResult ActOnGotoStmt(SourceLocation GotoLoc, SourceLocation LabelLoc, LabelDecl *TheDecl)
StmtResult ActOnReturnStmt(SourceLocation ReturnLoc, Expr *RetValExp, Scope *CurScope)
void setFunctionHasBranchIntoScope()
ExprResult ActOnCaseExpr(SourceLocation CaseLoc, ExprResult Val)
StmtResult ActOnExprStmt(ExprResult Arg, bool DiscardedValue=true)
FieldDecl * BuildCaptureField(RecordDecl *RD, const sema::Capture &Capture)
Build a FieldDecl suitable to hold the given capture.
AssignConvertType CheckSingleAssignmentConstraints(QualType LHSType, ExprResult &RHS, bool Diagnose=true, bool DiagnoseCFAudited=false, bool ConvertRHS=true)
Check assignment constraints for an assignment of RHS to LHSType.
StmtResult BuildIfStmt(SourceLocation IfLoc, IfStatementKind StatementKind, SourceLocation LParenLoc, Stmt *InitStmt, ConditionResult Cond, SourceLocation RParenLoc, Stmt *ThenVal, SourceLocation ElseLoc, Stmt *ElseVal)
FunctionDecl * getCurFunctionDecl(bool AllowLambda=false) const
Returns a pointer to the innermost enclosing function, or nullptr if the current context is not insid...
ExprResult PerformContextualImplicitConversion(SourceLocation Loc, Expr *FromE, ContextualImplicitConverter &Converter)
Perform a contextual implicit conversion.
ExprResult UsualUnaryConversions(Expr *E)
UsualUnaryConversions - Performs various conversions that are common to most operators (C99 6....
void DiagnoseCommaOperator(const Expr *LHS, SourceLocation Loc)
Look for instances where it is likely the comma operator is confused with another operator.
void DiagnoseExceptionUse(SourceLocation Loc, bool IsTry)
ExprResult CheckSwitchCondition(SourceLocation SwitchLoc, Expr *Cond)
bool DiagIfReachable(SourceLocation Loc, ArrayRef< const Stmt * > Stmts, const PartialDiagnostic &PD)
Conditionally issue a diagnostic based on the statements's reachability analysis.
void FinalizeDeclaration(Decl *D)
FinalizeDeclaration - called by ParseDeclarationAfterDeclarator to perform any semantic actions neces...
void ActOnCapturedRegionError()
DeclGroupPtrTy ConvertDeclToDeclGroup(Decl *Ptr, Decl *OwnedType=nullptr)
ExprResult DefaultFunctionArrayLvalueConversion(Expr *E, bool Diagnose=true)
ASTContext & getASTContext() const
std::unique_ptr< sema::FunctionScopeInfo, PoppedFunctionScopeDeleter > PoppedFunctionScopePtr
ExprResult BuildUnaryOp(Scope *S, SourceLocation OpLoc, UnaryOperatorKind Opc, Expr *Input, bool IsAfterAmp=false)
void PopExpressionEvaluationContext()
StmtResult ActOnSEHTryBlock(bool IsCXXTry, SourceLocation TryLoc, Stmt *TryBlock, Stmt *Handler)
ExprResult ImpCastExprToType(Expr *E, QualType Type, CastKind CK, ExprValueKind VK=VK_PRValue, const CXXCastPath *BasePath=nullptr, CheckedConversionKind CCK=CheckedConversionKind::Implicit)
ImpCastExprToType - If Expr is not of type 'Type', insert an implicit cast.
ObjCMethodDecl * getCurMethodDecl()
getCurMethodDecl - If inside of a method body, this returns a pointer to the method decl for the meth...
DeclRefExpr * BuildDeclRefExpr(ValueDecl *D, QualType Ty, ExprValueKind VK, SourceLocation Loc, const CXXScopeSpec *SS=nullptr)
ExprResult CheckConvertedConstantExpression(Expr *From, QualType T, llvm::APSInt &Value, CCEKind CCE)
void setFunctionHasIndirectGoto()
ExprResult BuildCaptureInit(const sema::Capture &Capture, SourceLocation ImplicitCaptureLoc, bool IsOpenMPMapping=false)
Initialize the given capture with a suitable expression.
StmtResult ActOnFinishSwitchStmt(SourceLocation SwitchLoc, Stmt *Switch, Stmt *Body)
NamedReturnInfo getNamedReturnInfo(Expr *&E, SimplerImplicitMoveMode Mode=SimplerImplicitMoveMode::Normal)
Determine whether the given expression might be move-eligible or copy-elidable in either a (co_)retur...
void DiagnoseUnusedExprResult(const Stmt *S, unsigned DiagID)
DiagnoseUnusedExprResult - If the statement passed in is an expression whose result is unused,...
FPOptions & getCurFPFeatures()
Sema(Preprocessor &pp, ASTContext &ctxt, ASTConsumer &consumer, TranslationUnitKind TUKind=TU_Complete, CodeCompleteConsumer *CompletionConsumer=nullptr)
@ UPPC_Expression
An arbitrary expression.
const LangOptions & getLangOpts() const
PoppedFunctionScopePtr PopFunctionScopeInfo(const sema::AnalysisBasedWarnings::Policy *WP=nullptr, Decl *D=nullptr, QualType BlockType=QualType())
Pop a function (or block or lambda or captured region) scope from the stack.
StmtResult ActOnWhileStmt(SourceLocation WhileLoc, SourceLocation LParenLoc, ConditionResult Cond, SourceLocation RParenLoc, Stmt *Body)
bool DiagnoseUnexpandedParameterPack(SourceLocation Loc, TypeSourceInfo *T, UnexpandedParameterPackContext UPPC)
If the given type contains an unexpanded parameter pack, diagnose the error.
const LangOptions & LangOpts
sema::LambdaScopeInfo * getCurLambda(bool IgnoreNonLambdaCapturingScope=false)
Retrieve the current lambda scope info, if any.
void ActOnStartOfCompoundStmt(bool IsStmtExpr)
bool DeduceFunctionTypeFromReturnExpr(FunctionDecl *FD, SourceLocation ReturnLoc, Expr *RetExpr, const AutoType *AT)
Deduce the return type for a function from a returned expression, per C++1y [dcl.spec....
void MarkAnyDeclReferenced(SourceLocation Loc, Decl *D, bool MightBeOdrUse)
Perform marking for a reference to an arbitrary declaration.
TypeLoc getReturnTypeLoc(FunctionDecl *FD) const
StmtResult ActOnExprStmtError()
const VarDecl * getCopyElisionCandidate(NamedReturnInfo &Info, QualType ReturnType)
Updates given NamedReturnInfo's move-eligible and copy-elidable statuses, considering the function re...
NamedDecl * getCurFunctionOrMethodDecl() const
getCurFunctionOrMethodDecl - Return the Decl for the current ObjC method or C function we're in,...
StmtResult ActOnNullStmt(SourceLocation SemiLoc, bool HasLeadingEmptyMacro=false)
RecordDecl * CreateCapturedStmtRecordDecl(CapturedDecl *&CD, SourceLocation Loc, unsigned NumParams)
void ActOnCapturedRegionStart(SourceLocation Loc, Scope *CurScope, CapturedRegionKind Kind, unsigned NumParams)
sema::FunctionScopeInfo * getCurFunction() const
void PushCompoundScope(bool IsStmtExpr)
DeclGroupPtrTy BuildDeclaratorGroup(MutableArrayRef< Decl * > Group)
BuildDeclaratorGroup - convert a list of declarations into a declaration group, performing any necess...
void DiagnoseAssignmentEnum(QualType DstType, QualType SrcType, Expr *SrcExpr)
DiagnoseAssignmentEnum - Warn if assignment to enum is a constant integer not in the range of enum va...
StmtResult ActOnEndOfDeferStmt(Stmt *Body, Scope *CurScope)
ExprResult ActOnCoawaitExpr(Scope *S, SourceLocation KwLoc, Expr *E)
bool findMacroSpelling(SourceLocation &loc, StringRef name)
Looks through the macro-expansion chain for the given location, looking for a macro expansion with th...
void DiagnoseEmptyStmtBody(SourceLocation StmtLoc, const Stmt *Body, unsigned DiagID)
Emit DiagID if statement located on StmtLoc has a suspicious null statement as a Body,...
void DiagnoseEmptyLoopBody(const Stmt *S, const Stmt *PossibleBody)
Warn if a for/while loop statement S, which is followed by PossibleBody, has a suspicious null statem...
ExprResult DefaultLvalueConversion(Expr *E)
DeclContext * CurContext
CurContext - This is the current declaration context of parsing.
DeclarationNameInfo GetNameFromUnqualifiedId(const UnqualifiedId &Name)
Retrieves the declaration name from a parsed unqualified-id.
void ActOnStartOfDeferStmt(SourceLocation DeferLoc, Scope *CurScope)
bool isUnevaluatedContext() const
Determines whether we are currently in a context that is not evaluated as per C++ [expr] p5.
StmtResult ActOnCapScopeReturnStmt(SourceLocation ReturnLoc, Expr *RetValExp, NamedReturnInfo &NRInfo, bool SupressSimplerImplicitMoves)
ActOnCapScopeReturnStmt - Utility routine to type-check return statements for capturing scopes.
StmtResult ActOnCapturedRegionEnd(Stmt *S)
StmtResult ActOnForStmt(SourceLocation ForLoc, SourceLocation LParenLoc, Stmt *First, ConditionResult Second, FullExprArg Third, SourceLocation RParenLoc, Stmt *Body)
StmtResult ActOnIndirectGotoStmt(SourceLocation GotoLoc, SourceLocation StarLoc, Expr *DestExp)
ExprResult CheckPlaceholderExpr(Expr *E)
Check for operands with placeholder types and complain if found.
bool inTemplateInstantiation() const
Determine whether we are currently performing template instantiation.
SourceManager & getSourceManager() const
ExprResult PerformMoveOrCopyInitialization(const InitializedEntity &Entity, const NamedReturnInfo &NRInfo, Expr *Value, bool SupressSimplerImplicitMoves=false)
Perform the initialization of a potentially-movable value, which is the result of return value.
void ActOnInitializerError(Decl *Dcl)
ActOnInitializerError - Given that there was an error parsing an initializer for the given declaratio...
StmtResult ActOnCXXForRangeStmt(Scope *S, SourceLocation ForLoc, SourceLocation CoawaitLoc, Stmt *InitStmt, Stmt *LoopVar, SourceLocation ColonLoc, Expr *Collection, SourceLocation RParenLoc, BuildForRangeKind Kind, ArrayRef< MaterializeTemporaryExpr * > LifetimeExtendTemps={})
ActOnCXXForRangeStmt - Check and build a C++11 for-range statement.
bool isSFINAEContext() const
StmtResult BuildReturnStmt(SourceLocation ReturnLoc, Expr *RetValExp, bool AllowRecovery=false)
ExprResult ActOnBinOp(Scope *S, SourceLocation TokLoc, tok::TokenKind Kind, Expr *LHSExpr, Expr *RHSExpr)
Binary Operators. 'Tok' is the token for the operator.
void PushCapturedRegionScope(Scope *RegionScope, CapturedDecl *CD, RecordDecl *RD, CapturedRegionKind K, unsigned OpenMPCaptureLevel=0)
void setFunctionHasMustTail()
void setFunctionHasBranchProtectedScope()
StmtResult ActOnFinishSEHFinallyBlock(SourceLocation Loc, Stmt *Block)
StmtResult BuildCXXForRangeStmt(SourceLocation ForLoc, SourceLocation CoawaitLoc, Stmt *InitStmt, SourceLocation ColonLoc, Stmt *RangeDecl, Stmt *Begin, Stmt *End, Expr *Cond, Expr *Inc, Stmt *LoopVarDecl, SourceLocation RParenLoc, BuildForRangeKind Kind, ArrayRef< MaterializeTemporaryExpr * > LifetimeExtendTemps={})
BuildCXXForRangeStmt - Build or instantiate a C++11 for-range statement.
StmtResult ActOnDoStmt(SourceLocation DoLoc, Stmt *Body, SourceLocation WhileLoc, SourceLocation CondLParen, Expr *Cond, SourceLocation CondRParen)
StmtResult ActOnStartOfSwitchStmt(SourceLocation SwitchLoc, SourceLocation LParenLoc, Stmt *InitStmt, ConditionResult Cond, SourceLocation RParenLoc)
ExprResult VerifyIntegerConstantExpression(Expr *E, llvm::APSInt *Result, VerifyICEDiagnoser &Diagnoser, AllowFoldKind CanFold=AllowFoldKind::No)
VerifyIntegerConstantExpression - Verifies that an expression is an ICE, and reports the appropriate ...
StmtResult ActOnMSDependentExistsStmt(SourceLocation KeywordLoc, bool IsIfExists, CXXScopeSpec &SS, UnqualifiedId &Name, Stmt *Nested)
@ PotentiallyEvaluated
The current expression is potentially evaluated at run time, which means that code may be generated t...
@ ImmediateFunctionContext
In addition of being constant evaluated, the current expression occurs in an immediate function conte...
StmtResult ActOnSEHExceptBlock(SourceLocation Loc, Expr *FilterExpr, Stmt *Block)
void ActOnAfterCompoundStatementLeadingPragmas()
StmtResult ActOnDeclStmt(DeclGroupPtrTy Decl, SourceLocation StartLoc, SourceLocation EndLoc)
bool RequireCompleteType(SourceLocation Loc, QualType T, CompleteTypeKind Kind, TypeDiagnoser &Diagnoser)
Ensure that the type T is a complete type.
void ActOnFields(Scope *S, SourceLocation RecLoc, Decl *TagDecl, ArrayRef< Decl * > Fields, SourceLocation LBrac, SourceLocation RBrac, const ParsedAttributesView &AttrList)
bool LookupQualifiedName(LookupResult &R, DeclContext *LookupCtx, bool InUnqualifiedLookup=false)
Perform qualified name lookup into a given context.
void DiscardCleanupsInEvaluationContext()
SmallVector< ExpressionEvaluationContextRecord, 8 > ExprEvalContexts
A stack of expression evaluation contexts.
void PushDeclContext(Scope *S, DeclContext *DC)
Set the current declaration context until it gets popped.
StmtResult ActOnAttributedStmt(const ParsedAttributes &AttrList, Stmt *SubStmt)
SourceManager & SourceMgr
DiagnosticsEngine & Diags
void ActOnStartSEHFinallyBlock()
ExprResult PerformCopyInitialization(const InitializedEntity &Entity, SourceLocation EqualLoc, ExprResult Init, bool TopLevelOfInitList=false, bool AllowExplicit=false)
void ActOnAbortSEHFinallyBlock()
friend class InitializationSequence
void diagnoseIgnoredQualifiers(unsigned DiagID, unsigned Quals, SourceLocation FallbackLoc, SourceLocation ConstQualLoc=SourceLocation(), SourceLocation VolatileQualLoc=SourceLocation(), SourceLocation RestrictQualLoc=SourceLocation(), SourceLocation AtomicQualLoc=SourceLocation(), SourceLocation UnalignedQualLoc=SourceLocation())
void ActOnDeferStmtError(Scope *CurScope)
QualType SubstAutoTypeDependent(QualType TypeWithAuto)
@ BFRK_Check
Determining whether a for-range statement could be built.
@ BFRK_Build
Initial building of a for-range statement.
@ BFRK_Rebuild
Instantiation or recovery rebuild of a for-range statement.
StmtResult ActOnCXXCatchBlock(SourceLocation CatchLoc, Decl *ExDecl, Stmt *HandlerBlock)
ActOnCXXCatchBlock - Takes an exception declaration and a handler block and creates a proper catch ha...
void ActOnCaseStmtBody(Stmt *CaseStmt, Stmt *SubStmt)
ActOnCaseStmtBody - This installs a statement as the body of a case.
void AddInitializerToDecl(Decl *dcl, Expr *init, bool DirectInit)
AddInitializerToDecl - Adds the initializer Init to the declaration dcl.
bool DiagnoseAssignmentResult(AssignConvertType ConvTy, SourceLocation Loc, QualType DstType, QualType SrcType, Expr *SrcExpr, AssignmentAction Action, bool *Complained=nullptr)
DiagnoseAssignmentResult - Emit a diagnostic, if required, for the assignment conversion type specifi...
SemaDiagnosticBuilder targetDiag(SourceLocation Loc, unsigned DiagID, const FunctionDecl *FD=nullptr)
ExprResult CreateRecoveryExpr(SourceLocation Begin, SourceLocation End, ArrayRef< Expr * > SubExprs, QualType T=QualType())
Attempts to produce a RecoveryExpr after some AST node cannot be created.
StmtResult ActOnIfStmt(SourceLocation IfLoc, IfStatementKind StatementKind, SourceLocation LParenLoc, Stmt *InitStmt, ConditionResult Cond, SourceLocation RParenLoc, Stmt *ThenVal, SourceLocation ElseLoc, Stmt *ElseVal)
std::string getTemplateArgumentBindingsText(const TemplateParameterList *Params, const TemplateArgumentList &Args)
Produces a formatted string that describes the binding of template parameters to template arguments.
ExprResult ActOnUnaryExprOrTypeTraitExpr(SourceLocation OpLoc, UnaryExprOrTypeTrait ExprKind, bool IsType, void *TyOrEx, SourceRange ArgRange)
ActOnUnaryExprOrTypeTraitExpr - Handle sizeof(type) and sizeof expr and the same for alignof and __al...
ForRangeStatus BuildForRangeBeginEndCall(SourceLocation Loc, SourceLocation RangeLoc, const DeclarationNameInfo &NameInfo, LookupResult &MemberLookup, OverloadCandidateSet *CandidateSet, Expr *Range, ExprResult *CallExpr)
Build a call to 'begin' or 'end' for a C++11 for-range statement.
sema::CompoundScopeInfo & getCurCompoundScope() const
TemplateDeductionResult DeduceAutoType(TypeLoc AutoTypeLoc, Expr *Initializer, QualType &Result, sema::TemplateDeductionInfo &Info, bool DependentDeduction=false, bool IgnoreConstraints=false, TemplateSpecCandidateSet *FailedTSC=nullptr)
Deduce the type for an auto type-specifier (C++11 [dcl.spec.auto]p6)
void ActOnFinishOfCompoundStmt()
StmtResult ActOnCompoundStmt(SourceLocation L, SourceLocation R, ArrayRef< Stmt * > Elts, bool isStmtExpr)
bool IsValueInFlagEnum(const EnumDecl *ED, const llvm::APInt &Val, bool AllowMask) const
IsValueInFlagEnum - Determine if a value is allowed as part of a flag enum.
StmtResult ActOnLabelStmt(SourceLocation IdentLoc, LabelDecl *TheDecl, SourceLocation ColonLoc, Stmt *SubStmt)
StmtResult ActOnCXXTryBlock(SourceLocation TryLoc, Stmt *TryBlock, ArrayRef< Stmt * > Handlers)
ActOnCXXTryBlock - Takes a try compound-statement and a number of handlers and creates a try statemen...
OpaquePtr< DeclGroupRef > DeclGroupPtrTy
StmtResult ActOnDefaultStmt(SourceLocation DefaultLoc, SourceLocation ColonLoc, Stmt *SubStmt, Scope *CurScope)
StmtResult ActOnCaseStmt(SourceLocation CaseLoc, ExprResult LHS, SourceLocation DotDotDotLoc, ExprResult RHS, SourceLocation ColonLoc)
SmallVector< std::pair< Scope *, SourceLocation >, 2 > CurrentDefer
Stack of '_Defer' statements that are currently being parsed, as well as the locations of their '_Def...
StmtResult FinishCXXForRangeStmt(Stmt *ForRange, Stmt *Body)
FinishCXXForRangeStmt - Attach the body to a C++0x for-range statement.
ExprResult ActOnFinishFullExpr(Expr *Expr, bool DiscardedValue)
Encodes a location in the source.
bool isValid() const
Return true if this is a valid SourceLocation object.
SourceLocation getLocWithOffset(IntTy Offset) const
Return a source location with the specified offset from this SourceLocation.
bool isMacroBodyExpansion(SourceLocation Loc) const
Tests whether the given source location represents the expansion of a macro body.
bool isInSystemMacro(SourceLocation loc) const
Returns whether Loc is expanded from a macro in a system header.
bool isInSystemHeader(SourceLocation Loc) const
Returns if a SourceLocation is in a system header.
A trivial tuple used to represent a source range.
SourceLocation getBegin() const
Stmt - This represents one statement.
SourceLocation getEndLoc() const LLVM_READONLY
SourceRange getSourceRange() const LLVM_READONLY
SourceLocation tokens are not useful in isolation - they are low level value objects created/interpre...
static std::tuple< bool, const Attr *, const Attr * > determineLikelihoodConflict(const Stmt *Then, const Stmt *Else)
static const Attr * getLikelihoodAttr(const Stmt *S)
SourceLocation getBeginLoc() const LLVM_READONLY
SwitchStmt - This represents a 'switch' stmt.
static SwitchStmt * Create(const ASTContext &Ctx, Stmt *Init, VarDecl *Var, Expr *Cond, SourceLocation LParenLoc, SourceLocation RParenLoc)
Create a switch statement.
SwitchCase * getSwitchCaseList()
DeclStmt * getConditionVariableDeclStmt()
If this SwitchStmt has a condition variable, return the faux DeclStmt associated with the creation of...
void setAllEnumCasesCovered()
Set a flag in the SwitchStmt indicating that if the 'switch (X)' is a switch over an enum value then ...
Represents the declaration of a struct/union/class/enum.
void startDefinition()
Starts the definition of this tag declaration.
TemplateSpecCandidateSet - A set of generalized overload candidates, used in template specializations...
void NoteCandidates(Sema &S, SourceLocation Loc)
NoteCandidates - When no template specialization match is found, prints diagnostic messages containin...
Base wrapper for a particular "section" of type source info.
QualType getType() const
Get the type for which this source info wrapper provides information.
T castAs() const
Convert to the specified TypeLoc type, asserting that this TypeLoc is of the desired type.
T getAsAdjusted() const
Convert to the specified TypeLoc type, returning a null TypeLoc if this TypeLoc is not of the desired...
A container of type source information.
TypeLoc getTypeLoc() const
Return the TypeLoc wrapper for the type source info.
QualType getType() const
Return the type wrapped by this type source info.
The base class of the type hierarchy.
bool isSignedIntegerOrEnumerationType() const
Determines whether this is an integer type that is signed or an enumeration types whose underlying ty...
bool isRValueReferenceType() const
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
bool isPointerType() 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
bool isEnumeralType() const
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool isIntegralOrEnumerationType() const
Determine whether this type is an integral or enumeration type.
AutoType * getContainedAutoType() const
Get the AutoType whose type will be deduced for a variable with an initializer of this type.
bool isSpecificBuiltinType(unsigned K) const
Test for a particular builtin type.
bool isDependentType() const
Whether this type is a dependent type, meaning that its definition somehow depends on a template para...
bool containsErrors() const
Whether this type is an error type.
bool isMatrixType() const
EnumDecl * castAsEnumDecl() const
const ArrayType * getAsArrayTypeUnsafe() const
A variant of getAs<> for array types which silently discards qualifiers from the outermost type.
bool isUndeducedType() const
Determine whether this type is an undeduced type, meaning that it somehow involves a C++11 'auto' typ...
bool isObjectType() const
Determine whether this type is an object type.
bool isIncompleteType(NamedDecl **Def=nullptr) const
Types are partitioned into 3 broad categories (C99 6.2.5p1): object types, function types,...
const T * getAsCanonical() const
If this type is canonically the specified type, return its canonical type cast to that specified type...
TypeClass getTypeClass() const
bool isCanonicalUnqualified() const
Determines if this type would be canonical if it had no further qualification.
const T * getAs() const
Member-template getAs<specific type>'.
bool isRecordType() const
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
Expr * getSubExpr() const
Represents a C++ unqualified-id that has been parsed.
void setType(QualType newType)
Represents a variable declaration or definition.
static VarDecl * Create(ASTContext &C, DeclContext *DC, SourceLocation StartLoc, SourceLocation IdLoc, const IdentifierInfo *Id, QualType T, TypeSourceInfo *TInfo, StorageClass S)
SourceRange getSourceRange() const override LLVM_READONLY
Source range that this declaration covers.
bool hasGlobalStorage() const
Returns true for all variables that do not have local storage.
bool isExceptionVariable() const
Determine whether this variable is the exception variable in a C++ catch statememt or an Objective-C ...
const Expr * getInit() const
bool hasLocalStorage() const
Returns true if a variable with function scope is a non-static local variable.
bool isLocalVarDecl() const
Returns true for local variable declarations other than parameters.
bool hasDependentAlignment() const
Determines if this variable's alignment is dependent.
Represents a C array with a specified size that is not an integer-constant-expression.
static WhileStmt * Create(const ASTContext &Ctx, VarDecl *Var, Expr *Cond, Stmt *Body, SourceLocation WL, SourceLocation LParenLoc, SourceLocation RParenLoc)
Create a while statement.
ValueDecl * getVariable() const
bool isVariableCapture() const
SourceLocation getLocation() const
Retrieve the location at which this variable was captured.
bool isVLATypeCapture() const
bool isThisCapture() const
bool isReferenceCapture() const
Retains information about a captured region.
unsigned short OpenMPLevel
unsigned short CapRegionKind
The kind of captured region.
RecordDecl * TheRecordDecl
The captured record type.
CapturedDecl * TheCapturedDecl
The CapturedDecl for this statement.
QualType ReturnType
ReturnType - The target type of return statements in this context, or null if unknown.
SmallVector< Capture, 4 > Captures
Captures - The captures.
bool HasImplicitReturnType
Contains information about the compound statement currently being parsed.
FPOptions InitialFPFeatures
FP options at the beginning of the compound statement, prior to any pragma.
void setHasEmptyLoopBodies()
Retains information about a function, method, or block that is currently being parsed.
llvm::PointerIntPair< SwitchStmt *, 1, bool > SwitchInfo
A SwitchStmt, along with a flag indicating if its list of case statements is incomplete (because we d...
SourceLocation FirstCXXOrObjCTryLoc
First C++ 'try' or ObjC @try statement in the current function.
SourceLocation FirstCoroutineStmtLoc
First coroutine statement in the current function.
enum clang::sema::FunctionScopeInfo::@340304006310276167163023075110222134352007243353 FirstTryType
StringRef getFirstCoroutineStmtKeyword() const
SourceLocation FirstReturnLoc
First 'return' statement in the current function.
SourceLocation FirstSEHTryLoc
First SEH '__try' statement in the current function.
void setHasCXXTry(SourceLocation TryLoc)
SmallVector< CompoundScopeInfo, 4 > CompoundScopes
The stack of currently active compound statement scopes in the function.
void setHasSEHTry(SourceLocation TryLoc)
SmallVector< SwitchInfo, 8 > SwitchStack
SwitchStack - This is the current set of active switch statements in the block.
CXXMethodDecl * CallOperator
The lambda's compiler-generated operator().
Provides information about an attempted template argument deduction, whose success or failure was des...
TemplateArgument SecondArg
The second template argument to which the template argument deduction failure refers.
TemplateArgument FirstArg
The first template argument to which the template argument deduction failure refers.
Defines the clang::TargetInfo interface.
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const internal::VariadicAllOfMatcher< Decl > decl
Matches declarations.
bool This(InterpState &S, CodePtr OpPC)
bool Cast(InterpState &S, CodePtr OpPC)
void checkExprLifetimeMustTailArg(Sema &SemaRef, const InitializedEntity &Entity, Expr *Init)
Check that the lifetime of the given expr (and its subobjects) is sufficient, assuming that it is pas...
The JSON file list parser is used to communicate input to InstallAPI.
CanQual< Type > CanQualType
Represents a canonical, potentially-qualified type.
bool isa(CodeGen::Address addr)
Expr * IgnoreElidableImplicitConstructorSingleStep(Expr *E)
if(T->getSizeExpr()) TRY_TO(TraverseStmt(const_cast< Expr * >(T -> getSizeExpr())))
@ OR_Deleted
Succeeded, but refers to a deleted function.
@ OR_Success
Overload resolution succeeded.
bool isReservedInAllContexts(ReservedIdentifierStatus Status)
Determine whether an identifier is reserved in all contexts.
IfStatementKind
In an if statement, this denotes whether the statement is a constexpr or consteval if statement.
Expr * IgnoreExprNodes(Expr *E, FnTys &&... Fns)
Given an expression E and functions Fn_1,...,Fn_n : Expr * -> Expr *, Recursively apply each of the f...
@ RQ_None
No ref-qualifier was provided.
@ OCD_AllCandidates
Requests that all candidates be shown.
LambdaCaptureKind
The different capture forms in a lambda introducer.
@ LCK_ByRef
Capturing by reference.
@ Seq
'seq' clause, allowed on 'loop' and 'routine' directives.
bool operator==(const CallGraphNode::CallRecord &LHS, const CallGraphNode::CallRecord &RHS)
SmallVector< Attr *, 4 > AttrVec
AttrVec - A vector of Attr, which is how they are stored on the AST.
nullptr
This class represents a compute construct, representing a 'Kind' of ‘parallel’, 'serial',...
CapturedRegionKind
The different kinds of captured statement.
bool isLambdaCallOperator(const CXXMethodDecl *MD)
@ Result
The result type of a method or function.
AssignConvertType
AssignConvertType - All of the 'assignment' semantic checks return this enum to indicate whether the ...
bool hasArmZT0State(const FunctionDecl *FD)
Returns whether the given FunctionDecl has Arm ZT0 state.
@ Struct
The "struct" keyword.
@ Type
The name was classified as a type.
bool isLambdaConversionOperator(CXXConversionDecl *C)
@ Deduced
The normal deduced case.
std::pair< SourceLocation, PartialDiagnostic > PartialDiagnosticAt
A partial diagnostic along with the source location where this diagnostic occurs.
@ VK_XValue
An x-value expression is a reference to an object with independent storage but which can be "moved",...
@ VK_LValue
An l-value expression is a reference to an object with independent storage.
Expr * IgnoreParensSingleStep(Expr *E)
Expr * IgnoreImplicitAsWrittenSingleStep(Expr *E)
DynamicRecursiveASTVisitorBase< false > DynamicRecursiveASTVisitor
TemplateDeductionResult
Describes the result of template argument deduction.
@ Success
Template argument deduction was successful.
@ Inconsistent
Template argument deduction produced inconsistent deduced values for the given template parameter.
@ AlreadyDiagnosed
Some error which was already diagnosed.
U cast(CodeGen::Address addr)
@ CaseValue
Expression in a case label.
bool IsArmStreamingFunction(const FunctionDecl *FD, bool IncludeLocallyStreaming)
Returns whether the given FunctionDecl has an __arm[_locally]_streaming attribute.
ActionResult< Expr * > ExprResult
@ CapturedContext
Parameter for captured context.
ActionResult< Stmt * > StmtResult
bool hasArmZAState(const FunctionDecl *FD)
Returns whether the given FunctionDecl has Arm ZA state.
DeclarationNameInfo - A collector data type for bundling together a DeclarationName and the correspon...
EvalResult is a struct with detailed info about an evaluated expression.
@ MoveEligibleAndCopyElidable
bool isMoveEligible() const
bool isCopyElidable() const
const VarDecl * Candidate
static unsigned getHashValue(const CatchHandlerType &Base)
static bool isEqual(const CatchHandlerType &LHS, const CatchHandlerType &RHS)