2016-04-08 00:16:36 +08:00
|
|
|
//===--- StringLiteralWithEmbeddedNulCheck.cpp - clang-tidy----------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "StringLiteralWithEmbeddedNulCheck.h"
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
|
|
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
|
|
|
|
|
|
|
using namespace clang::ast_matchers;
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace tidy {
|
2018-03-01 07:30:29 +08:00
|
|
|
namespace bugprone {
|
2016-04-08 00:16:36 +08:00
|
|
|
|
2018-02-19 03:02:35 +08:00
|
|
|
namespace {
|
2016-04-08 00:16:36 +08:00
|
|
|
AST_MATCHER(StringLiteral, containsNul) {
|
|
|
|
for (size_t i = 0; i < Node.getLength(); ++i)
|
|
|
|
if (Node.getCodeUnit(i) == '\0')
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
2018-02-19 03:02:35 +08:00
|
|
|
} // namespace
|
2016-04-08 00:16:36 +08:00
|
|
|
|
|
|
|
void StringLiteralWithEmbeddedNulCheck::registerMatchers(MatchFinder *Finder) {
|
|
|
|
// Match a string that contains embedded NUL character. Extra-checks are
|
|
|
|
// applied in |check| to find incorectly escaped characters.
|
|
|
|
Finder->addMatcher(stringLiteral(containsNul()).bind("strlit"), this);
|
|
|
|
|
|
|
|
// The remaining checks only apply to C++.
|
|
|
|
if (!getLangOpts().CPlusPlus)
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto StrLitWithNul =
|
|
|
|
ignoringParenImpCasts(stringLiteral(containsNul()).bind("truncated"));
|
|
|
|
|
|
|
|
// Match string constructor.
|
|
|
|
const auto StringConstructorExpr = expr(anyOf(
|
|
|
|
cxxConstructExpr(argumentCountIs(1),
|
|
|
|
hasDeclaration(cxxMethodDecl(hasName("basic_string")))),
|
|
|
|
// If present, the second argument is the alloc object which must not
|
|
|
|
// be present explicitly.
|
|
|
|
cxxConstructExpr(argumentCountIs(2),
|
|
|
|
hasDeclaration(cxxMethodDecl(hasName("basic_string"))),
|
|
|
|
hasArgument(1, cxxDefaultArgExpr()))));
|
|
|
|
|
|
|
|
// Detect passing a suspicious string literal to a string constructor.
|
|
|
|
// example: std::string str = "abc\0def";
|
|
|
|
Finder->addMatcher(
|
|
|
|
cxxConstructExpr(StringConstructorExpr, hasArgument(0, StrLitWithNul)),
|
|
|
|
this);
|
|
|
|
|
|
|
|
// Detect passing a suspicious string literal through an overloaded operator.
|
|
|
|
Finder->addMatcher(cxxOperatorCallExpr(hasAnyArgument(StrLitWithNul)), this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void StringLiteralWithEmbeddedNulCheck::check(
|
|
|
|
const MatchFinder::MatchResult &Result) {
|
|
|
|
if (const auto *SL = Result.Nodes.getNodeAs<StringLiteral>("strlit")) {
|
|
|
|
for (size_t Offset = 0, Length = SL->getLength(); Offset < Length;
|
|
|
|
++Offset) {
|
|
|
|
// Find a sequence of character like "\0x12".
|
|
|
|
if (Offset + 3 < Length && SL->getCodeUnit(Offset) == '\0' &&
|
|
|
|
SL->getCodeUnit(Offset + 1) == 'x' &&
|
|
|
|
isDigit(SL->getCodeUnit(Offset + 2)) &&
|
|
|
|
isDigit(SL->getCodeUnit(Offset + 3))) {
|
2018-08-10 06:42:26 +08:00
|
|
|
diag(SL->getBeginLoc(), "suspicious embedded NUL character");
|
2016-04-08 00:16:36 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (const auto *SL = Result.Nodes.getNodeAs<StringLiteral>("truncated")) {
|
2018-08-10 06:42:26 +08:00
|
|
|
diag(SL->getBeginLoc(),
|
2016-04-08 00:16:36 +08:00
|
|
|
"truncated string literal with embedded NUL character");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-01 07:30:29 +08:00
|
|
|
} // namespace bugprone
|
2016-04-08 00:16:36 +08:00
|
|
|
} // namespace tidy
|
|
|
|
} // namespace clang
|