10#include "clang/AST/ASTContext.h"
11#include "clang/ASTMatchers/ASTMatchFinder.h"
12#include "clang/Frontend/CompilerInstance.h"
13#include "clang/Lex/Lexer.h"
14#include "clang/Lex/PPCallbacks.h"
15#include "clang/Lex/Preprocessor.h"
38static Preprocessor *
PP;
48 if (
const auto *DestVAT =
50 return DestVAT->getSizeExpr();
52 if (
const auto *DestVD = Result.Nodes.getNodeAs<VarDecl>(
DestVarDeclName))
53 if (
const TypeLoc DestTL = DestVD->getTypeSourceInfo()->getTypeLoc())
54 if (
const auto DestCTL = DestTL.getAs<ConstantArrayTypeLoc>())
55 return DestCTL.getSizeExpr();
63 const MatchFinder::MatchResult &Result) {
67 E = E->IgnoreImpCasts();
69 if (
const auto *LengthDRE = dyn_cast<DeclRefExpr>(E))
70 if (
const auto *LengthVD = dyn_cast<VarDecl>(LengthDRE->getDecl()))
71 if (!isa<ParmVarDecl>(LengthVD))
72 if (
const Expr *LengthInit = LengthVD->getInit();
73 LengthInit && !LengthInit->isValueDependent()) {
74 Expr::EvalResult Length;
75 if (LengthInit->EvaluateAsInt(Length, *Result.Context))
76 return Length.Val.getInt().getZExtValue();
79 if (
const auto *LengthIL = dyn_cast<IntegerLiteral>(E))
80 return LengthIL->getValue().getZExtValue();
82 if (
const auto *StrDRE = dyn_cast<DeclRefExpr>(E))
83 if (
const auto *StrVD = dyn_cast<VarDecl>(StrDRE->getDecl()))
84 if (
const Expr *StrInit = StrVD->getInit())
85 if (
const auto *StrSL =
86 dyn_cast<StringLiteral>(StrInit->IgnoreImpCasts()))
87 return StrSL->getLength();
89 if (
const auto *SrcSL = dyn_cast<StringLiteral>(E))
90 return SrcSL->getLength();
99 return getLength(DestCapacityExpr, Result);
105static const CallExpr *
getStrlenExpr(
const MatchFinder::MatchResult &Result) {
106 if (
const auto *StrlenExpr =
108 if (
const Decl *D = StrlenExpr->getCalleeDecl())
109 if (
const FunctionDecl *FD = D->getAsFunction())
110 if (
const IdentifierInfo *II = FD->getIdentifier())
111 if (II->isStr(
"strlen") || II->isStr(
"wcslen"))
123 if (
const int Length =
127 if (
const int Length =
133 if (
const Expr *Arg = StrlenCE->getArg(0)->IgnoreImpCasts())
134 if (
const int ArgLength =
getLength(Arg, Result))
142 const MatchFinder::MatchResult &Result) {
146 return Lexer::getSourceText(
147 CharSourceRange::getTokenRange(E->getSourceRange()),
148 *Result.SourceManager, Result.Context->getLangOpts(),
nullptr);
153 const MatchFinder::MatchResult &Result) {
154 return Lexer::getLocForEndOfToken(E->getEndLoc(), 0, *Result.SourceManager,
155 Result.Context->getLangOpts());
165static bool isInjectUL(
const MatchFinder::MatchResult &Result) {
177 const StringRef DestCapacityExprStr =
179 const StringRef LengthExprStr =
182 return !DestCapacityExprStr.empty() && !LengthExprStr.empty() &&
183 DestCapacityExprStr.contains(LengthExprStr);
188 if (
const auto *DestDRE = Result.Nodes.getNodeAs<DeclRefExpr>(
DestExprName))
189 if (
const auto *SrcDRE = Result.Nodes.getNodeAs<DeclRefExpr>(
SrcExprName))
190 return DestDRE->getDecl()->getCanonicalDecl() ==
191 SrcDRE->getDecl()->getCanonicalDecl();
198 const auto *DestExpr =
199 Result.Nodes.getNodeAs<CXXMemberCallExpr>(
DestExprName);
200 const auto *SrcExpr = Result.Nodes.getNodeAs<CXXMemberCallExpr>(
SrcExprName);
201 const auto *LengthExpr =
204 StringRef DestStr =
"", SrcStr =
"", LengthStr =
"";
206 if (
const CXXMethodDecl *DestMD = DestExpr->getMethodDecl())
207 DestStr = DestMD->getName();
210 if (
const CXXMethodDecl *SrcMD = SrcExpr->getMethodDecl())
211 SrcStr = SrcMD->getName();
214 if (
const CXXMethodDecl *LengthMD = LengthExpr->getMethodDecl())
215 LengthStr = LengthMD->getName();
217 return (LengthStr ==
"length" || LengthStr ==
"size") &&
218 (SrcStr ==
"data" || DestStr ==
"data");
230 const int SrcLength =
233 if (GivenLength != 0 && SrcLength != 0 && GivenLength == SrcLength)
236 if (
const auto *LengthExpr = Result.Nodes.getNodeAs<Expr>(
LengthExprName))
237 if (isa<BinaryOperator>(LengthExpr->IgnoreParenImpCasts()))
242 if (
const auto *ArgDRE =
243 dyn_cast<DeclRefExpr>(StrlenCE->getArg(0)->IgnoreImpCasts()))
244 if (
const auto *SrcVD = Result.Nodes.getNodeAs<VarDecl>(
SrcVarDeclName))
245 return dyn_cast<VarDecl>(ArgDRE->getDecl()) == SrcVD;
265 const int DestCapacity =
getLength(DestCapacityExpr, Result);
268 if (GivenLength != 0 && DestCapacity != 0)
273 const StringRef DestCapacityExprStr =
exprToStr(DestCapacityExpr, Result);
274 if (DestCapacityExprStr.contains(
"+1") || DestCapacityExprStr.contains(
"+ 1"))
295 const MatchFinder::MatchResult &Result,
296 DiagnosticBuilder &Diag) {
297 LengthExpr = LengthExpr->IgnoreParenImpCasts();
300 bool IsMacroDefinition =
false;
301 const StringRef LengthExprStr =
exprToStr(LengthExpr, Result);
302 Preprocessor::macro_iterator It =
PP->macro_begin();
303 while (It !=
PP->macro_end() && !IsMacroDefinition) {
304 if (It->first->getName() == LengthExprStr)
305 IsMacroDefinition =
true;
311 if (!IsMacroDefinition) {
312 if (
const auto *LengthIL = dyn_cast<IntegerLiteral>(LengthExpr)) {
313 const uint64_t NewLength =
314 LengthIL->getValue().getZExtValue() +
317 const auto NewLengthFix = FixItHint::CreateReplacement(
318 LengthIL->getSourceRange(),
319 (Twine(NewLength) + (
isInjectUL(Result) ?
"UL" :
"")).str());
320 Diag << NewLengthFix;
326 const auto *BO = dyn_cast<BinaryOperator>(LengthExpr);
327 if (BO && BO->getOpcode() == BO_Add &&
329 const Expr *LhsExpr = BO->getLHS()->IgnoreImpCasts();
330 const Expr *RhsExpr = BO->getRHS()->IgnoreImpCasts();
332 if (
const auto *LhsIL = dyn_cast<IntegerLiteral>(LhsExpr)) {
333 if (LhsIL->getValue().getZExtValue() == 1) {
334 Diag << FixItHint::CreateRemoval(
335 {LhsIL->getBeginLoc(),
336 RhsExpr->getBeginLoc().getLocWithOffset(-1)});
341 if (
const auto *RhsIL = dyn_cast<IntegerLiteral>(RhsExpr)) {
342 if (RhsIL->getValue().getZExtValue() == 1) {
343 Diag << FixItHint::CreateRemoval(
344 {LhsExpr->getEndLoc().getLocWithOffset(1), RhsIL->getEndLoc()});
351 const bool NeedInnerParen = BO && BO->getOpcode() != BO_Add;
354 Diag << FixItHint::CreateInsertion(LengthExpr->getBeginLoc(),
"(");
356 SmallString<8> Injection;
363 Diag << FixItHint::CreateInsertion(
exprLocEnd(LengthExpr, Result), Injection);
367 const MatchFinder::MatchResult &Result,
368 DiagnosticBuilder &Diag) {
369 const auto *LengthExpr = Result.Nodes.getNodeAs<Expr>(
LengthExprName);
374 const MatchFinder::MatchResult &Result,
375 DiagnosticBuilder &Diag) {
376 const auto *FunctionExpr = Result.Nodes.getNodeAs<CallExpr>(
FunctionExprName);
383 DiagnosticBuilder &Diag) {
384 const auto *Dest = Result.Nodes.getNodeAs<Expr>(
DestExprName);
388 const std::string TempTyStr = Dest->getType().getAsString();
389 const StringRef TyStr = TempTyStr;
390 if (TyStr.starts_with(
"char") || TyStr.starts_with(
"wchar_t"))
393 Diag << FixItHint::CreateInsertion(Dest->getBeginLoc(),
"(char *)");
400 DiagnosticBuilder &Diag) {
409static void removeArg(
int ArgPos,
const MatchFinder::MatchResult &Result,
410 DiagnosticBuilder &Diag) {
415 const auto *FunctionExpr = Result.Nodes.getNodeAs<CallExpr>(
FunctionExprName);
416 const Expr *ArgToRemove = FunctionExpr->getArg(ArgPos);
417 const Expr *LHSArg = FunctionExpr->getArg(ArgPos - 1);
418 const auto RemoveArgFix = FixItHint::CreateRemoval(
420 exprLocEnd(ArgToRemove, Result).getLocWithOffset(-1)));
421 Diag << RemoveArgFix;
425 const MatchFinder::MatchResult &Result,
426 DiagnosticBuilder &Diag) {
427 const auto *FunctionExpr = Result.Nodes.getNodeAs<CallExpr>(
FunctionExprName);
428 const int FuncNameLength =
429 FunctionExpr->getDirectCallee()->getIdentifier()->getLength();
430 const SourceRange FuncNameRange(
431 FunctionExpr->getBeginLoc(),
432 FunctionExpr->getBeginLoc().getLocWithOffset(FuncNameLength - 1));
434 const auto FuncNameFix =
435 FixItHint::CreateReplacement(FuncNameRange, NewFuncName);
440 const MatchFinder::MatchResult &Result,
441 DiagnosticBuilder &Diag) {
442 SmallString<10> NewFuncName;
443 NewFuncName = (Name[0] !=
'w') ?
"str" :
"wcs";
444 NewFuncName += IsCopy ?
"cpy" :
"ncpy";
445 NewFuncName += IsSafe ?
"_s" :
"";
450 const MatchFinder::MatchResult &Result,
451 DiagnosticBuilder &Diag) {
452 const auto *FunctionExpr = Result.Nodes.getNodeAs<CallExpr>(
FunctionExprName);
453 SmallString<64> NewSecondArg;
456 NewSecondArg = Twine(IsOverflows ? DestLength + 1 : DestLength).str();
460 (IsOverflows ? (!
isInjectUL(Result) ?
" + 1" :
" + 1UL") :
""))
464 NewSecondArg +=
", ";
465 const auto InsertNewArgFix = FixItHint::CreateInsertion(
466 FunctionExpr->getArg(1)->getBeginLoc(), NewSecondArg);
467 Diag << InsertNewArgFix;
471 const MatchFinder::MatchResult &Result,
472 DiagnosticBuilder &Diag) {
473 const auto *FunctionExpr = Result.Nodes.getNodeAs<CallExpr>(
FunctionExprName);
474 const int FuncLocStartColumn = Result.SourceManager->getPresumedColumnNumber(
475 FunctionExpr->getBeginLoc());
476 const SourceRange SpaceRange(
477 FunctionExpr->getBeginLoc().getLocWithOffset(-FuncLocStartColumn + 1),
478 FunctionExpr->getBeginLoc());
479 const StringRef SpaceBeforeStmtStr = Lexer::getSourceText(
480 CharSourceRange::getCharRange(SpaceRange), *Result.SourceManager,
481 Result.Context->getLangOpts(),
nullptr);
483 SmallString<128> NewAddNullTermExprStr;
484 NewAddNullTermExprStr =
485 (Twine(
'\n') + SpaceBeforeStmtStr +
488 "] = " + ((Name[0] !=
'w') ? R
"('\0';)" : R"(L'\0';)"))
491 const auto AddNullTerminatorExprFix = FixItHint::CreateInsertion(
492 exprLocEnd(FunctionExpr, Result).getLocWithOffset(1),
493 NewAddNullTermExprStr);
494 Diag << AddNullTerminatorExprFix;
504 WantToUseSafeFunctions(Options.get(
"WantToUseSafeFunctions", true)) {}
508 Options.store(Opts,
"WantToUseSafeFunctions", WantToUseSafeFunctions);
512 const SourceManager &SM, Preprocessor *Pp, Preprocessor *ModuleExpanderPP) {
517AST_MATCHER_P(Expr, hasDefinition, ast_matchers::internal::Matcher<Expr>,
519 const Expr *SimpleNode = &Node;
520 SimpleNode = SimpleNode->IgnoreParenImpCasts();
522 if (InnerMatcher.matches(*SimpleNode, Finder, Builder))
525 auto DREHasInit = ignoringImpCasts(
526 declRefExpr(to(varDecl(hasInitializer(ignoringImpCasts(InnerMatcher))))));
528 if (DREHasInit.matches(*SimpleNode, Finder, Builder))
531 const char *
const VarDeclName =
"variable-declaration";
532 auto DREHasDefinition = ignoringImpCasts(declRefExpr(
533 to(varDecl().bind(VarDeclName)),
534 hasAncestor(compoundStmt(hasDescendant(binaryOperator(
535 hasLHS(declRefExpr(to(varDecl(equalsBoundNode(VarDeclName))))),
536 hasRHS(ignoringImpCasts(InnerMatcher))))))));
538 if (DREHasDefinition.matches(*SimpleNode, Finder, Builder))
547 binaryOperator(hasOperatorName(
"+"),
548 hasEitherOperand(ignoringParenImpCasts(integerLiteral())));
551 binaryOperator(hasOperatorName(
"-"),
552 hasEitherOperand(ignoringParenImpCasts(integerLiteral())));
554 auto HasIncOp = anyOf(ignoringImpCasts(IncOp), hasDescendant(IncOp));
555 auto HasDecOp = anyOf(ignoringImpCasts(DecOp), hasDescendant(DecOp));
557 auto Container = ignoringImpCasts(cxxMemberCallExpr(hasDescendant(declRefExpr(
558 hasType(hasUnqualifiedDesugaredType(recordType(hasDeclaration(recordDecl(
559 hasAnyName(
"::std::vector",
"::std::list",
"::std::deque"))))))))));
561 auto StringTy = type(hasUnqualifiedDesugaredType(recordType(
562 hasDeclaration(cxxRecordDecl(hasName(
"::std::basic_string"))))));
565 anyOf(hasType(StringTy), hasType(qualType(pointsTo(StringTy))));
567 auto CharTyArray = hasType(qualType(hasCanonicalType(
570 auto CharTyPointer = hasType(
571 qualType(hasCanonicalType(pointerType(pointee(isAnyCharacter())))));
573 auto AnyOfCharTy = anyOf(CharTyArray, CharTyPointer);
581 callExpr(callee(functionDecl(hasAnyName(
"::strlen",
"::wcslen"))))
586 cxxMemberCallExpr(on(expr(AnyOfStringTy).bind(
"Foo")),
587 has(memberExpr(member(hasAnyName(
"size",
"length")))))
591 auto SizeOfCharExpr = unaryExprOrTypeTraitExpr(has(expr(AnyOfCharTy)));
594 ignoringImpCasts(anyOf(Strlen, SizeOrLength, hasDescendant(Strlen),
595 hasDescendant(SizeOrLength)));
599 ignoringImpCasts(declRefExpr(to(varDecl(hasInitializer(WrongLength)))));
601 auto AnyOfCallOrDREWithoutInc = anyOf(DREWithoutInc, WrongLength);
604 auto CallExprReturnWithoutInc = ignoringImpCasts(callExpr(callee(functionDecl(
605 hasBody(has(returnStmt(hasReturnValue(AnyOfCallOrDREWithoutInc))))))));
608 auto DREHasReturnWithoutInc = ignoringImpCasts(
609 declRefExpr(to(varDecl(hasInitializer(CallExprReturnWithoutInc)))));
611 auto AnyOfWrongLengthInit =
612 anyOf(WrongLength, AnyOfCallOrDREWithoutInc, CallExprReturnWithoutInc,
613 DREHasReturnWithoutInc);
621 auto SizeExpr = anyOf(SizeOfCharExpr, integerLiteral(equals(1)));
623 auto MallocLengthExpr = allOf(
625 hasAnyName(
"::alloca",
"::calloc",
"malloc",
"realloc"))),
629 auto DestMalloc = anyOf(callExpr(MallocLengthExpr),
630 hasDescendant(callExpr(MallocLengthExpr)));
633 auto DestCXXNewExpr = ignoringImpCasts(
636 auto AnyOfDestInit = anyOf(DestMalloc, DestCXXNewExpr);
639 auto DestArrayTyDecl = declRefExpr(
644 auto DestUnknownDecl =
649 auto AnyOfDestDecl = ignoringImpCasts(
650 anyOf(allOf(hasDefinition(anyOf(AnyOfDestInit, DestArrayTyDecl,
651 hasDescendant(DestArrayTyDecl))),
653 anyOf(DestUnknownDecl, hasDescendant(DestUnknownDecl))));
655 auto NullTerminatorExpr = binaryOperator(
656 hasLHS(anyOf(hasDescendant(declRefExpr(to(varDecl(
658 hasDescendant(declRefExpr(
660 hasRHS(ignoringImpCasts(
661 anyOf(characterLiteral(equals(0U)), integerLiteral(equals(0))))));
665 anyOf(hasAncestor(cxxMemberCallExpr().bind(
SrcExprName)),
669 ignoringImpCasts(anyOf(stringLiteral().bind(
SrcExprName),
671 SrcDecl, hasDescendant(SrcDecl)));
678 CallContext(StringRef Name, std::optional<unsigned> DestinationPos,
679 std::optional<unsigned> SourcePos,
unsigned LengthPos,
681 : Name(Name), DestinationPos(DestinationPos), SourcePos(SourcePos),
682 LengthPos(LengthPos), WithIncrease(WithIncrease) {};
685 std::optional<unsigned> DestinationPos;
686 std::optional<unsigned> SourcePos;
691 auto MatchDestination = [=](CallContext CC) {
692 return hasArgument(*CC.DestinationPos,
694 unless(hasAncestor(compoundStmt(
695 hasDescendant(NullTerminatorExpr)))),
699 auto MatchSource = [=](CallContext CC) {
700 return hasArgument(*CC.SourcePos, AnyOfSrcDecl);
703 auto MatchGivenLength = [=](CallContext CC) {
708 allOf(unless(hasDefinition(SizeOfCharExpr)),
709 allOf(CC.WithIncrease
710 ? ignoringImpCasts(hasDefinition(HasIncOp))
712 allOf(unless(hasDefinition(HasIncOp)),
713 hasDefinition(optionally(
714 binaryOperator().bind(
716 AnyOfWrongLengthInit))),
720 auto MatchCall = [=](CallContext CC) {
721 const std::string CharHandlerFuncName =
"::" + CC.Name.str();
724 const std::string WcharHandlerFuncName =
725 "::" + (CC.Name.starts_with(
"mem") ?
"w" + CC.Name.str()
726 :
"wcs" + CC.Name.substr(3).str());
728 return allOf(callee(functionDecl(
729 hasAnyName(CharHandlerFuncName, WcharHandlerFuncName))),
730 MatchGivenLength(CC));
733 auto Match = [=](CallContext CC) {
734 if (CC.DestinationPos && CC.SourcePos)
735 return allOf(MatchCall(CC), MatchDestination(CC), MatchSource(CC));
737 if (CC.DestinationPos && !CC.SourcePos)
738 return allOf(MatchCall(CC), MatchDestination(CC),
739 hasArgument(*CC.DestinationPos, anything()));
741 if (!CC.DestinationPos && CC.SourcePos)
742 return allOf(MatchCall(CC), MatchSource(CC),
743 hasArgument(*CC.SourcePos, anything()));
745 llvm_unreachable(
"Unhandled match");
749 auto Memcpy = Match({
"memcpy", 0, 1, 2,
false});
752 auto MemcpyS = Match({
"memcpy_s", 0, 2, 3,
false});
755 auto Memchr = Match({
"memchr", std::nullopt, 0, 2,
false});
758 auto Memmove = Match({
"memmove", 0, 1, 2,
false});
761 auto MemmoveS = Match({
"memmove_s", 0, 2, 3,
false});
764 auto StrncmpRHS = Match({
"strncmp", std::nullopt, 1, 2,
true});
765 auto StrncmpLHS = Match({
"strncmp", std::nullopt, 0, 2,
true});
768 auto Strxfrm = Match({
"strxfrm", 0, 1, 2,
false});
771 auto StrerrorS = Match({
"strerror_s", 0, std::nullopt, 1,
false});
773 auto AnyOfMatchers = anyOf(Memcpy, MemcpyS, Memmove, MemmoveS, StrncmpRHS,
774 StrncmpLHS, Strxfrm, StrerrorS);
781 unless(hasAncestor(castExpr(unless(implicitCastExpr())))))
785 castExpr(allOf(unless(implicitCastExpr()),
792 const MatchFinder::MatchResult &Result) {
793 const auto *FunctionExpr = Result.Nodes.getNodeAs<CallExpr>(
FunctionExprName);
794 if (FunctionExpr->getBeginLoc().isMacroID())
797 if (WantToUseSafeFunctions &&
PP->isMacroDefined(
"__STDC_LIB_EXT1__")) {
798 std::optional<bool> AreSafeFunctionsWanted;
800 Preprocessor::macro_iterator It =
PP->macro_begin();
801 while (It !=
PP->macro_end() && !AreSafeFunctionsWanted) {
802 if (It->first->getName() ==
"__STDC_WANT_LIB_EXT1__") {
803 const auto *MI =
PP->getMacroInfo(It->first);
806 const auto &T = MI->tokens().back();
807 if (T.isLiteral() && T.getLiteralData()) {
808 const StringRef ValueStr =
809 StringRef(T.getLiteralData(), T.getLength());
810 llvm::APInt IntValue;
811 ValueStr.getAsInteger(10, IntValue);
812 AreSafeFunctionsWanted = IntValue.getZExtValue();
820 if (AreSafeFunctionsWanted)
821 UseSafeFunctions = *AreSafeFunctionsWanted;
824 const StringRef Name = FunctionExpr->getDirectCallee()->getName();
825 if (Name.starts_with(
"mem") || Name.starts_with(
"wmem"))
826 memoryHandlerFunctionFix(Name, Result);
827 else if (Name ==
"strerror_s")
828 strerrorSFix(Result);
829 else if (Name.ends_with(
"ncmp"))
830 ncmpFix(Name, Result);
831 else if (Name.ends_with(
"xfrm"))
832 xfrmFix(Name, Result);
835void NotNullTerminatedResultCheck::memoryHandlerFunctionFix(
836 StringRef Name,
const MatchFinder::MatchResult &Result) {
840 if (Name.ends_with(
"chr")) {
841 memchrFix(Name, Result);
845 if ((Name.contains(
"cpy") || Name.contains(
"move")) &&
851 "the result from calling '%0' is not null-terminated")
854 if (Name.ends_with(
"cpy")) {
855 memcpyFix(Name, Result, Diag);
856 }
else if (Name.ends_with(
"cpy_s")) {
857 memcpySFix(Name, Result, Diag);
858 }
else if (Name.ends_with(
"move")) {
859 memmoveFix(Name, Result, Diag);
860 }
else if (Name.ends_with(
"move_s")) {
866void NotNullTerminatedResultCheck::memcpyFix(
867 StringRef Name,
const MatchFinder::MatchResult &Result,
868 DiagnosticBuilder &Diag) {
875 const bool IsSafe = UseSafeFunctions && IsOverflows &&
isKnownDest(Result) &&
878 const bool IsDestLengthNotRequired =
879 IsSafe && getLangOpts().CPlusPlus &&
884 if (IsSafe && !IsDestLengthNotRequired)
890 if (!IsCopy && !IsSafe)
894void NotNullTerminatedResultCheck::memcpySFix(
895 StringRef Name,
const MatchFinder::MatchResult &Result,
896 DiagnosticBuilder &Diag) {
900 const bool RemoveDestLength =
901 getLangOpts().CPlusPlus &&
904 const bool IsSafe = IsOverflows;
908 if (!IsSafe || (IsSafe && RemoveDestLength))
916 if (!IsCopy && !IsSafe)
920void NotNullTerminatedResultCheck::memchrFix(
921 StringRef Name,
const MatchFinder::MatchResult &Result) {
922 const auto *FunctionExpr = Result.Nodes.getNodeAs<CallExpr>(
FunctionExprName);
923 if (
const auto *GivenCL = dyn_cast<CharacterLiteral>(FunctionExpr->getArg(1)))
924 if (GivenCL->getValue() != 0)
927 auto Diag = diag(FunctionExpr->getArg(2)->IgnoreParenCasts()->getBeginLoc(),
928 "the length is too short to include the null terminator");
930 if (
const auto *CastExpr = Result.Nodes.getNodeAs<Expr>(
CastExprName)) {
931 const auto CastRemoveFix = FixItHint::CreateRemoval(
932 SourceRange(CastExpr->getBeginLoc(),
933 FunctionExpr->getBeginLoc().getLocWithOffset(-1)));
934 Diag << CastRemoveFix;
937 const StringRef NewFuncName = (Name[0] !=
'w') ?
"strchr" :
"wcschr";
942void NotNullTerminatedResultCheck::memmoveFix(
943 StringRef Name,
const MatchFinder::MatchResult &Result,
944 DiagnosticBuilder &Diag)
const {
948 renameFunc((Name[0] !=
'w') ?
"memmove_s" :
"wmemmove_s", Result, Diag);
955void NotNullTerminatedResultCheck::strerrorSFix(
956 const MatchFinder::MatchResult &Result) {
959 "the result from calling 'strerror_s' is not null-terminated and "
960 "missing the last character of the error message");
966void NotNullTerminatedResultCheck::ncmpFix(
967 StringRef Name,
const MatchFinder::MatchResult &Result) {
968 const auto *FunctionExpr = Result.Nodes.getNodeAs<CallExpr>(
FunctionExprName);
969 const Expr *FirstArgExpr = FunctionExpr->getArg(0)->IgnoreImpCasts();
970 const Expr *SecondArgExpr = FunctionExpr->getArg(1)->IgnoreImpCasts();
971 bool IsLengthTooLong =
false;
974 const Expr *LengthExprArg = StrlenExpr->getArg(0);
975 const StringRef FirstExprStr =
exprToStr(FirstArgExpr, Result).trim();
976 const StringRef SecondExprStr =
exprToStr(SecondArgExpr, Result).trim();
977 const StringRef LengthArgStr =
exprToStr(LengthExprArg, Result).trim();
979 LengthArgStr == FirstExprStr || LengthArgStr == SecondExprStr;
981 const int SrcLength =
984 if (SrcLength != 0 && GivenLength != 0)
985 IsLengthTooLong = GivenLength > SrcLength;
991 auto Diag = diag(FunctionExpr->getArg(2)->IgnoreParenCasts()->getBeginLoc(),
992 "comparison length is too long and might lead to a "
998void NotNullTerminatedResultCheck::xfrmFix(
999 StringRef Name,
const MatchFinder::MatchResult &Result) {
1005 "the result from calling '%0' is not null-terminated")
Every ClangTidyCheck reports errors through a DiagnosticsEngine provided by this context.
void registerPPCallbacks(const SourceManager &SM, Preprocessor *PP, Preprocessor *ModuleExpanderPP) override
void registerMatchers(ast_matchers::MatchFinder *Finder) override
void storeOptions(ClangTidyOptions::OptionMap &Opts) override
NotNullTerminatedResultCheck(StringRef Name, ClangTidyContext *Context)
void check(const ast_matchers::MatchFinder::MatchResult &Result) override
constexpr llvm::StringLiteral CastExprName
static bool isStringDataAndLength(const MatchFinder::MatchResult &Result)
static bool isCorrectGivenLength(const MatchFinder::MatchResult &Result)
static const CallExpr * getStrlenExpr(const MatchFinder::MatchResult &Result)
static void renameFunc(StringRef NewFuncName, const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
static bool isKnownDest(const MatchFinder::MatchResult &Result)
static bool isDestAndSrcEquals(const MatchFinder::MatchResult &Result)
static bool isDestExprFix(const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
constexpr llvm::StringLiteral FunctionExprName
static SourceLocation exprLocEnd(const Expr *E, const MatchFinder::MatchResult &Result)
constexpr llvm::StringLiteral UnknownLengthName
static int getGivenLength(const MatchFinder::MatchResult &Result)
constexpr llvm::StringLiteral WrongLengthExprName
constexpr llvm::StringLiteral LengthExprName
static bool isDestBasedOnGivenLength(const MatchFinder::MatchResult &Result)
constexpr llvm::StringLiteral DestMallocExprName
constexpr llvm::StringLiteral DestArrayTyName
constexpr llvm::StringLiteral UnknownDestName
static const Expr * getDestCapacityExpr(const MatchFinder::MatchResult &Result)
static bool isGivenLengthEqualToSrcLength(const MatchFinder::MatchResult &Result)
constexpr llvm::StringLiteral SrcVarDeclName
static int getDestCapacity(const MatchFinder::MatchResult &Result)
static bool isDestCapacityOverflows(const MatchFinder::MatchResult &Result)
static bool isInjectUL(const MatchFinder::MatchResult &Result)
static bool isFixedGivenLengthAndUnknownSrc(const MatchFinder::MatchResult &Result)
static void lengthExprHandle(const Expr *LengthExpr, LengthHandleKind LengthHandle, const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
static void lengthArgHandle(LengthHandleKind LengthHandle, const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
static void lengthArgPosHandle(unsigned ArgPos, LengthHandleKind LengthHandle, const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
static bool isDestCapacityFix(const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
static StringRef exprToStr(const Expr *E, const MatchFinder::MatchResult &Result)
static void insertNullTerminatorExpr(StringRef Name, const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
constexpr llvm::StringLiteral DestVarDeclName
static void removeArg(int ArgPos, const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
static unsigned getLength(const Expr *E, const MatchFinder::MatchResult &Result)
static void insertDestCapacityArg(bool IsOverflows, StringRef Name, const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
static void renameMemcpy(StringRef Name, bool IsCopy, bool IsSafe, const MatchFinder::MatchResult &Result, DiagnosticBuilder &Diag)
constexpr llvm::StringLiteral SrcExprName
constexpr llvm::StringLiteral DestExprName
llvm::StringMap< ClangTidyValue > OptionMap