23#include "llvm/ADT/APSInt.h"
24#include "llvm/ADT/SmallVector.h"
25#include "llvm/ADT/StringRef.h"
26#include "llvm/Support/Casting.h"
34using namespace ast_matchers;
41 std::string VisitStmt(
const Stmt *S) {
return S->getStmtClassName(); }
44 return "BinaryOperator(" + BO->
getOpcodeStr().str() +
")";
58static std::string getDREAncestorString(
const DeclRefExpr *DRE,
62 StmtDebugPrinter StmtPriner;
65 SS << StmtPriner.Visit(St);
69 if (StParents.
size() > 1)
70 return "unavailable due to multiple parents";
71 if (StParents.
size() == 0)
91 internal::ASTMatchFinder *Finder,
92 internal::BoundNodesTreeBuilder *Builder,
93 internal::ASTMatchFinder::BindKind Bind,
94 const bool ignoreUnevaluatedContext)
95 : Matcher(Matcher), Finder(Finder), Builder(Builder), Bind(Bind),
96 Matches(
false), ignoreUnevaluatedContext(ignoreUnevaluatedContext) {
107 *Builder = ResultBindings;
126 if (isa<FunctionDecl, BlockDecl, ObjCMethodDecl>(
Node))
134 if (ignoreUnevaluatedContext)
136 return DynamicRecursiveASTVisitor::TraverseGenericSelectionExpr(
Node);
142 if (ignoreUnevaluatedContext)
144 return DynamicRecursiveASTVisitor::TraverseUnaryExprOrTypeTraitExpr(
Node);
149 if (ignoreUnevaluatedContext)
151 return DynamicRecursiveASTVisitor::TraverseTypeOfExprTypeLoc(
Node);
156 if (ignoreUnevaluatedContext)
158 return DynamicRecursiveASTVisitor::TraverseDecltypeTypeLoc(
Node);
163 if (ignoreUnevaluatedContext)
165 return DynamicRecursiveASTVisitor::TraverseCXXNoexceptExpr(
Node);
170 if (ignoreUnevaluatedContext)
172 return DynamicRecursiveASTVisitor::TraverseCXXTypeidExpr(
Node);
178 return DynamicRecursiveASTVisitor::TraverseCXXDefaultInitExpr(
Node);
194 template <
typename T>
bool match(
const T &
Node) {
195 internal::BoundNodesTreeBuilder RecursiveBuilder(*Builder);
198 &RecursiveBuilder)) {
199 ResultBindings.addMatch(RecursiveBuilder);
201 if (Bind != internal::ASTMatchFinder::BK_All)
207 const internal::DynTypedMatcher *
const Matcher;
208 internal::ASTMatchFinder *
const Finder;
209 internal::BoundNodesTreeBuilder *
const Builder;
210 internal::BoundNodesTreeBuilder ResultBindings;
211 const internal::ASTMatchFinder::BindKind Bind;
213 bool ignoreUnevaluatedContext;
225 const DynTypedMatcher &DTM =
static_cast<DynTypedMatcher
>(innerMatcher);
234 const DynTypedMatcher &DTM =
static_cast<DynTypedMatcher
>(innerMatcher);
244 return !Handler->isSafeBufferOptOut(
Node.getBeginLoc());
249 return Handler->ignoreUnsafeBufferInContainer(
Node.getBeginLoc());
254 if (Finder->getASTContext().getLangOpts().CPlusPlus)
255 return Handler->ignoreUnsafeBufferInLibcCall(
Node.getBeginLoc());
260 return innerMatcher.matches(*
Node.getSubExpr(), Finder, Builder);
265 return Node.getOpcode() == UnaryOperator::Opcode::UO_PreInc;
275 hasCastKind(CastKind::CK_LValueToRValue),
276 castSubExpr(innerMatcher)),
287static internal::Matcher<Stmt>
299 forEachArgumentWithParamType(
305 auto CastOperandMatcher =
307 hasCastKind(CastKind::CK_PointerToBoolean)),
310 auto CompOperandMatcher =
316 auto PtrSubtractionMatcher =
323 eachOf(hasLHS(InnerMatcher),
324 hasRHS(InnerMatcher)));
327 return stmt(
anyOf(CallArgMatcher, CastOperandMatcher, CompOperandMatcher,
328 PtrSubtractionMatcher));
343static internal::Matcher<Stmt>
350 auto IfStmtThen =
ifStmt(hasThen(InnerMatcher));
351 auto IfStmtElse =
ifStmt(hasElse(InnerMatcher));
353 return stmt(
anyOf(CompStmt, IfStmtThen, IfStmtElse));
366 assert(
Node.getNumArgs() == 2 &&
367 "expecting a two-parameter std::span constructor");
368 const Expr *Arg0 =
Node.getArg(0)->IgnoreImplicit();
369 const Expr *Arg1 =
Node.getArg(1)->IgnoreImplicit();
370 auto HaveEqualConstantValues = [&Finder](
const Expr *E0,
const Expr *E1) {
372 if (
auto E1CV = E1->getIntegerConstantExpr(Finder->getASTContext())) {
373 return APSInt::compareValues(*E0CV, *E1CV) == 0;
377 auto AreSameDRE = [](
const Expr *E0,
const Expr *E1) {
378 if (
auto *DRE0 = dyn_cast<DeclRefExpr>(E0))
379 if (
auto *DRE1 = dyn_cast<DeclRefExpr>(E1)) {
380 return DRE0->getDecl() == DRE1->getDecl();
384 std::optional<APSInt> Arg1CV =
387 if (Arg1CV && Arg1CV->isZero())
391 case Stmt::CXXNewExprClass:
392 if (
auto Size = cast<CXXNewExpr>(Arg0)->getArraySize()) {
394 return AreSameDRE((*Size)->IgnoreImplicit(), Arg1) ||
395 HaveEqualConstantValues(*Size, Arg1);
398 if (!cast<CXXNewExpr>(Arg0)->hasPlaceholderType()) {
400 return Arg1CV && Arg1CV->isOne();
403 case Stmt::UnaryOperatorClass:
404 if (cast<UnaryOperator>(Arg0)->getOpcode() ==
405 UnaryOperator::Opcode::UO_AddrOf)
407 return Arg1CV && Arg1CV->isOne();
409 case Stmt::CallExprClass:
410 if (
const auto *CE = dyn_cast<CallExpr>(Arg0)) {
411 const auto FnDecl = CE->getDirectCallee();
412 if (FnDecl && FnDecl->getNameAsString() ==
"addressof" &&
413 FnDecl->isInStdNamespace()) {
414 return Arg1CV && Arg1CV->isOne();
424 if (
auto *ConstArrTy =
425 Finder->getASTContext().getAsConstantArrayType(Arg0Ty)) {
426 const APSInt ConstArrSize =
APSInt(ConstArrTy->getSize());
429 return Arg1CV && APSInt::compareValues(ConstArrSize, *Arg1CV) == 0;
443 if (
const auto *CATy =
444 dyn_cast<ConstantArrayType>(
Node.getBase()
445 ->IgnoreParenImpCasts()
447 ->getUnqualifiedDesugaredType())) {
448 limit = CATy->getLimitedSize();
449 }
else if (
const auto *SLiteral = dyn_cast<StringLiteral>(
450 Node.getBase()->IgnoreParenImpCasts())) {
451 limit = SLiteral->getLength() + 1;
457 if (
Node.getIdx()->EvaluateAsInt(EVResult, Finder->getASTContext())) {
458 llvm::APSInt ArrIdx = EVResult.
Val.
getInt();
461 if (ArrIdx.isNonNegative() && ArrIdx.getLimitedValue() < limit)
468 return Node.getNumArgs() ==
Num;
471namespace libc_func_matchers {
487 StringRef
matchName(StringRef FunName,
bool isBuiltin) {
489 if (isBuiltin && FunName.starts_with(
"__builtin_"))
493 FunName.drop_front(10 ));
495 if (FunName.starts_with(
"__asan_"))
503 if (Name.starts_with(
"__") && Name.ends_with(
"_chk"))
505 Name.drop_front(2).drop_back(4) );
510 if (Name.ends_with(
"_s"))
511 return Name.drop_back(2 );
544 bool isKprintf =
false) {
545 class StringFormatStringHandler
549 const Expr *&UnsafeArg;
552 StringFormatStringHandler(
const CallExpr *
Call,
unsigned FmtArgIdx,
553 const Expr *&UnsafeArg)
554 :
Call(
Call), FmtArgIdx(FmtArgIdx), UnsafeArg(UnsafeArg) {}
557 const char *startSpecifier,
558 unsigned specifierLen,
560 if (FS.getConversionSpecifier().getKind() ==
561 analyze_printf::PrintfConversionSpecifier::sArg) {
562 unsigned ArgIdx = FS.getPositionalArgIndex() + FmtArgIdx;
564 if (0 < ArgIdx && ArgIdx < Call->getNumArgs())
566 UnsafeArg =
Call->getArg(ArgIdx);
575 const Expr *Fmt =
Call->getArg(FmtArgIdx);
580 if (SL->getCharByteWidth() == 1)
581 FmtStr = SL->getString();
582 else if (
auto EvaledFmtStr = SL->tryEvaluateString(Ctx))
583 FmtStr = *EvaledFmtStr;
585 goto CHECK_UNSAFE_PTR;
587 StringFormatStringHandler Handler(
Call, FmtArgIdx, UnsafeArg);
590 Handler, FmtStr.begin(), FmtStr.end(), Ctx.
getLangOpts(),
598 llvm::make_range(
Call->arg_begin() + FmtArgIdx,
Call->arg_end()),
599 [&UnsafeArg](
const Expr *Arg) ->
bool {
600 if (Arg->getType()->isPointerType() && !isNullTermPointer(Arg)) {
618 static std::unique_ptr<std::set<StringRef>> PredefinedNames =
nullptr;
619 if (!PredefinedNames)
621 std::make_unique<std::set<StringRef>, std::set<StringRef>>({
693 auto *II =
Node.getIdentifier();
699 II->getName(),
Node.getBuiltinID());
702 if (PredefinedNames->find(Name) != PredefinedNames->end())
705 std::string NameWCS = Name.str();
706 size_t WcsPos = NameWCS.find(
"wcs");
708 while (WcsPos != std::string::npos) {
709 NameWCS[WcsPos++] =
's';
710 NameWCS[WcsPos++] =
't';
711 NameWCS[WcsPos++] =
'r';
712 WcsPos = NameWCS.find(
"wcs", WcsPos);
714 if (PredefinedNames->find(NameWCS) != PredefinedNames->end())
718 return Name.ends_with(
"scanf");
725 auto *II =
Node.getIdentifier();
731 II->getName(),
Node.getBuiltinID());
733 if (!Name.ends_with(
"printf"))
735 return Name.starts_with(
"v");
741 auto *II =
Node.getIdentifier();
747 II->getName(),
Node.getBuiltinID());
749 if (!Name.ends_with(
"printf") ||
751 Name.starts_with(
"v"))
754 StringRef Prefix = Name.drop_back(6);
756 if (Prefix.ends_with(
"w"))
757 Prefix = Prefix.drop_back(1);
758 return Prefix ==
"s";
765 auto *II =
Node.getIdentifier();
771 II->getName(),
Node.getBuiltinID());
773 if (!Name.ends_with(
"printf") || Name.starts_with(
"v"))
776 StringRef Prefix = Name.drop_back(6);
778 if (Prefix.ends_with(
"w"))
779 Prefix = Prefix.drop_back(1);
781 return Prefix.empty() || Prefix ==
"k" || Prefix ==
"f" || Prefix ==
"sn";
789 clang::ast_matchers::internal::Matcher<Expr>,
790 UnsafeStringArgMatcher) {
794 assert(FD &&
"It should have been checked that FD is non-null.");
813 const Expr *UnsafeArg;
816 return UnsafeStringArgMatcher.matches(*UnsafeArg, Finder, Builder);
822 bool isKprintf =
false;
823 const Expr *UnsafeArg;
826 isKprintf = II->getName() ==
"kprintf";
828 return UnsafeStringArgMatcher.matches(*UnsafeArg, Finder, Builder);
838 const Expr *UnsafeArg;
841 return UnsafeStringArgMatcher.matches(*UnsafeArg, Finder, Builder);
847 for (
auto Arg :
Node.arguments())
849 if (UnsafeStringArgMatcher.matches(*Arg, Finder, Builder))
873 assert(FD &&
"It should have been checked that FD is non-null.");
884 const Expr *Buf =
Node.getArg(0), *Size =
Node.getArg(1);
887 !Size->getType()->isIntegerType())
891 static StringRef SizedObjs[] = {
"span",
"array",
"vector",
892 "basic_string_view",
"basic_string"};
894 Size = Size->IgnoreParenImpCasts();
895 if (
auto *MCEPtr = dyn_cast<CXXMemberCallExpr>(Buf))
896 if (
auto *MCESize = dyn_cast<CXXMemberCallExpr>(Size)) {
897 auto *DREOfPtr = dyn_cast<DeclRefExpr>(
898 MCEPtr->getImplicitObjectArgument()->IgnoreParenImpCasts());
899 auto *DREOfSize = dyn_cast<DeclRefExpr>(
900 MCESize->getImplicitObjectArgument()->IgnoreParenImpCasts());
902 if (!DREOfPtr || !DREOfSize)
904 if (DREOfPtr->getDecl() != DREOfSize->getDecl())
906 if (MCEPtr->getMethodDecl()->getName() !=
"data")
909 if (MCESize->getMethodDecl()->getName() ==
"size_bytes" ||
914 MCESize->getMethodDecl()->getName() ==
"size")
915 for (StringRef SizedObj : SizedObjs)
916 if (MCEPtr->getRecordDecl()->isInStdNamespace() &&
917 MCEPtr->getRecordDecl()->getCanonicalDecl()->getName() ==
931 if (Size->EvaluateAsConstantExpr(ER, Ctx)) {
934 return APSInt::compareValues(EVal,
APSInt(CAT->getSize(),
true)) != 0;
965#include "clang/Analysis/Analyses/UnsafeBufferUsageGadgets.def"
971 using Matcher =
decltype(
stmt());
973 Gadget(Kind K) : K(K) {}
978 StringRef getDebugName()
const {
983#include "clang/Analysis/Analyses/UnsafeBufferUsageGadgets.def"
985 llvm_unreachable(
"Unhandled Gadget::Kind enum");
989 virtual bool isWarningGadget()
const = 0;
997 virtual DeclUseList getClaimedVarUseSites()
const = 0;
999 virtual ~Gadget() =
default;
1007class WarningGadget :
public Gadget {
1009 WarningGadget(Kind K) : Gadget(K) {}
1011 static bool classof(
const Gadget *G) {
return G->isWarningGadget(); }
1012 bool isWarningGadget() const final {
return true; }
1015 bool IsRelatedToDecl,
1023class FixableGadget :
public Gadget {
1025 FixableGadget(Kind K) : Gadget(K) {}
1027 static bool classof(
const Gadget *G) {
return !G->isWarningGadget(); }
1028 bool isWarningGadget() const final {
return false; }
1033 virtual std::optional<FixItList> getFixits(
const FixitStrategy &)
const {
1034 return std::nullopt;
1043 virtual std::optional<std::pair<const VarDecl *, const VarDecl *>>
1044 getStrategyImplications()
const {
1045 return std::nullopt;
1049static auto toSupportedVariable() {
return to(
varDecl()); }
1051using FixableGadgetList = std::vector<std::unique_ptr<FixableGadget>>;
1052using WarningGadgetList = std::vector<std::unique_ptr<WarningGadget>>;
1056class IncrementGadget :
public WarningGadget {
1057 static constexpr const char *
const OpTag =
"op";
1062 : WarningGadget(
Kind::Increment),
1065 static bool classof(
const Gadget *G) {
1066 return G->getKind() == Kind::Increment;
1069 static Matcher matcher() {
1077 bool IsRelatedToDecl,
1083 DeclUseList getClaimedVarUseSites()
const override {
1085 if (
const auto *DRE =
1087 Uses.push_back(DRE);
1090 return std::move(Uses);
1096class DecrementGadget :
public WarningGadget {
1097 static constexpr const char *
const OpTag =
"op";
1102 : WarningGadget(
Kind::Decrement),
1105 static bool classof(
const Gadget *G) {
1106 return G->getKind() == Kind::Decrement;
1109 static Matcher matcher() {
1117 bool IsRelatedToDecl,
1123 DeclUseList getClaimedVarUseSites()
const override {
1124 if (
const auto *DRE =
1135class ArraySubscriptGadget :
public WarningGadget {
1136 static constexpr const char *
const ArraySubscrTag =
"ArraySubscript";
1141 : WarningGadget(
Kind::ArraySubscript),
1144 static bool classof(
const Gadget *G) {
1145 return G->getKind() == Kind::ArraySubscript;
1148 static Matcher matcher() {
1151 hasBase(ignoringParenImpCasts(
1154 isSafeArraySubscript(),
1158 ))).bind(ArraySubscrTag));
1163 bool IsRelatedToDecl,
1169 DeclUseList getClaimedVarUseSites()
const override {
1170 if (
const auto *DRE =
1183class PointerArithmeticGadget :
public WarningGadget {
1184 static constexpr const char *
const PointerArithmeticTag =
"ptrAdd";
1185 static constexpr const char *
const PointerArithmeticPointerTag =
"ptrAddPtr";
1191 : WarningGadget(
Kind::PointerArithmetic),
1193 Ptr(Result.
Nodes.getNodeAs<
Expr>(PointerArithmeticPointerTag)) {}
1195 static bool classof(
const Gadget *G) {
1196 return G->getKind() == Kind::PointerArithmetic;
1199 static Matcher matcher() {
1200 auto HasIntegerType =
anyOf(hasType(isInteger()), hasType(
enumType()));
1202 allOf(hasOperatorName(
"+"),
1204 hasLHS(HasIntegerType));
1206 allOf(
anyOf(hasOperatorName(
"+"), hasOperatorName(
"-"),
1207 hasOperatorName(
"+="), hasOperatorName(
"-=")),
1209 hasRHS(HasIntegerType));
1212 .bind(PointerArithmeticTag));
1216 bool IsRelatedToDecl,
1222 DeclUseList getClaimedVarUseSites()
const override {
1233class SpanTwoParamConstructorGadget :
public WarningGadget {
1234 static constexpr const char *
const SpanTwoParamConstructorTag =
1235 "spanTwoParamConstructor";
1240 : WarningGadget(
Kind::SpanTwoParamConstructor),
1242 SpanTwoParamConstructorTag)) {}
1244 static bool classof(
const Gadget *G) {
1245 return G->getKind() == Kind::SpanTwoParamConstructor;
1248 static Matcher matcher() {
1251 parameterCountIs(2)));
1254 unless(isSafeSpanTwoParamConstruct()))
1255 .bind(SpanTwoParamConstructorTag));
1259 return stmt(
unless(ignoreUnsafeBufferInContainer(Handler)), matcher());
1263 bool IsRelatedToDecl,
1269 DeclUseList getClaimedVarUseSites()
const override {
1272 if (
auto *DRE = dyn_cast<DeclRefExpr>(Ctor->
getArg(0))) {
1273 if (isa<VarDecl>(DRE->
getDecl()))
1284class PointerInitGadget :
public FixableGadget {
1286 static constexpr const char *
const PointerInitLHSTag =
"ptrInitLHS";
1287 static constexpr const char *
const PointerInitRHSTag =
"ptrInitRHS";
1293 : FixableGadget(
Kind::PointerInit),
1294 PtrInitLHS(Result.
Nodes.getNodeAs<
VarDecl>(PointerInitLHSTag)),
1297 static bool classof(
const Gadget *G) {
1298 return G->getKind() == Kind::PointerInit;
1301 static Matcher matcher() {
1302 auto PtrInitStmt =
declStmt(hasSingleDecl(
1303 varDecl(hasInitializer(ignoringImpCasts(
1305 .bind(PointerInitRHSTag))))
1306 .bind(PointerInitLHSTag)));
1308 return stmt(PtrInitStmt);
1311 virtual std::optional<FixItList>
1317 virtual DeclUseList getClaimedVarUseSites()
const override {
1318 return DeclUseList{PtrInitRHS};
1321 virtual std::optional<std::pair<const VarDecl *, const VarDecl *>>
1322 getStrategyImplications()
const override {
1323 return std::make_pair(PtrInitLHS, cast<VarDecl>(PtrInitRHS->
getDecl()));
1332class PtrToPtrAssignmentGadget :
public FixableGadget {
1334 static constexpr const char *
const PointerAssignLHSTag =
"ptrLHS";
1335 static constexpr const char *
const PointerAssignRHSTag =
"ptrRHS";
1341 : FixableGadget(
Kind::PtrToPtrAssignment),
1345 static bool classof(
const Gadget *G) {
1346 return G->getKind() == Kind::PtrToPtrAssignment;
1349 static Matcher matcher() {
1351 allOf(hasOperatorName(
"="),
1352 hasRHS(ignoringParenImpCasts(
1354 .bind(PointerAssignRHSTag))),
1356 .bind(PointerAssignLHSTag))));
1361 virtual std::optional<FixItList>
1365 virtual DeclUseList getClaimedVarUseSites()
const override {
1366 return DeclUseList{PtrLHS, PtrRHS};
1369 virtual std::optional<std::pair<const VarDecl *, const VarDecl *>>
1370 getStrategyImplications()
const override {
1371 return std::make_pair(cast<VarDecl>(PtrLHS->
getDecl()),
1372 cast<VarDecl>(PtrRHS->
getDecl()));
1381class CArrayToPtrAssignmentGadget :
public FixableGadget {
1383 static constexpr const char *
const PointerAssignLHSTag =
"ptrLHS";
1384 static constexpr const char *
const PointerAssignRHSTag =
"ptrRHS";
1390 : FixableGadget(
Kind::CArrayToPtrAssignment),
1394 static bool classof(
const Gadget *G) {
1395 return G->getKind() == Kind::CArrayToPtrAssignment;
1398 static Matcher matcher() {
1400 allOf(hasOperatorName(
"="),
1401 hasRHS(ignoringParenImpCasts(
1403 toSupportedVariable())
1404 .bind(PointerAssignRHSTag))),
1406 .bind(PointerAssignLHSTag))));
1411 virtual std::optional<FixItList>
1415 virtual DeclUseList getClaimedVarUseSites()
const override {
1416 return DeclUseList{PtrLHS, PtrRHS};
1419 virtual std::optional<std::pair<const VarDecl *, const VarDecl *>>
1420 getStrategyImplications()
const override {
1427class UnsafeBufferUsageAttrGadget :
public WarningGadget {
1428 constexpr static const char *
const OpTag =
"attr_expr";
1433 : WarningGadget(
Kind::UnsafeBufferUsageAttr),
1434 Op(Result.
Nodes.getNodeAs<
Expr>(OpTag)) {}
1436 static bool classof(
const Gadget *G) {
1437 return G->getKind() == Kind::UnsafeBufferUsageAttr;
1440 static Matcher matcher() {
1441 auto HasUnsafeFieldDecl =
1444 auto HasUnsafeFnDecl =
1448 memberExpr(HasUnsafeFieldDecl).bind(OpTag)));
1452 bool IsRelatedToDecl,
1458 DeclUseList getClaimedVarUseSites()
const override {
return {}; }
1464class UnsafeBufferUsageCtorAttrGadget :
public WarningGadget {
1465 constexpr static const char *
const OpTag =
"cxx_construct_expr";
1470 : WarningGadget(
Kind::UnsafeBufferUsageCtorAttr),
1473 static bool classof(
const Gadget *G) {
1474 return G->getKind() == Kind::UnsafeBufferUsageCtorAttr;
1477 static Matcher matcher() {
1478 auto HasUnsafeCtorDecl =
1481 auto HasTwoParamSpanCtorDecl = SpanTwoParamConstructorGadget::matcher();
1488 bool IsRelatedToDecl,
1494 DeclUseList getClaimedVarUseSites()
const override {
return {}; }
1501class DataInvocationGadget :
public WarningGadget {
1502 constexpr static const char *
const OpTag =
"data_invocation_expr";
1507 : WarningGadget(
Kind::DataInvocation),
1510 static bool classof(
const Gadget *G) {
1511 return G->getKind() == Kind::DataInvocation;
1514 static Matcher matcher() {
1526 bool IsRelatedToDecl,
1532 DeclUseList getClaimedVarUseSites()
const override {
return {}; }
1535class UnsafeLibcFunctionCallGadget :
public WarningGadget {
1537 const Expr *UnsafeArg =
nullptr;
1538 constexpr static const char *
const Tag =
"UnsafeLibcFunctionCall";
1540 constexpr static const char *
const UnsafeSprintfTag =
1541 "UnsafeLibcFunctionCall_sprintf";
1542 constexpr static const char *
const UnsafeSizedByTag =
1543 "UnsafeLibcFunctionCall_sized_by";
1544 constexpr static const char *
const UnsafeStringTag =
1545 "UnsafeLibcFunctionCall_string";
1546 constexpr static const char *
const UnsafeVaListTag =
1547 "UnsafeLibcFunctionCall_va_list";
1559 } WarnedFunKind = OTHERS;
1563 : WarningGadget(
Kind::UnsafeLibcFunctionCall),
1565 if (Result.Nodes.getNodeAs<
Decl>(UnsafeSprintfTag))
1566 WarnedFunKind = SPRINTF;
1567 else if (
auto *
E = Result.Nodes.getNodeAs<
Expr>(UnsafeStringTag)) {
1568 WarnedFunKind = STRING;
1570 }
else if (Result.Nodes.getNodeAs<
CallExpr>(UnsafeSizedByTag)) {
1571 WarnedFunKind = SIZED_BY;
1572 UnsafeArg =
Call->getArg(0);
1573 }
else if (Result.Nodes.getNodeAs<
Decl>(UnsafeVaListTag))
1574 WarnedFunKind = VA_LIST;
1578 return stmt(
unless(ignoreUnsafeLibcCall(Handler)),
1584 functionDecl(libc_func_matchers::isPredefinedUnsafeLibcFunc()),
1588 functionDecl(libc_func_matchers::isUnsafeVaListPrintfFunc())
1589 .bind(UnsafeVaListTag),
1592 functionDecl(libc_func_matchers::isUnsafeSprintfFunc())
1593 .bind(UnsafeSprintfTag)))),
1605 libc_func_matchers::hasUnsafeSnprintfBuffer())
1606 .bind(UnsafeSizedByTag),
1610 libc_func_matchers::hasUnsafePrintfStringArg(
1611 expr().bind(UnsafeStringTag)))));
1614 const Stmt *getBaseStmt()
const {
return Call; }
1619 bool IsRelatedToDecl,
1624 DeclUseList getClaimedVarUseSites()
const override {
return {}; }
1630class ULCArraySubscriptGadget :
public FixableGadget {
1632 static constexpr const char *
const ULCArraySubscriptTag =
1633 "ArraySubscriptUnderULC";
1638 : FixableGadget(
Kind::ULCArraySubscript),
1640 assert(Node !=
nullptr &&
"Expecting a non-null matching result");
1643 static bool classof(
const Gadget *G) {
1644 return G->getKind() == Kind::ULCArraySubscript;
1647 static Matcher matcher() {
1649 auto BaseIsArrayOrPtrDRE = hasBase(
1650 ignoringParenImpCasts(
declRefExpr(ArrayOrPtr, toSupportedVariable())));
1657 virtual std::optional<FixItList>
1661 virtual DeclUseList getClaimedVarUseSites()
const override {
1662 if (
const auto *DRE =
1663 dyn_cast<DeclRefExpr>(
Node->getBase()->IgnoreImpCasts())) {
1673class UPCStandalonePointerGadget :
public FixableGadget {
1675 static constexpr const char *
const DeclRefExprTag =
"StandalonePointer";
1680 : FixableGadget(
Kind::UPCStandalonePointer),
1682 assert(Node !=
nullptr &&
"Expecting a non-null matching result");
1685 static bool classof(
const Gadget *G) {
1686 return G->getKind() == Kind::UPCStandalonePointer;
1689 static Matcher matcher() {
1691 auto target =
expr(ignoringParenImpCasts(
1693 .bind(DeclRefExprTag)));
1697 virtual std::optional<FixItList>
1701 virtual DeclUseList getClaimedVarUseSites()
const override {
return {
Node}; }
1704class PointerDereferenceGadget :
public FixableGadget {
1705 static constexpr const char *
const BaseDeclRefExprTag =
"BaseDRE";
1706 static constexpr const char *
const OperatorTag =
"op";
1713 : FixableGadget(
Kind::PointerDereference),
1718 static bool classof(
const Gadget *G) {
1719 return G->getKind() == Kind::PointerDereference;
1722 static Matcher matcher() {
1725 hasOperatorName(
"*"),
1726 has(
expr(ignoringParenImpCasts(
1727 declRefExpr(toSupportedVariable()).bind(BaseDeclRefExprTag)))))
1733 DeclUseList getClaimedVarUseSites()
const override {
1734 return {BaseDeclRefExpr};
1737 virtual std::optional<FixItList>
1745class UPCAddressofArraySubscriptGadget :
public FixableGadget {
1747 static constexpr const char *
const UPCAddressofArraySubscriptTag =
1748 "AddressofArraySubscriptUnderUPC";
1753 : FixableGadget(
Kind::ULCArraySubscript),
1755 UPCAddressofArraySubscriptTag)) {
1756 assert(Node !=
nullptr &&
"Expecting a non-null matching result");
1759 static bool classof(
const Gadget *G) {
1760 return G->getKind() == Kind::UPCAddressofArraySubscript;
1763 static Matcher matcher() {
1766 hasOperatorName(
"&"),
1768 ignoringParenImpCasts(
declRefExpr(toSupportedVariable()))))))
1769 .bind(UPCAddressofArraySubscriptTag)))));
1772 virtual std::optional<FixItList>
1776 virtual DeclUseList getClaimedVarUseSites()
const override {
1777 const auto *ArraySubst = cast<ArraySubscriptExpr>(
Node->getSubExpr());
1779 cast<DeclRefExpr>(ArraySubst->getBase()->IgnoreParenImpCasts());
1789class DeclUseTracker {
1790 using UseSetTy = SmallSet<const DeclRefExpr *, 16>;
1791 using DefMapTy = DenseMap<const VarDecl *, const DeclStmt *>;
1794 std::unique_ptr<UseSetTy> Uses{std::make_unique<UseSetTy>()};
1798 DeclUseTracker() =
default;
1799 DeclUseTracker(
const DeclUseTracker &) =
delete;
1800 DeclUseTracker &operator=(
const DeclUseTracker &) =
delete;
1801 DeclUseTracker(DeclUseTracker &&) =
default;
1802 DeclUseTracker &operator=(DeclUseTracker &&) =
default;
1805 void discoverUse(
const DeclRefExpr *DRE) { Uses->insert(DRE); }
1809 assert(Uses->count(DRE) &&
1810 "DRE not found or claimed by multiple matchers!");
1815 bool hasUnclaimedUses(
const VarDecl *VD)
const {
1817 return any_of(*Uses, [VD](
const DeclRefExpr *DRE) {
1822 UseSetTy getUnclaimedUses(
const VarDecl *VD)
const {
1824 for (
auto use : *Uses) {
1826 ReturnSet.insert(use);
1832 void discoverDecl(
const DeclStmt *DS) {
1834 if (
const auto *VD = dyn_cast<VarDecl>(
D)) {
1846 return Defs.lookup(VD);
1855 static constexpr const char *
const UPCPreIncrementTag =
1856 "PointerPreIncrementUnderUPC";
1861 : FixableGadget(Kind::UPCPreIncrement),
1863 assert(
Node !=
nullptr &&
"Expecting a non-null matching result");
1867 return G->getKind() == Kind::UPCPreIncrement;
1877 hasUnaryOperand(
declRefExpr(toSupportedVariable())))
1878 .bind(UPCPreIncrementTag)))));
1881 virtual std::optional<FixItList>
1886 return {dyn_cast<DeclRefExpr>(
Node->getSubExpr())};
1894 static constexpr const char *
const UUCAddAssignTag =
1895 "PointerAddAssignUnderUUC";
1896 static constexpr const char *
const OffsetTag =
"Offset";
1899 const Expr *Offset =
nullptr;
1903 : FixableGadget(Kind::UUCAddAssign),
1905 Offset(Result.
Nodes.getNodeAs<
Expr>(OffsetTag)) {
1906 assert(
Node !=
nullptr &&
"Expecting a non-null matching result");
1910 return G->getKind() == Kind::UUCAddAssign;
1920 toSupportedVariable())),
1921 hasRHS(
expr().bind(OffsetTag)))
1922 .bind(UUCAddAssignTag)))));
1926 virtual std::optional<FixItList>
1931 return {dyn_cast<DeclRefExpr>(
Node->getLHS())};
1938 static constexpr const char *
const BaseDeclRefExprTag =
"BaseDRE";
1939 static constexpr const char *
const DerefOpTag =
"DerefOp";
1940 static constexpr const char *
const AddOpTag =
"AddOp";
1941 static constexpr const char *
const OffsetTag =
"Offset";
1950 : FixableGadget(Kind::DerefSimplePtrArithFixable),
1960 ignoringImpCasts(
declRefExpr(toSupportedVariable()).
1961 bind(BaseDeclRefExprTag)));
1962 auto PlusOverPtrAndInteger =
expr(
anyOf(
1970 hasOperatorName(
"*"),
1971 hasUnaryOperand(ignoringParens(PlusOverPtrAndInteger)))
1976 virtual std::optional<FixItList>
1983 return {BaseDeclRefExpr};
1990 bool EmitSuggestions, FixableGadgetList &FixableGadgets,
1991 WarningGadgetList &WarningGadgets,
1992 DeclUseTracker &Tracker) {
1995 GadgetFinderCallback(FixableGadgetList &FixableGadgets,
1996 WarningGadgetList &WarningGadgets,
1997 DeclUseTracker &Tracker)
1998 : FixableGadgets(FixableGadgets), WarningGadgets(WarningGadgets),
2007 [[maybe_unused]]
int numFound = 0;
2008#define NEXT ++numFound
2011 if (
const auto *DRE = Result.Nodes.getNodeAs<
DeclRefExpr>(
"any_dre")) {
2012 Tracker.discoverUse(DRE);
2016 if (
const auto *DS = Result.Nodes.getNodeAs<
DeclStmt>(
"any_ds")) {
2017 Tracker.discoverDecl(DS);
2024#define FIXABLE_GADGET(name) \
2025 if (Result.Nodes.getNodeAs<Stmt>(#name)) { \
2026 FixableGadgets.push_back(std::make_unique<name##Gadget>(Result)); \
2029#include "clang/Analysis/Analyses/UnsafeBufferUsageGadgets.def"
2030#define WARNING_GADGET(name) \
2031 if (Result.Nodes.getNodeAs<Stmt>(#name)) { \
2032 WarningGadgets.push_back(std::make_unique<name##Gadget>(Result)); \
2035#include "clang/Analysis/Analyses/UnsafeBufferUsageGadgets.def"
2037 assert(numFound >= 1 &&
"Gadgets not found in match result!");
2038 assert(numFound <= 1 &&
"Conflicting bind tags in gadgets!");
2041 FixableGadgetList &FixableGadgets;
2042 WarningGadgetList &WarningGadgets;
2043 DeclUseTracker &Tracker;
2047 GadgetFinderCallback CB{FixableGadgets, WarningGadgets, Tracker};
2052 forEachDescendantEvaluatedStmt(
stmt(
anyOf(
2055 allOf(x ## Gadget::matcher().bind(#x), \
2056 notInSafeBufferOptOut(&Handler)),
2058 allOf(x ## Gadget::matcher(&Handler).bind(#x), \
2059 notInSafeBufferOptOut(&Handler)),
2060#include
"clang/Analysis/Analyses/UnsafeBufferUsageGadgets.def"
2069 if (EmitSuggestions) {
2075 x ## Gadget::matcher().bind(#x),
2076#include
"clang/Analysis/Analyses/UnsafeBufferUsageGadgets.def"
2098 return N1->getBeginLoc().getRawEncoding() <
2099 N2->getBeginLoc().getRawEncoding();
2104 std::map<const VarDecl *, std::set<const WarningGadget *>,
2118 for (
auto &G : AllUnsafeOperations) {
2119 DeclUseList ClaimedVarUseSites = G->getClaimedVarUseSites();
2121 bool AssociatedWithVarDecl =
false;
2122 for (
const DeclRefExpr *DRE : ClaimedVarUseSites) {
2123 if (
const auto *VD = dyn_cast<VarDecl>(DRE->
getDecl())) {
2124 result.
byVar[VD].insert(G.get());
2125 AssociatedWithVarDecl =
true;
2129 if (!AssociatedWithVarDecl) {
2130 result.
noVar.push_back(G.get());
2138 std::map<const VarDecl *, std::set<const FixableGadget *>,
2148 for (
auto &F : AllFixableOperations) {
2149 DeclUseList DREs = F->getClaimedVarUseSites();
2152 if (
const auto *VD = dyn_cast<VarDecl>(DRE->
getDecl())) {
2153 FixablesForUnsafeVars.
byVar[VD].insert(F.get());
2157 return FixablesForUnsafeVars;
2164 std::vector<const FixItHint *> All;
2168 std::sort(All.begin(), All.end(),
2170 return SM.isBeforeInTranslationUnit(H1->RemoveRange.getBegin(),
2171 H2->RemoveRange.getBegin());
2179 Hint->RemoveRange.getBegin())) {
2191std::optional<FixItList>
2192PtrToPtrAssignmentGadget::getFixits(
const FixitStrategy &S)
const {
2193 const auto *LeftVD = cast<VarDecl>(PtrLHS->
getDecl());
2194 const auto *RightVD = cast<VarDecl>(PtrRHS->
getDecl());
2195 switch (S.lookup(LeftVD)) {
2196 case FixitStrategy::Kind::Span:
2197 if (S.lookup(RightVD) == FixitStrategy::Kind::Span)
2199 return std::nullopt;
2200 case FixitStrategy::Kind::Wontfix:
2201 return std::nullopt;
2202 case FixitStrategy::Kind::Iterator:
2203 case FixitStrategy::Kind::Array:
2204 return std::nullopt;
2205 case FixitStrategy::Kind::Vector:
2206 llvm_unreachable(
"unsupported strategies for FixableGadgets");
2208 return std::nullopt;
2215std::optional<FixItList>
2216CArrayToPtrAssignmentGadget::getFixits(
const FixitStrategy &S)
const {
2217 const auto *LeftVD = cast<VarDecl>(PtrLHS->
getDecl());
2218 const auto *RightVD = cast<VarDecl>(PtrRHS->
getDecl());
2235 if (S.lookup(LeftVD) == FixitStrategy::Kind::Span) {
2236 if (S.lookup(RightVD) == FixitStrategy::Kind::Wontfix) {
2239 }
else if (S.lookup(LeftVD) == FixitStrategy::Kind::Wontfix) {
2240 if (S.lookup(RightVD) == FixitStrategy::Kind::Array) {
2244 return std::nullopt;
2247std::optional<FixItList>
2249 const auto *LeftVD = PtrInitLHS;
2250 const auto *RightVD = cast<VarDecl>(PtrInitRHS->
getDecl());
2251 switch (S.lookup(LeftVD)) {
2252 case FixitStrategy::Kind::Span:
2253 if (S.lookup(RightVD) == FixitStrategy::Kind::Span)
2255 return std::nullopt;
2256 case FixitStrategy::Kind::Wontfix:
2257 return std::nullopt;
2258 case FixitStrategy::Kind::Iterator:
2259 case FixitStrategy::Kind::Array:
2260 return std::nullopt;
2261 case FixitStrategy::Kind::Vector:
2262 llvm_unreachable(
"unsupported strategies for FixableGadgets");
2264 return std::nullopt;
2270 if (ConstVal->isNegative())
2277std::optional<FixItList>
2278ULCArraySubscriptGadget::getFixits(
const FixitStrategy &S)
const {
2279 if (
const auto *DRE =
2280 dyn_cast<DeclRefExpr>(
Node->getBase()->IgnoreImpCasts()))
2281 if (
const auto *VD = dyn_cast<VarDecl>(DRE->
getDecl())) {
2282 switch (S.lookup(VD)) {
2283 case FixitStrategy::Kind::Span: {
2290 return std::nullopt;
2294 case FixitStrategy::Kind::Array:
2296 case FixitStrategy::Kind::Wontfix:
2297 case FixitStrategy::Kind::Iterator:
2298 case FixitStrategy::Kind::Vector:
2299 llvm_unreachable(
"unsupported strategies for FixableGadgets");
2302 return std::nullopt;
2305static std::optional<FixItList>
2308std::optional<FixItList>
2309UPCAddressofArraySubscriptGadget::getFixits(
const FixitStrategy &S)
const {
2310 auto DREs = getClaimedVarUseSites();
2311 const auto *VD = cast<VarDecl>(DREs.front()->getDecl());
2313 switch (S.lookup(VD)) {
2314 case FixitStrategy::Kind::Span:
2316 case FixitStrategy::Kind::Wontfix:
2317 case FixitStrategy::Kind::Iterator:
2318 case FixitStrategy::Kind::Array:
2319 return std::nullopt;
2320 case FixitStrategy::Kind::Vector:
2321 llvm_unreachable(
"unsupported strategies for FixableGadgets");
2323 return std::nullopt;
2328 static const char *
const EOL =
"\n";
2335 std::string
s = std::string(
"<# ");
2336 s += HintTextToUser;
2342template <
typename NodeTy>
2343static std::optional<SourceLocation>
2352 return std::nullopt;
2356template <
typename NodeTy>
2364 return std::nullopt;
2371 std::optional<SourceLocation> LastCharLoc =
getPastLoc(
E,
SM, LangOpts);
2378 return std::nullopt;
2391 return std::nullopt;
2403static std::optional<StringRef>
2412 return std::nullopt;
2413 return getRangeText({ParmIdentBeginLoc, ParmIdentEndLoc},
SM, LangOpts);
2426 bool AttrRangeOverlapping = llvm::any_of(VD->
attrs(), [&](
Attr *At) ->
bool {
2427 return !(SM.isBeforeInTranslationUnit(At->getRange().getEnd(),
2428 VD->getBeginLoc())) &&
2429 !(SM.isBeforeInTranslationUnit(VD->getEndLoc(),
2430 At->getRange().getBegin()));
2434 AttrRangeOverlapping;
2458static std::optional<std::string>
2461 std::optional<Qualifiers> *QualifiersToAppend) {
2466 "Expecting a VarDecl of type of pointer to object type");
2475 case TypeLoc::ConstantArray:
2476 case TypeLoc::IncompleteArray:
2477 case TypeLoc::VariableArray:
2478 case TypeLoc::DependentSizedArray:
2479 case TypeLoc::Decayed:
2480 assert(isa<ParmVarDecl>(VD) &&
"An array type shall not be treated as a "
2481 "pointer type unless it decays.");
2484 case TypeLoc::Pointer:
2488 return std::nullopt;
2493 return std::nullopt;
2501 return std::nullopt;
2508 if (!PteEndOfTokenLoc.
isValid())
2511 return std::nullopt;
2512 if (!
SM.isBeforeInTranslationUnit(PteEndOfTokenLoc, IdentLoc)) {
2519 return std::nullopt;
2554 std::optional<Qualifiers> Quals = std::nullopt) {
2555 const char *
const SpanOpen =
"std::span<";
2558 return SpanOpen + EltTyText.str() +
' ' + Quals->getAsString() +
'>';
2559 return SpanOpen + EltTyText.str() +
'>';
2562std::optional<FixItList>
2564 const VarDecl *VD = dyn_cast<VarDecl>(BaseDeclRefExpr->
getDecl());
2566 if (VD &&
s.lookup(VD) == FixitStrategy::Kind::Span) {
2570 if (ConstVal->isNegative())
2571 return std::nullopt;
2599 std::optional<SourceLocation> LHSLocation =
getPastLoc(LHS,
SM, LangOpts);
2601 return std::nullopt;
2606 std::optional<SourceLocation> AddOpLocation =
2608 std::optional<SourceLocation> DerefOpLocation =
2611 if (!AddOpLocation || !DerefOpLocation)
2612 return std::nullopt;
2622 return std::nullopt;
2625std::optional<FixItList>
2626PointerDereferenceGadget::getFixits(
const FixitStrategy &S)
const {
2628 switch (S.lookup(VD)) {
2629 case FixitStrategy::Kind::Span: {
2637 if (
auto LocPastOperand =
2644 case FixitStrategy::Kind::Iterator:
2645 case FixitStrategy::Kind::Array:
2646 return std::nullopt;
2647 case FixitStrategy::Kind::Vector:
2648 llvm_unreachable(
"FixitStrategy not implemented yet!");
2649 case FixitStrategy::Kind::Wontfix:
2650 llvm_unreachable(
"Invalid strategy!");
2653 return std::nullopt;
2660 std::optional<SourceLocation> EndOfOperand =
2666 return std::nullopt;
2671std::optional<FixItList>
2672UPCStandalonePointerGadget::getFixits(
const FixitStrategy &S)
const {
2673 const auto VD = cast<VarDecl>(
Node->getDecl());
2674 switch (S.lookup(VD)) {
2675 case FixitStrategy::Kind::Array:
2676 case FixitStrategy::Kind::Span: {
2681 case FixitStrategy::Kind::Wontfix:
2682 case FixitStrategy::Kind::Iterator:
2683 return std::nullopt;
2684 case FixitStrategy::Kind::Vector:
2685 llvm_unreachable(
"unsupported strategies for FixableGadgets");
2688 return std::nullopt;
2693static std::optional<FixItList>
2695 const auto *ArraySub = cast<ArraySubscriptExpr>(
Node->getSubExpr());
2696 const auto *DRE = cast<DeclRefExpr>(ArraySub->getBase()->IgnoreImpCasts());
2700 const Expr *Idx = ArraySub->getIdx();
2703 std::stringstream SS;
2704 bool IdxIsLitZero =
false;
2707 if ((*ICE).isZero())
2708 IdxIsLitZero =
true;
2709 std::optional<StringRef> DreString =
getExprText(DRE,
SM, LangOpts);
2711 return std::nullopt;
2715 SS << (*DreString).str() <<
".data()";
2717 std::optional<StringRef> IndexString =
getExprText(Idx,
SM, LangOpts);
2719 return std::nullopt;
2721 SS <<
"&" << (*DreString).str() <<
".data()"
2722 <<
"[" << (*IndexString).str() <<
"]";
2728std::optional<FixItList>
2732 if (DREs.size() != 1)
2733 return std::nullopt;
2735 if (
const VarDecl *VD = dyn_cast<VarDecl>(DREs.front()->getDecl())) {
2736 if (S.lookup(VD) == FixitStrategy::Kind::Span) {
2740 StringRef varName = VD->
getName();
2744 return std::nullopt;
2749 std::string SS = varName.str() +
" = " + varName.str() +
".subspan";
2753 std::optional<SourceLocation> AddAssignLocation =
getEndCharLoc(
2755 if (!AddAssignLocation)
2756 return std::nullopt;
2767 return std::nullopt;
2770std::optional<FixItList>
2774 if (DREs.size() != 1)
2775 return std::nullopt;
2777 if (
const VarDecl *VD = dyn_cast<VarDecl>(DREs.front()->getDecl())) {
2778 if (S.lookup(VD) == FixitStrategy::Kind::Span) {
2780 std::stringstream SS;
2781 StringRef varName = VD->
getName();
2785 SS <<
"(" << varName.data() <<
" = " << varName.data()
2786 <<
".subspan(1)).data()";
2787 std::optional<SourceLocation> PreIncLocation =
2789 if (!PreIncLocation)
2790 return std::nullopt;
2797 return std::nullopt;
2815static std::optional<FixItList>
2817 const StringRef UserFillPlaceHolder) {
2825 if (
Init->isNullPointerConstant(
2830 NPC_ValueDependentIsNotNull)) {
2831 std::optional<SourceLocation> InitLocation =
2834 return std::nullopt;
2842 std::string ExtentText = UserFillPlaceHolder.data();
2843 StringRef One =
"1";
2848 if (
auto CxxNew = dyn_cast<CXXNewExpr>(
Init->IgnoreImpCasts())) {
2853 if (
const Expr *Ext = CxxNew->getArraySize().value_or(
nullptr)) {
2854 if (!Ext->HasSideEffects(Ctx)) {
2855 std::optional<StringRef> ExtentString =
getExprText(Ext,
SM, LangOpts);
2857 return std::nullopt;
2858 ExtentText = *ExtentString;
2860 }
else if (!CxxNew->isArray())
2872 if (
auto AddrOfExpr = dyn_cast<UnaryOperator>(
Init->IgnoreImpCasts()))
2873 if (AddrOfExpr->getOpcode() == UnaryOperatorKind::UO_AddrOf &&
2874 isa_and_present<DeclRefExpr>(AddrOfExpr->getSubExpr()))
2881 std::optional<SourceLocation> LocPassInit =
getPastLoc(
Init,
SM, LangOpts);
2884 return std::nullopt;
2886 StrBuffer.append(
", ");
2887 StrBuffer.append(ExtentText);
2888 StrBuffer.append(
"}");
2894#define DEBUG_NOTE_DECL_FAIL(D, Msg) \
2895 Handler.addDebugNoteForVar((D), (D)->getBeginLoc(), \
2896 "failed to produce fixit for declaration '" + \
2897 (D)->getNameAsString() + "'" + (Msg))
2899#define DEBUG_NOTE_DECL_FAIL(D, Msg)
2905static std::optional<std::string>
2909 std::optional<Qualifiers> PteTyQualifiers = std::nullopt;
2914 return std::nullopt;
2916 std::string SpanTyText =
"std::span<";
2918 SpanTyText.append(*PteTyText);
2920 if (PteTyQualifiers) {
2921 SpanTyText.append(
" ");
2922 SpanTyText.append(PteTyQualifiers->getAsString());
2924 SpanTyText.append(
">");
2943 const StringRef UserFillPlaceHolder,
2957 std::stringstream SS;
2962 std::optional<FixItList> InitFixIts =
2966 FixIts.insert(FixIts.end(), std::make_move_iterator(InitFixIts->begin()),
2967 std::make_move_iterator(InitFixIts->end()));
2974 if (!EndLocForReplacement.
isValid()) {
3024static std::optional<FixItList>
3030 return std::nullopt;
3035 std::vector<std::string> NewTysTexts(NumParms);
3036 std::vector<bool> ParmsMask(NumParms,
false);
3037 bool AtLeastOneParmToFix =
false;
3039 for (
unsigned i = 0; i < NumParms; i++) {
3042 if (S.lookup(PVD) == FixitStrategy::Kind::Wontfix)
3044 if (S.lookup(PVD) != FixitStrategy::Kind::Span)
3046 return std::nullopt;
3048 std::optional<Qualifiers> PteTyQuals = std::nullopt;
3049 std::optional<std::string> PteTyText =
3054 return std::nullopt;
3058 ParmsMask[i] =
true;
3059 AtLeastOneParmToFix =
true;
3061 if (!AtLeastOneParmToFix)
3068 const auto NewOverloadSignatureCreator =
3069 [&
SM, &LangOpts, &NewTysTexts,
3070 &ParmsMask](
const FunctionDecl *FD) -> std::optional<std::string> {
3071 std::stringstream SS;
3079 SS << Prefix->str();
3081 return std::nullopt;
3085 for (
unsigned i = 0; i < NumParms; i++) {
3093 SS << NewTysTexts[i];
3096 SS <<
' ' << II->
getName().str();
3097 }
else if (
auto ParmTypeText =
3101 SS << ParmTypeText->str();
3103 return std::nullopt;
3104 if (i != NumParms - 1)
3113 const auto OldOverloadDefCreator =
3114 [&Handler, &
SM, &LangOpts, &NewTysTexts,
3115 &ParmsMask](
const FunctionDecl *FD) -> std::optional<std::string> {
3116 std::stringstream SS;
3124 << FDPrefix->str() <<
"{";
3126 return std::nullopt;
3129 SS <<
"return " << FunQualName->str() <<
"(";
3131 return std::nullopt;
3135 for (
unsigned i = 0; i < NumParms; i++) {
3144 return std::nullopt;
3151 if (i != NumParms - 1)
3162 std::optional<SourceLocation>
Loc =
getPastLoc(FReDecl,
SM, LangOpts);
3166 if (FReDecl->isThisDeclarationADefinition()) {
3167 assert(FReDecl == FD &&
"inconsistent function definition");
3170 if (
auto OldOverloadDef = OldOverloadDefCreator(FReDecl))
3176 if (!FReDecl->hasAttr<UnsafeBufferUsageAttr>()) {
3179 FReDecl->getBeginLoc(),
" ")));
3182 if (
auto NewOverloadDecl = NewOverloadSignatureCreator(FReDecl))
3204 std::optional<Qualifiers> PteTyQualifiers = std::nullopt;
3220 std::stringstream SS;
3223 if (PteTyQualifiers)
3232 SS <<
' ' << PVDNameText->str();
3238 const DeclUseTracker &Tracker,
3241 const DeclStmt *DS = Tracker.lookupDecl(VD);
3244 " : variables declared this way not implemented yet");
3268 const QualType &ArrayEltT = CAT->getElementType();
3269 assert(!ArrayEltT.
isNull() &&
"Trying to fix a non-array type variable!");
3278 auto MaybeElemTypeTxt =
3281 if (!MaybeElemTypeTxt)
3283 const llvm::StringRef ElemTypeTxt = MaybeElemTypeTxt->trim();
3288 while (NextTok && !NextTok->is(tok::l_square) &&
3301 if (!MaybeArraySizeTxt)
3303 const llvm::StringRef ArraySizeTxt = MaybeArraySizeTxt->trim();
3304 if (ArraySizeTxt.empty()) {
3315 std::optional<StringRef> IdentText =
3324 raw_svector_ostream OS(Replacement);
3325 OS <<
"std::array<" << ElemTypeTxt <<
", " << ArraySizeTxt <<
"> "
3326 << IdentText->str();
3336 const DeclUseTracker &Tracker,
3339 const DeclStmt *DS = Tracker.lookupDecl(VD);
3340 assert(DS &&
"Fixing non-local variables not implemented yet!");
3359 const DeclUseTracker &Tracker,
ASTContext &Ctx,
3361 if (
const auto *PVD = dyn_cast<ParmVarDecl>(VD)) {
3362 auto *FD = dyn_cast<clang::FunctionDecl>(PVD->getDeclContext());
3363 if (!FD || FD !=
D) {
3374 if (FD->isMain() || FD->isConstexpr() ||
3375 FD->getTemplatedKind() != FunctionDecl::TemplatedKind::TK_NonTemplate ||
3378 isa<CXXMethodDecl>(FD) ||
3380 (FD->hasBody() && isa<CXXTryStmt>(FD->getBody())) ||
3381 FD->isOverloadedOperator()) {
3388 case FixitStrategy::Kind::Span: {
3390 if (
const auto *PVD = dyn_cast<ParmVarDecl>(VD))
3399 case FixitStrategy::Kind::Array: {
3406 case FixitStrategy::Kind::Iterator:
3407 case FixitStrategy::Kind::Vector:
3408 llvm_unreachable(
"FixitStrategy not implemented yet!");
3409 case FixitStrategy::Kind::Wontfix:
3410 llvm_unreachable(
"Invalid strategy!");
3412 llvm_unreachable(
"Unknown strategy!");
3420 return llvm::any_of(FixIts, [](
const FixItHint &Hint) {
3431 return isa<ParmVarDecl>(VD) &&
3439 std::map<const VarDecl *, FixItList> &FixItsForVariable,
3444 for (
const auto &[VD, Ignore] : FixItsForVariable) {
3446 if (llvm::any_of(Grp,
3447 [&FixItsForVariable](
const VarDecl *GrpMember) ->
bool {
3448 return !FixItsForVariable.count(GrpMember);
3453 ToErase.push_back(
Member);
3456 for (
auto *VarToErase : ToErase)
3457 FixItsForVariable.erase(VarToErase);
3468 std::map<const VarDecl *, FixItList> &FixItsForVariable ,
3472 FixItList FixItsSharedByParms{};
3474 std::optional<FixItList> OverloadFixes =
3477 if (OverloadFixes) {
3478 FixItsSharedByParms.append(*OverloadFixes);
3484 FixItsForVariable.erase(
Member);
3486 return FixItsSharedByParms;
3490static std::map<const VarDecl *, FixItList>
3499 std::map<const VarDecl *, FixItList> FixItsForVariable;
3504 for (
const auto &[VD, Fixables] : FixablesForAllVars.
byVar) {
3505 FixItsForVariable[VD] =
3506 fixVariable(VD, S.lookup(VD),
D, Tracker, Ctx, Handler);
3509 if (FixItsForVariable[VD].empty()) {
3510 FixItsForVariable.erase(VD);
3513 for (
const auto &F : Fixables) {
3514 std::optional<FixItList> Fixits = F->getFixits(S);
3517 FixItsForVariable[VD].insert(FixItsForVariable[VD].end(),
3518 Fixits->begin(), Fixits->end());
3523 VD, F->getSourceLoc(),
3524 (
"gadget '" + F->getDebugName() +
"' refused to produce a fix")
3527 FixItsForVariable.erase(VD);
3546 FixItList FixItsSharedByParms{};
3548 if (
auto *FD = dyn_cast<FunctionDecl>(
D))
3550 FixItsForVariable, VarGrpMgr, FD, S, Ctx, Handler);
3554 std::map<const VarDecl *, FixItList> FinalFixItsForVariable{
3557 for (
auto &[Var, Ignore] : FixItsForVariable) {
3558 bool AnyParm =
false;
3559 const auto VarGroupForVD = VarGrpMgr.
getGroupOfVar(Var, &AnyParm);
3561 for (
const VarDecl *GrpMate : VarGroupForVD) {
3564 if (FixItsForVariable.count(GrpMate))
3565 FinalFixItsForVariable[Var].append(FixItsForVariable[GrpMate]);
3569 assert(!FixItsSharedByParms.empty() &&
3570 "Should not try to fix a parameter that does not belong to a "
3572 FinalFixItsForVariable[Var].append(FixItsSharedByParms);
3579 for (
auto Iter = FinalFixItsForVariable.begin();
3580 Iter != FinalFixItsForVariable.end();)
3583 Iter = FinalFixItsForVariable.erase(
Iter);
3586 return FinalFixItsForVariable;
3589template <
typename VarDeclIterTy>
3593 for (
const VarDecl *VD : UnsafeVars) {
3595 S.set(VD, FixitStrategy::Kind::Array);
3597 S.set(VD, FixitStrategy::Kind::Span);
3604 const std::vector<VarGrpTy> Groups;
3605 const std::map<const VarDecl *, unsigned> &VarGrpMap;
3606 const llvm::SetVector<const VarDecl *> &GrpsUnionForParms;
3610 const std::vector<VarGrpTy> &Groups,
3611 const std::map<const VarDecl *, unsigned> &VarGrpMap,
3612 const llvm::SetVector<const VarDecl *> &GrpsUnionForParms)
3613 : Groups(Groups), VarGrpMap(VarGrpMap),
3614 GrpsUnionForParms(GrpsUnionForParms) {}
3617 if (GrpsUnionForParms.contains(Var)) {
3620 return GrpsUnionForParms.getArrayRef();
3625 auto It = VarGrpMap.find(Var);
3627 if (It == VarGrpMap.end())
3629 return Groups[It->second];
3633 return GrpsUnionForParms.getArrayRef();
3638 WarningGadgetList WarningGadgets, DeclUseTracker Tracker,
3640 if (!EmitSuggestions) {
3644 for (
const auto &G : WarningGadgets) {
3645 G->handleUnsafeOperation(Handler,
false,
3651 assert(FixableGadgets.size() == 0 &&
3652 "Fixable gadgets found but suggestions not requested!");
3658 if (!WarningGadgets.empty()) {
3662 for (
const auto &G : FixableGadgets) {
3663 for (
const auto *DRE : G->getClaimedVarUseSites()) {
3664 Tracker.claimUse(DRE);
3680 if (WarningGadgets.empty())
3688 std::map<const VarDecl *, FixItList> FixItsForVariableGroup;
3691 for (
auto it = FixablesForAllVars.
byVar.cbegin();
3692 it != FixablesForAllVars.
byVar.cend();) {
3694 if ((!it->first->isLocalVarDecl() && !isa<ParmVarDecl>(it->first))) {
3697 (
"failed to produce fixit for '" +
3698 it->first->getNameAsString() +
3699 "' : neither local nor a parameter"));
3701 it = FixablesForAllVars.
byVar.erase(it);
3702 }
else if (it->first->getType().getCanonicalType()->isReferenceType()) {
3705 (
"failed to produce fixit for '" +
3706 it->first->getNameAsString() +
3707 "' : has a reference type"));
3709 it = FixablesForAllVars.
byVar.erase(it);
3710 }
else if (Tracker.hasUnclaimedUses(it->first)) {
3711 it = FixablesForAllVars.
byVar.erase(it);
3712 }
else if (it->first->isInitCapture()) {
3715 (
"failed to produce fixit for '" +
3716 it->first->getNameAsString() +
3717 "' : init capture"));
3719 it = FixablesForAllVars.
byVar.erase(it);
3726 for (
const auto &it : UnsafeOps.
byVar) {
3727 const VarDecl *
const UnsafeVD = it.first;
3728 auto UnclaimedDREs = Tracker.getUnclaimedUses(UnsafeVD);
3729 if (UnclaimedDREs.empty())
3733 std::string UnclaimedUseTrace =
3738 (
"failed to produce fixit for '" + UnfixedVDName +
3739 "' : has an unclaimed use\nThe unclaimed DRE trace: " +
3740 UnclaimedUseTrace));
3746 using DepMapTy = DenseMap<const VarDecl *, llvm::SetVector<const VarDecl *>>;
3747 DepMapTy DependenciesMap{};
3748 DepMapTy PtrAssignmentGraph{};
3750 for (
auto it : FixablesForAllVars.
byVar) {
3751 for (
const FixableGadget *fixable : it.second) {
3752 std::optional<std::pair<const VarDecl *, const VarDecl *>> ImplPair =
3753 fixable->getStrategyImplications();
3755 std::pair<const VarDecl *, const VarDecl *> Impl = std::move(*ImplPair);
3756 PtrAssignmentGraph[Impl.first].insert(Impl.second);
3778 std::set<const VarDecl *> VisitedVarsDirected{};
3779 for (
const auto &[Var, ignore] : UnsafeOps.
byVar) {
3780 if (VisitedVarsDirected.find(Var) == VisitedVarsDirected.end()) {
3782 std::queue<const VarDecl *> QueueDirected{};
3783 QueueDirected.push(Var);
3784 while (!QueueDirected.empty()) {
3785 const VarDecl *CurrentVar = QueueDirected.front();
3786 QueueDirected.pop();
3787 VisitedVarsDirected.insert(CurrentVar);
3788 auto AdjacentNodes = PtrAssignmentGraph[CurrentVar];
3789 for (
const VarDecl *Adj : AdjacentNodes) {
3790 if (VisitedVarsDirected.find(Adj) == VisitedVarsDirected.end()) {
3791 QueueDirected.push(Adj);
3793 DependenciesMap[Var].insert(Adj);
3794 DependenciesMap[Adj].insert(Var);
3801 std::vector<VarGrpTy> Groups;
3805 std::map<const VarDecl *, unsigned> VarGrpMap;
3807 llvm::SetVector<const VarDecl *>
3812 std::set<const VarDecl *> VisitedVars{};
3813 for (
const auto &[Var, ignore] : UnsafeOps.
byVar) {
3814 if (VisitedVars.find(Var) == VisitedVars.end()) {
3815 VarGrpTy &VarGroup = Groups.emplace_back();
3816 std::queue<const VarDecl *> Queue{};
3819 while (!Queue.empty()) {
3820 const VarDecl *CurrentVar = Queue.front();
3822 VisitedVars.insert(CurrentVar);
3823 VarGroup.push_back(CurrentVar);
3824 auto AdjacentNodes = DependenciesMap[CurrentVar];
3825 for (
const VarDecl *Adj : AdjacentNodes) {
3826 if (VisitedVars.find(Adj) == VisitedVars.end()) {
3832 bool HasParm =
false;
3833 unsigned GrpIdx = Groups.size() - 1;
3835 for (
const VarDecl *
V : VarGroup) {
3836 VarGrpMap[
V] = GrpIdx;
3841 GrpsUnionForParms.insert(VarGroup.begin(), VarGroup.end());
3863 for (
auto I = FixablesForAllVars.
byVar.begin();
3864 I != FixablesForAllVars.
byVar.end();) {
3866 if (!VisitedVars.count((*I).first)) {
3868 I = FixablesForAllVars.
byVar.erase(I);
3876 VisitedVars, [&FixablesForAllVars](
const VarDecl *
V) {
3878 return FixablesForAllVars.
byVar.count(
V);
3882 if (isa<NamedDecl>(
D))
3885 FixItsForVariableGroup =
3887 Tracker, Handler, VarGrpMgr);
3889 for (
const auto &G : UnsafeOps.
noVar) {
3890 G->handleUnsafeOperation(Handler,
false,
3894 for (
const auto &[VD, WarningGadgets] : UnsafeOps.
byVar) {
3895 auto FixItsIt = FixItsForVariableGroup.find(VD);
3897 FixItsIt != FixItsForVariableGroup.end()
3898 ? std::move(FixItsIt->second)
3901 for (
const auto &G : WarningGadgets) {
3902 G->handleUnsafeOperation(Handler,
true,
3910 bool EmitSuggestions) {
3919 if (
const auto *FD = dyn_cast<FunctionDecl>(
D)) {
3923 if (
const auto *MD = dyn_cast<CXXMethodDecl>(
D)) {
3924 if (MD->getParent()->isLambda() && MD->getParent()->isLocalClass())
3929 if (FReDecl->isExternC()) {
3932 EmitSuggestions =
false;
3937 Stmts.push_back(FD->getBody());
3939 if (
const auto *ID = dyn_cast<CXXConstructorDecl>(
D)) {
3941 Stmts.push_back(CI->getInit());
3944 }
else if (isa<BlockDecl>(
D) || isa<ObjCMethodDecl>(
D)) {
3948 assert(!Stmts.empty());
3950 FixableGadgetList FixableGadgets;
3951 WarningGadgetList WarningGadgets;
3952 DeclUseTracker Tracker;
3953 for (
Stmt *S : Stmts) {
3955 WarningGadgets, Tracker);
3957 applyGadgets(
D, std::move(FixableGadgets), std::move(WarningGadgets),
3958 std::move(Tracker), Handler, EmitSuggestions);
Defines the clang::ASTContext interface.
BoundNodesTreeBuilder Nodes
#define AST_MATCHER(Type, DefineMatcher)
AST_MATCHER(Type, DefineMatcher) { ... } defines a zero parameter function named DefineMatcher() that...
#define AST_MATCHER_P(Type, DefineMatcher, ParamType, Param)
AST_MATCHER_P(Type, DefineMatcher, ParamType, Param) { ... } defines a single-parameter function name...
static Decl::Kind getKind(const Decl *D)
static const Decl * getCanonicalDecl(const Decl *D)
llvm::MachO::Target Target
Defines the clang::Preprocessor interface.
static bool hasAttr(const Decl *D, bool IgnoreImplicitAttr)
Defines the clang::SourceLocation class and associated facilities.
static QualType getPointeeType(const MemRegion *R)
C Language Family Type Representation.
static std::string getUserFillPlaceHolder(StringRef HintTextToUser="placeholder")
static FixItList fixVariableWithSpan(const VarDecl *VD, const DeclUseTracker &Tracker, ASTContext &Ctx, UnsafeBufferUsageHandler &Handler)
static std::optional< FixItList > fixUPCAddressofArraySubscriptWithSpan(const UnaryOperator *Node)
static std::optional< StringRef > getExprText(const Expr *E, const SourceManager &SM, const LangOptions &LangOpts)
static WarningGadgetSets groupWarningGadgetsByVar(const WarningGadgetList &AllUnsafeOperations)
static StringRef getEndOfLine()
static std::optional< FixItList > FixVarInitializerWithSpan(const Expr *Init, ASTContext &Ctx, const StringRef UserFillPlaceHolder)
static std::optional< SourceLocation > getEndCharLoc(const NodeTy *Node, const SourceManager &SM, const LangOptions &LangOpts)
static FixItList fixVariableWithArray(const VarDecl *VD, const DeclUseTracker &Tracker, const ASTContext &Ctx, UnsafeBufferUsageHandler &Handler)
static std::string getSpanTypeText(StringRef EltTyText, std::optional< Qualifiers > Quals=std::nullopt)
static SourceRange getSourceRangeToTokenEnd(const Decl *D, const SourceManager &SM, const LangOptions &LangOpts)
static FixItList fixLocalVarDeclWithSpan(const VarDecl *D, ASTContext &Ctx, const StringRef UserFillPlaceHolder, UnsafeBufferUsageHandler &Handler)
static std::optional< FixItList > createDataFixit(const ASTContext &Ctx, const DeclRefExpr *DRE)
static FixItList createFunctionOverloadsForParms(std::map< const VarDecl *, FixItList > &FixItsForVariable, const VariableGroupsManager &VarGrpMgr, const FunctionDecl *FD, const FixitStrategy &S, ASTContext &Ctx, UnsafeBufferUsageHandler &Handler)
#define FIXABLE_GADGET(name)
#define WARNING_OPTIONAL_GADGET(x)
static FixItList fixVarDeclWithArray(const VarDecl *D, const ASTContext &Ctx, UnsafeBufferUsageHandler &Handler)
void applyGadgets(const Decl *D, FixableGadgetList FixableGadgets, WarningGadgetList WarningGadgets, DeclUseTracker Tracker, UnsafeBufferUsageHandler &Handler, bool EmitSuggestions)
#define WARNING_GADGET(name)
static FixItList fixVariable(const VarDecl *VD, FixitStrategy::Kind K, const Decl *D, const DeclUseTracker &Tracker, ASTContext &Ctx, UnsafeBufferUsageHandler &Handler)
static FixItList fixParamWithSpan(const ParmVarDecl *PVD, const ASTContext &Ctx, UnsafeBufferUsageHandler &Handler)
static std::optional< StringRef > getRangeText(SourceRange SR, const SourceManager &SM, const LangOptions &LangOpts)
static FixitStrategy getNaiveStrategy(llvm::iterator_range< VarDeclIterTy > UnsafeVars)
static std::optional< std::string > createSpanTypeForVarDecl(const VarDecl *VD, const ASTContext &Ctx)
static SourceLocation getVarDeclIdentifierLoc(const VarDecl *VD)
static std::optional< SourceLocation > getPastLoc(const NodeTy *Node, const SourceManager &SM, const LangOptions &LangOpts)
static bool hasConflictingOverload(const FunctionDecl *FD)
static std::optional< StringRef > getVarDeclIdentifierText(const VarDecl *VD, const SourceManager &SM, const LangOptions &LangOpts)
static std::optional< std::string > getPointeeTypeText(const VarDecl *VD, const SourceManager &SM, const LangOptions &LangOpts, std::optional< Qualifiers > *QualifiersToAppend)
static bool isNonNegativeIntegerExpr(const Expr *Expr, const VarDecl *VD, const ASTContext &Ctx)
static bool overlapWithMacro(const FixItList &FixIts)
static bool hasUnsupportedSpecifiers(const VarDecl *VD, const SourceManager &SM)
#define DEBUG_NOTE_DECL_FAIL(D, Msg)
static void findGadgets(const Stmt *S, ASTContext &Ctx, const UnsafeBufferUsageHandler &Handler, bool EmitSuggestions, FixableGadgetList &FixableGadgets, WarningGadgetList &WarningGadgets, DeclUseTracker &Tracker)
Scan the function and return a list of gadgets found with provided kits.
static std::map< const VarDecl *, FixItList > getFixIts(FixableGadgetSets &FixablesForAllVars, const FixitStrategy &S, ASTContext &Ctx, const Decl *D, const DeclUseTracker &Tracker, UnsafeBufferUsageHandler &Handler, const VariableGroupsManager &VarGrpMgr)
static std::optional< FixItList > createOverloadsForFixedParams(const FixitStrategy &S, const FunctionDecl *FD, const ASTContext &Ctx, UnsafeBufferUsageHandler &Handler)
static void eraseVarsForUnfixableGroupMates(std::map< const VarDecl *, FixItList > &FixItsForVariable, const VariableGroupsManager &VarGrpMgr)
static FixableGadgetSets groupFixablesByVar(FixableGadgetList &&AllFixableOperations)
static bool isParameterOf(const VarDecl *VD, const Decl *D)
static std::optional< StringRef > getFunNameText(const FunctionDecl *FD, const SourceManager &SM, const LangOptions &LangOpts)
__device__ __2f16 float __ockl_bool s
virtual std::optional< FixItList > getFixits(const FixitStrategy &s) const final
DerefSimplePtrArithFixableGadget(const MatchFinder::MatchResult &Result)
SourceLocation getSourceLoc() const override
virtual DeclUseList getClaimedVarUseSites() const final
virtual std::optional< FixItList > getFixits(const FixitStrategy &S) const override
SourceLocation getSourceLoc() const override
virtual DeclUseList getClaimedVarUseSites() const override
static bool classof(const Gadget *G)
UPCPreIncrementGadget(const MatchFinder::MatchResult &Result)
static bool classof(const Gadget *G)
UUCAddAssignGadget(const MatchFinder::MatchResult &Result)
virtual std::optional< FixItList > getFixits(const FixitStrategy &S) const override
virtual DeclUseList getClaimedVarUseSites() const override
SourceLocation getSourceLoc() const override
VariableGroupsManagerImpl(const std::vector< VarGrpTy > &Groups, const std::map< const VarDecl *, unsigned > &VarGrpMap, const llvm::SetVector< const VarDecl * > &GrpsUnionForParms)
VarGrpRef getGroupOfVar(const VarDecl *Var, bool *HasParm) const override
Returns the set of variables (including Var) that need to be fixed together in one step.
VarGrpRef getGroupOfParms() const override
Returns the non-empty group of variables that include parameters of the analyzing function,...
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
SourceManager & getSourceManager()
const ConstantArrayType * getAsConstantArrayType(QualType T) const
DynTypedNodeList getParents(const NodeT &Node)
Forwards to get node parents from the ParentMapContext.
QualType getFILEType() const
Retrieve the C FILE type.
const LangOptions & getLangOpts() const
const TargetInfo & getTargetInfo() const
ArraySubscriptExpr - [C99 6.5.2.1] Array Subscripting.
SourceLocation getBeginLoc() const LLVM_READONLY
Attr - This represents one attribute.
A builtin binary operation expression such as "x + y" or "x <= y".
SourceLocation getBeginLoc() const LLVM_READONLY
static StringRef getOpcodeStr(Opcode Op)
getOpcodeStr - Turn an Opcode enum value into the punctuation char it corresponds to,...
Represents a call to a C++ constructor.
Expr * getArg(unsigned Arg)
Return the specified argument.
SourceLocation getBeginLoc() const LLVM_READONLY
Represents a C++ base or member initializer.
A use of a default initializer in a constructor or in aggregate initialization.
Represents a static or instance method of a struct/union/class.
Represents a C++11 noexcept expression (C++ [expr.unary.noexcept]).
Represents a C++ struct/union/class.
CXXRecordDecl * getCanonicalDecl() override
Retrieves the "canonical" declaration of the given declaration.
A C++ typeid expression (C++ [expr.typeid]), which gets the type_info that corresponds to the supplie...
CallExpr - Represents a function call (C99 6.5.2.2, C++ [expr.call]).
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
static const char * getCastKindName(CastKind CK)
Represents a character-granular source range.
static CharSourceRange getCharRange(SourceRange R)
SourceLocation getEnd() const
ConstStmtVisitor - This class implements a simple visitor for Stmt subclasses.
bool isSingleResult() const
lookup_result lookup(DeclarationName Name) const
lookup - Find the declarations (if any) with the given Name in this context.
A reference to a declared variable, function, enum, etc.
SourceLocation getBeginLoc() const LLVM_READONLY
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.
Decl - This represents one declaration (or definition), e.g.
bool isInStdNamespace() const
SourceLocation getEndLoc() const LLVM_READONLY
ASTContext & getASTContext() const LLVM_READONLY
bool isImplicit() const
isImplicit - Indicates whether the declaration was implicitly generated by the implementation.
virtual Stmt * getBody() const
getBody - If this Decl represents a declaration for a body of code, such as a function or method defi...
SourceLocation getLocation() const
DeclContext * getDeclContext()
SourceLocation getBeginLoc() const LLVM_READONLY
virtual SourceRange getSourceRange() const LLVM_READONLY
Source range that this declaration covers.
NestedNameSpecifier * getQualifier() const
Retrieve the nested-name-specifier that qualifies the name of this declaration, if it was present in ...
SourceLocation getBeginLoc() const LLVM_READONLY
NestedNameSpecifierLoc getQualifierLoc() const
Retrieve the nested-name-specifier (with source-location information) that qualifies the name of this...
TypeSourceInfo * getTypeSourceInfo() const
Container for either a single DynTypedNode or for an ArrayRef to DynTypedNode.
const DynTypedNode * begin() const
A dynamically typed AST node container.
SourceRange getSourceRange() const
For nodes which represent textual entities in the source code, return their SourceRange.
const T * get() const
Retrieve the stored node as type T.
static DynTypedNode create(const T &Node)
Creates a DynTypedNode from Node.
Recursive AST visitor that supports extension via dynamic dispatch.
virtual bool TraverseDecl(Decl *D)
Recursively visit a declaration, by dispatching to Traverse*Decl() based on the argument's dynamic ty...
virtual bool TraverseStmt(Stmt *S)
Recursively visit a statement or expression, by dispatching to Traverse*() based on the argument's dy...
bool ShouldVisitImplicitCode
Whether this visitor should recurse into implicit code, e.g.
bool ShouldVisitTemplateInstantiations
Whether this visitor should recurse into template instantiations.
ExplicitCastExpr - An explicit cast written in the source code.
This represents one expression.
Expr * IgnoreParenImpCasts() LLVM_READONLY
Skip past any parentheses and implicit casts which might surround this expression until reaching a fi...
Expr * IgnoreImplicit() LLVM_READONLY
Skip past any implicit AST nodes which might surround this expression until reaching a fixed point.
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
NullPointerConstantValueDependence
Enumeration used to describe how isNullPointerConstant() should cope with value-dependent expressions...
std::optional< llvm::APSInt > getIntegerConstantExpr(const ASTContext &Ctx, SourceLocation *Loc=nullptr) const
isIntegerConstantExpr - Return the value if this expression is a valid integer constant expression.
Annotates a diagnostic with some code that should be inserted, removed, or replaced to fix the proble...
CharSourceRange RemoveRange
Code that should be replaced to correct the error.
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.
Represents a function declaration or definition.
const ParmVarDecl * getParamDecl(unsigned i) const
Stmt * getBody(const FunctionDecl *&Definition) const
Retrieve the body (definition) of the function.
param_iterator param_begin()
unsigned getNumParams() const
Return the number of parameters this function must have based on its FunctionType.
DeclarationNameInfo getNameInfo() const
Represents a C11 generic selection.
One of these records is kept for each identifier that is lexed.
StringRef getName() const
Return the actual identifier string.
ImplicitCastExpr - Allows us to explicitly represent implicit type conversions, which have no direct ...
Keeps track of the various options that can be enabled, which controls the dialect of C or C++ that i...
static StringRef getSourceText(CharSourceRange Range, const SourceManager &SM, const LangOptions &LangOpts, bool *Invalid=nullptr)
Returns a string for the source that the range encompasses.
static bool isAtEndOfMacroExpansion(SourceLocation loc, const SourceManager &SM, const LangOptions &LangOpts, SourceLocation *MacroEnd=nullptr)
Returns true if the given MacroID location points at the last token of the macro expansion.
static unsigned MeasureTokenLength(SourceLocation Loc, const SourceManager &SM, const LangOptions &LangOpts)
MeasureTokenLength - Relex the token at the specified location and return its length in bytes in the ...
static std::optional< Token > findNextToken(SourceLocation Loc, const SourceManager &SM, const LangOptions &LangOpts)
Finds the token that comes right after the given location.
static SourceLocation getLocForEndOfToken(SourceLocation Loc, unsigned Offset, const SourceManager &SM, const LangOptions &LangOpts)
Computes the source location just past the end of the token at this source location.
IdentifierInfo * getIdentifier() const
Get the identifier that names this declaration, if there is one.
StringRef getName() const
Get the name of identifier for this declaration as a StringRef.
DeclarationName getDeclName() const
Get the actual, stored name of the declaration, which may be a special name.
std::string getNameAsString() const
Get a human-readable name for the declaration, even if it is one of the special kinds of names (C++ c...
SourceLocation getBeginLoc() const
Retrieve the location of the beginning of this nested-name-specifier.
Represents a parameter to a function.
bool hasDefaultArg() const
Determines whether this parameter has a default argument, either parsed or not.
SourceRange getSourceRange() const override LLVM_READONLY
Source range that this declaration covers.
Wrapper for source info for pointers.
PointerType - C99 6.7.5.1 - Pointer Declarators.
A (possibly-)qualified type.
bool hasQualifiers() const
Determine whether this type has any qualifiers.
bool isNull() const
Return true if this QualType doesn't point to a type yet.
Qualifiers getQualifiers() const
Retrieve the set of qualifiers applied to this type.
QualType getCanonicalType() const
bool isConstQualified() const
Determine whether this type is const-qualified.
std::string getAsString() const
redecl_range redecls() const
Returns an iterator range for all the redeclarations of the same decl.
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.
This class handles loading and caching of source files into memory.
A trivial tuple used to represent a source range.
SourceLocation getEnd() const
SourceLocation getBegin() const
Stmt - This represents one statement.
SourceLocation getEndLoc() const LLVM_READONLY
StmtClass getStmtClass() const
SourceLocation getBeginLoc() const LLVM_READONLY
Exposes information about the current target.
Base wrapper for a particular "section" of type source info.
UnqualTypeLoc getUnqualifiedLoc() const
Skips past any qualifiers, if this is qualified.
TypeLoc getNextTypeLoc() const
Get the next TypeLoc pointed by this TypeLoc, e.g for "int*" the TypeLoc is a PointerLoc and next Typ...
T castAs() const
Convert to the specified TypeLoc type, asserting that this TypeLoc is of the desired type.
SourceRange getSourceRange() const LLVM_READONLY
Get the full source range.
TypeLocClass getTypeLocClass() const
SourceLocation getEndLoc() const
Get the end source location.
SourceLocation getBeginLoc() const
Get the begin source location.
TypeLoc getTypeLoc() const
Return the TypeLoc wrapper for the type source info.
bool isFunctionPointerType() const
bool isPointerType() const
bool isIntegerType() const
isIntegerType() does not include complex integers (a GCC extension).
const T * castAs() const
Member-template castAs<specific type>.
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
bool isUnsignedIntegerType() const
Return true if this is an integer type that is unsigned, according to C99 6.2.5p6 [which returns true...
UnaryExprOrTypeTraitExpr - expression with either a type or (unevaluated) expression operand.
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
SourceLocation getOperatorLoc() const
getOperatorLoc - Return the location of the operator.
Expr * getSubExpr() const
SourceLocation getBeginLoc() const LLVM_READONLY
static StringRef getOpcodeStr(Opcode Op)
getOpcodeStr - Turn an Opcode enum value into the punctuation char it corresponds to,...
The interface that lets the caller handle unsafe buffer usage analysis results by overriding this cla...
void addDebugNoteForVar(const VarDecl *VD, SourceLocation Loc, std::string Text)
virtual std::string getUnsafeBufferUsageAttributeTextAt(SourceLocation Loc, StringRef WSSuffix="") const =0
virtual void handleUnsafeOperation(const Stmt *Operation, bool IsRelatedToDecl, ASTContext &Ctx)=0
Invoked when an unsafe operation over raw pointers is found.
virtual void handleUnsafeVariableGroup(const VarDecl *Variable, const VariableGroupsManager &VarGrpMgr, FixItList &&Fixes, const Decl *D, const FixitStrategy &VarTargetTypes)=0
Invoked when a fix is suggested against a variable.
virtual void handleUnsafeOperationInContainer(const Stmt *Operation, bool IsRelatedToDecl, ASTContext &Ctx)=0
Invoked when an unsafe operation with a std container is found.
virtual void handleUnsafeLibcCall(const CallExpr *Call, unsigned PrintfInfo, ASTContext &Ctx, const Expr *UnsafeArg=nullptr)=0
Invoked when a call to an unsafe libc function is found.
Represents a variable declaration or definition.
bool isConstexpr() const
Whether this variable is (C++11) constexpr.
VarDecl * getCanonicalDecl() override
Retrieves the "canonical" declaration of the given declaration.
bool isInlineSpecified() const
bool hasConstantInitialization() const
Determine whether this variable has constant initialization.
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.
virtual VarGrpRef getGroupOfVar(const VarDecl *Var, bool *HasParm=nullptr) const =0
Returns the set of variables (including Var) that need to be fixed together in one step.
virtual VarGrpRef getGroupOfParms() const =0
Returns the non-empty group of variables that include parameters of the analyzing function,...
bool findMatch(const DynTypedNode &DynNode)
bool TraverseCXXNoexceptExpr(CXXNoexceptExpr *Node) override
bool TraverseDecltypeTypeLoc(DecltypeTypeLoc Node) override
bool TraverseCXXDefaultInitExpr(CXXDefaultInitExpr *Node) override
bool TraverseUnaryExprOrTypeTraitExpr(UnaryExprOrTypeTraitExpr *Node) override
bool TraverseGenericSelectionExpr(GenericSelectionExpr *Node) override
bool TraverseTypeOfExprTypeLoc(TypeOfExprTypeLoc Node) override
bool TraverseStmt(Stmt *Node) override
Recursively visit a statement or expression, by dispatching to Traverse*() based on the argument's dy...
bool TraverseCXXTypeidExpr(CXXTypeidExpr *Node) override
MatchDescendantVisitor(const internal::DynTypedMatcher *Matcher, internal::ASTMatchFinder *Finder, internal::BoundNodesTreeBuilder *Builder, internal::ASTMatchFinder::BindKind Bind, const bool ignoreUnevaluatedContext)
bool TraverseDecl(Decl *Node) override
Recursively visit a declaration, by dispatching to Traverse*Decl() based on the argument's dynamic ty...
Called when the Match registered for it was successfully found in the AST.
virtual void run(const MatchResult &Result)=0
Called on every match by the MatchFinder.
A class to allow finding matches over the Clang AST.
void addMatcher(const DeclarationMatcher &NodeMatch, MatchCallback *Action)
Adds a matcher to execute when running over the AST.
void match(const T &Node, ASTContext &Context)
Calls the registered callbacks on all matches on the given Node.
static bool isNullTermPointer(const Expr *Ptr)
static bool hasUnsafeFormatOrSArg(const CallExpr *Call, const Expr *&UnsafeArg, const unsigned FmtArgIdx, ASTContext &Ctx, bool isKprintf=false)
const internal::VariadicDynCastAllOfMatcher< Decl, VarDecl > varDecl
Matches variable declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, DeclRefExpr > declRefExpr
Matches expressions that refer to declarations.
const AstTypeMatcher< EnumType > enumType
Matches enum types.
static auto isInUnspecifiedLvalueContext(internal::Matcher< Expr > innerMatcher)
const internal::VariadicOperatorMatcherFunc< 1, 1 > unless
Matches if the provided matcher does not match.
const internal::VariadicDynCastAllOfMatcher< Stmt, ImplicitCastExpr > implicitCastExpr
Matches the implicit cast nodes of Clang's AST.
const internal::VariadicDynCastAllOfMatcher< Stmt, StringLiteral > stringLiteral
Matches string literals (also matches wide string literals).
const AstTypeMatcher< PointerType > pointerType
Matches pointer types, but does not match Objective-C object pointer types.
const internal::VariadicDynCastAllOfMatcher< Decl, BindingDecl > bindingDecl
Matches binding declarations Example matches foo and bar (matcher = bindingDecl()
internal::Matcher< NamedDecl > hasName(StringRef Name)
Matches NamedDecl nodes that have the specified name.
const internal::VariadicDynCastAllOfMatcher< Stmt, CallExpr > callExpr
Matches call expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, CompoundStmt > compoundStmt
Matches compound statements.
static auto hasArrayType()
const internal::ArgumentAdaptingMatcherFunc< internal::ForEachMatcher > forEach
Matches AST nodes that have child AST nodes that match the provided matcher.
const AstTypeMatcher< ArrayType > arrayType
Matches all kinds of arrays.
const internal::VariadicDynCastAllOfMatcher< Stmt, UnaryOperator > unaryOperator
Matches unary operator expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, ArraySubscriptExpr > arraySubscriptExpr
Matches array subscript expressions.
static internal::Matcher< Stmt > isInUnspecifiedUntypedContext(internal::Matcher< Stmt > InnerMatcher)
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXMemberCallExpr > cxxMemberCallExpr
Matches member call expressions.
const internal::VariadicDynCastAllOfMatcher< Decl, CXXConstructorDecl > cxxConstructorDecl
Matches C++ constructor declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, ArrayInitIndexExpr > arrayInitIndexExpr
The arrayInitIndexExpr consists of two subexpressions: a common expression (the source array) that is...
const internal::VariadicDynCastAllOfMatcher< Stmt, BinaryOperator > binaryOperator
Matches binary operator expressions.
const internal::VariadicDynCastAllOfMatcher< Stmt, ParenExpr > parenExpr
Matches parentheses used in expressions.
const internal::ArgumentAdaptingMatcherFunc< internal::HasMatcher > has
Matches AST nodes that have child AST nodes that match the provided matcher.
const internal::VariadicDynCastAllOfMatcher< Stmt, ExplicitCastExpr > explicitCastExpr
Matches explicit cast expressions.
internal::PolymorphicMatcher< internal::ValueEqualsMatcher, void(internal::AllNodeBaseTypes), ValueT > equals(const ValueT &Value)
Matches literals that are equal to the given value of type ValueT.
const AstTypeMatcher< ConstantArrayType > constantArrayType
Matches C arrays with a specified constant size.
const internal::VariadicOperatorMatcherFunc< 2, std::numeric_limits< unsigned >::max()> eachOf
Matches if any of the given matchers matches.
const internal::VariadicDynCastAllOfMatcher< Stmt, CXXConstructExpr > cxxConstructExpr
Matches constructor call expressions (including implicit ones).
const internal::VariadicDynCastAllOfMatcher< Decl, FieldDecl > fieldDecl
Matches field declarations.
static internal::Matcher< Stmt > isInUnspecifiedPointerContext(internal::Matcher< Stmt > InnerMatcher)
const internal::VariadicOperatorMatcherFunc< 2, std::numeric_limits< unsigned >::max()> allOf
Matches if all given matchers match.
const internal::VariadicDynCastAllOfMatcher< Decl, FunctionDecl > functionDecl
Matches function declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, MemberExpr > memberExpr
Matches member expressions.
static auto hasPointerType()
const internal::VariadicDynCastAllOfMatcher< Stmt, IntegerLiteral > integerLiteral
Matches integer literals of all sizes / encodings, e.g.
internal::PolymorphicMatcher< internal::HasDeclarationMatcher, void(internal::HasDeclarationSupportedTypes), internal::Matcher< Decl > > hasDeclaration(const internal::Matcher< Decl > &InnerMatcher)
Matches a node if the declaration associated with that node matches the given matcher.
const internal::VariadicDynCastAllOfMatcher< Stmt, DeclStmt > declStmt
Matches declaration statements.
const internal::VariadicAllOfMatcher< Stmt > stmt
Matches statements.
const internal::VariadicDynCastAllOfMatcher< Stmt, Expr > expr
Matches expressions.
const internal::VariadicOperatorMatcherFunc< 2, std::numeric_limits< unsigned >::max()> anyOf
Matches if any of the given matchers matches.
const internal::VariadicFunction< internal::PolymorphicMatcher< internal::HasAnyOperatorNameMatcher, AST_POLYMORPHIC_SUPPORTED_TYPES(BinaryOperator, CXXOperatorCallExpr, CXXRewrittenBinaryOperator, UnaryOperator), std::vector< std::string > >, StringRef, internal::hasAnyOperatorNameFunc > hasAnyOperatorName
Matches operator expressions (binary or unary) that have any of the specified names.
const internal::VariadicDynCastAllOfMatcher< Decl, CXXMethodDecl > cxxMethodDecl
Matches method declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, CastExpr > castExpr
Matches any cast nodes of Clang's AST.
const internal::VariadicDynCastAllOfMatcher< Stmt, IfStmt > ifStmt
Matches if statements.
bool anyConflict(const llvm::SmallVectorImpl< FixItHint > &FixIts, const SourceManager &SM)
bool Call(InterpState &S, CodePtr OpPC, const Function *Func, uint32_t VarArgSize)
The JSON file list parser is used to communicate input to InstallAPI.
void checkUnsafeBufferUsage(const Decl *D, UnsafeBufferUsageHandler &Handler, bool EmitSuggestions)
std::vector< const VarDecl * > VarGrpTy
const FunctionProtoType * T
Diagnostic wrappers for TextAPI types for error reporting.
bool operator()(const NodeTy *N1, const NodeTy *N2) const
std::map< const VarDecl *, std::set< const FixableGadget * >, CompareNode< VarDecl > > byVar
std::map< const VarDecl *, std::set< const WarningGadget * >, CompareNode< VarDecl > > byVar
llvm::SmallVector< const WarningGadget *, 16 > noVar
SourceLocation getBeginLoc() const
getBeginLoc - Retrieve the location of the first token.
SourceLocation getEndLoc() const LLVM_READONLY
EvalResult is a struct with detailed info about an evaluated expression.
APValue Val
Val - This is the value the expression can be folded to.
Wraps an identifier and optional source location for the identifier.
Contains all information for a given match.
StringRef matchLibcName(StringRef Name)
StringRef matchLibcNameOrBuiltinChk(StringRef Name)
StringRef matchName(StringRef FunName, bool isBuiltin)