forked from OSchip/llvm-project
[clang-tidy] add check cppcoreguidelines-pro-type-static-cast-downcast
Summary: This check flags all usages of static_cast, where a base class is casted to a derived class. In those cases, a fixit is provided to convert the cast to a dynamic_cast. Use of these casts can violate type safety and cause the program to access a variable that is actually of type X to be accessed as if it were of an unrelated type Z. This rule is part of the "Type safety" profile of the C++ Core Guidelines, see https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md#-type2-dont-use-static_cast-downcasts-use-dynamic_cast-instead Depends on D13313 Reviewers: alexfh, sbenza, bkramer, aaron.ballman Subscribers: cfe-commits Differential Revision: http://reviews.llvm.org/D13368 llvm-svn: 250098
This commit is contained in:
parent
8ebfb60efa
commit
a704d4bb27
|
@ -4,6 +4,7 @@ add_clang_library(clangTidyCppCoreGuidelinesModule
|
|||
CppCoreGuidelinesTidyModule.cpp
|
||||
ProTypeConstCastCheck.cpp
|
||||
ProTypeReinterpretCastCheck.cpp
|
||||
ProTypeStaticCastDowncastCheck.cpp
|
||||
|
||||
LINK_LIBS
|
||||
clangAST
|
||||
|
|
|
@ -12,6 +12,7 @@
|
|||
#include "../ClangTidyModuleRegistry.h"
|
||||
#include "ProTypeConstCastCheck.h"
|
||||
#include "ProTypeReinterpretCastCheck.h"
|
||||
#include "ProTypeStaticCastDowncastCheck.h"
|
||||
|
||||
namespace clang {
|
||||
namespace tidy {
|
||||
|
@ -25,6 +26,8 @@ public:
|
|||
"cppcoreguidelines-pro-type-const-cast");
|
||||
CheckFactories.registerCheck<ProTypeReinterpretCastCheck>(
|
||||
"cppcoreguidelines-pro-type-reinterpret-cast");
|
||||
CheckFactories.registerCheck<ProTypeStaticCastDowncastCheck>(
|
||||
"cppcoreguidelines-pro-type-static-cast-downcast");
|
||||
}
|
||||
};
|
||||
|
||||
|
|
|
@ -0,0 +1,52 @@
|
|||
//===--- ProTypeStaticCastDowncastCheck.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 "ProTypeStaticCastDowncastCheck.h"
|
||||
#include "clang/AST/ASTContext.h"
|
||||
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
||||
|
||||
using namespace clang::ast_matchers;
|
||||
|
||||
namespace clang {
|
||||
namespace tidy {
|
||||
|
||||
void ProTypeStaticCastDowncastCheck::registerMatchers(MatchFinder *Finder) {
|
||||
if (!getLangOpts().CPlusPlus)
|
||||
return;
|
||||
|
||||
Finder->addMatcher(
|
||||
cxxStaticCastExpr(unless(isInTemplateInstantiation())).bind("cast"),
|
||||
this);
|
||||
}
|
||||
|
||||
void ProTypeStaticCastDowncastCheck::check(const MatchFinder::MatchResult &Result) {
|
||||
const auto *MatchedCast = Result.Nodes.getNodeAs<CXXStaticCastExpr>("cast");
|
||||
if (MatchedCast->getCastKind() != CK_BaseToDerived)
|
||||
return;
|
||||
|
||||
QualType SourceType = MatchedCast->getSubExpr()->getType();
|
||||
const auto *SourceDecl = SourceType->getPointeeCXXRecordDecl();
|
||||
if (!SourceDecl) // The cast is from object to reference
|
||||
SourceDecl = SourceType->getAsCXXRecordDecl();
|
||||
if (!SourceDecl)
|
||||
return;
|
||||
|
||||
if (SourceDecl->isPolymorphic())
|
||||
diag(MatchedCast->getOperatorLoc(),
|
||||
"do not use static_cast to downcast from a base to a derived class; "
|
||||
"use dynamic_cast instead")
|
||||
<< FixItHint::CreateReplacement(MatchedCast->getOperatorLoc(),
|
||||
"dynamic_cast");
|
||||
else
|
||||
diag(MatchedCast->getOperatorLoc(),
|
||||
"do not use static_cast to downcast from a base to a derived class");
|
||||
}
|
||||
|
||||
} // namespace tidy
|
||||
} // namespace clang
|
|
@ -0,0 +1,33 @@
|
|||
//===--- ProTypeStaticCastDowncastCheck.h - clang-tidy-----------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_CPPCOREGUIDELINES_PRO_TYPE_STATIC_CAST_DOWNCAST_H
|
||||
#define LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_CPPCOREGUIDELINES_PRO_TYPE_STATIC_CAST_DOWNCAST_H
|
||||
|
||||
#include "../ClangTidy.h"
|
||||
|
||||
namespace clang {
|
||||
namespace tidy {
|
||||
|
||||
/// Checks for usages of static_cast, where a base class is downcasted to a derived class.
|
||||
///
|
||||
/// For the user-facing documentation see:
|
||||
/// http://clang.llvm.org/extra/clang-tidy/checks/cppcoreguidelines-pro-type-static-cast-downcast.html
|
||||
class ProTypeStaticCastDowncastCheck : public ClangTidyCheck {
|
||||
public:
|
||||
ProTypeStaticCastDowncastCheck(StringRef Name, ClangTidyContext *Context)
|
||||
: ClangTidyCheck(Name, Context) {}
|
||||
void registerMatchers(ast_matchers::MatchFinder *Finder) override;
|
||||
void check(const ast_matchers::MatchFinder::MatchResult &Result) override;
|
||||
};
|
||||
|
||||
} // namespace tidy
|
||||
} // namespace clang
|
||||
|
||||
#endif // LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_CPPCOREGUIDELINES_PRO_TYPE_STATIC_CAST_DOWNCAST_H
|
|
@ -0,0 +1,10 @@
|
|||
cppcoreguidelines-pro-type-static-cast-downcast
|
||||
===============================================
|
||||
|
||||
This check flags all usages of static_cast, where a base class is casted to a derived class.
|
||||
In those cases, a fixit is provided to convert the cast to a dynamic_cast.
|
||||
|
||||
Use of these casts can violate type safety and cause the program to access a variable that is actually of type X to be accessed as if it were of an unrelated type Z.
|
||||
|
||||
This rule is part of the "Type safety" profile of the C++ Core Guidelines, see
|
||||
https://github.com/isocpp/CppCoreGuidelines/blob/master/CppCoreGuidelines.md#-type2-dont-use-static_cast-downcasts-use-dynamic_cast-instead
|
|
@ -6,6 +6,7 @@ List of clang-tidy Checks
|
|||
cert-variadic-function-def
|
||||
cppcoreguidelines-pro-type-const-cast
|
||||
cppcoreguidelines-pro-type-reinterpret-cast
|
||||
cppcoreguidelines-pro-type-static-cast-downcast
|
||||
google-build-explicit-make-pair
|
||||
google-build-namespaces
|
||||
google-build-using-namespace
|
||||
|
|
|
@ -0,0 +1,118 @@
|
|||
// RUN: %python %S/check_clang_tidy.py %s cppcoreguidelines-pro-type-static-cast-downcast %t
|
||||
|
||||
class Base {
|
||||
};
|
||||
|
||||
class Derived : public Base {
|
||||
};
|
||||
|
||||
class Base2 {
|
||||
};
|
||||
|
||||
class MultiDerived : public Base, public Base2 {
|
||||
};
|
||||
|
||||
class PolymorphicBase {
|
||||
public:
|
||||
virtual ~PolymorphicBase();
|
||||
};
|
||||
|
||||
class PolymorphicDerived : public PolymorphicBase {
|
||||
};
|
||||
|
||||
class PolymorphicMultiDerived : public Base, public PolymorphicBase {
|
||||
};
|
||||
|
||||
void pointers() {
|
||||
|
||||
auto P0 = static_cast<Derived*>(new Base());
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:13: warning: do not use static_cast to downcast from a base to a derived class [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
|
||||
const Base* B0;
|
||||
auto PC0 = static_cast<const Derived*>(B0);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: do not use static_cast to downcast from a base to a derived class [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
|
||||
auto P1 = static_cast<Base*>(new Derived()); // OK, upcast to a public base
|
||||
auto P2 = static_cast<Base*>(new MultiDerived()); // OK, upcast to a public base
|
||||
auto P3 = static_cast<Base2*>(new MultiDerived()); // OK, upcast to a public base
|
||||
}
|
||||
|
||||
void pointers_polymorphic() {
|
||||
|
||||
auto PP0 = static_cast<PolymorphicDerived*>(new PolymorphicBase());
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: do not use static_cast to downcast from a base to a derived class; use dynamic_cast instead [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
// CHECK-FIXES: auto PP0 = dynamic_cast<PolymorphicDerived*>(new PolymorphicBase());
|
||||
|
||||
const PolymorphicBase* B0;
|
||||
auto PPC0 = static_cast<const PolymorphicDerived*>(B0);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:15: warning: do not use static_cast to downcast from a base to a derived class; use dynamic_cast instead [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
// CHECK-FIXES: auto PPC0 = dynamic_cast<const PolymorphicDerived*>(B0);
|
||||
|
||||
|
||||
auto B1 = static_cast<PolymorphicBase*>(new PolymorphicDerived()); // OK, upcast to a public base
|
||||
auto B2 = static_cast<PolymorphicBase*>(new PolymorphicMultiDerived()); // OK, upcast to a public base
|
||||
auto B3 = static_cast<Base*>(new PolymorphicMultiDerived()); // OK, upcast to a public base
|
||||
}
|
||||
|
||||
void arrays() {
|
||||
Base ArrayOfBase[10];
|
||||
auto A0 = static_cast<Derived*>(ArrayOfBase);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:13: warning: do not use static_cast to downcast from a base to a derived class [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
}
|
||||
|
||||
void arrays_polymorphic() {
|
||||
PolymorphicBase ArrayOfPolymorphicBase[10];
|
||||
auto AP0 = static_cast<PolymorphicDerived*>(ArrayOfPolymorphicBase);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: do not use static_cast to downcast from a base to a derived class; use dynamic_cast instead
|
||||
// CHECK-FIXES: auto AP0 = dynamic_cast<PolymorphicDerived*>(ArrayOfPolymorphicBase);
|
||||
}
|
||||
|
||||
void references() {
|
||||
Base B0;
|
||||
auto R0 = static_cast<Derived&>(B0);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:13: warning: do not use static_cast to downcast from a base to a derived class [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
Base& RefToBase = B0;
|
||||
auto R1 = static_cast<Derived&>(RefToBase);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:13: warning: do not use static_cast to downcast from a base to a derived class [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
|
||||
const Base& ConstRefToBase = B0;
|
||||
auto RC1 = static_cast<const Derived&>(ConstRefToBase);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: do not use static_cast to downcast from a base to a derived class [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
|
||||
|
||||
Derived RD1;
|
||||
auto R2 = static_cast<Base&>(RD1); // OK, upcast to a public base
|
||||
}
|
||||
|
||||
void references_polymorphic() {
|
||||
PolymorphicBase B0;
|
||||
auto RP0 = static_cast<PolymorphicDerived&>(B0);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: do not use static_cast to downcast from a base to a derived class; use dynamic_cast instead
|
||||
// CHECK-FIXES: auto RP0 = dynamic_cast<PolymorphicDerived&>(B0);
|
||||
|
||||
PolymorphicBase& RefToPolymorphicBase = B0;
|
||||
auto RP1 = static_cast<PolymorphicDerived&>(RefToPolymorphicBase);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: do not use static_cast to downcast from a base to a derived class; use dynamic_cast instead [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
// CHECK-FIXES: auto RP1 = dynamic_cast<PolymorphicDerived&>(RefToPolymorphicBase);
|
||||
|
||||
const PolymorphicBase& ConstRefToPolymorphicBase = B0;
|
||||
auto RPC2 = static_cast<const PolymorphicDerived&>(ConstRefToPolymorphicBase);
|
||||
// CHECK-MESSAGES: :[[@LINE-1]]:15: warning: do not use static_cast to downcast from a base to a derived class; use dynamic_cast instead [cppcoreguidelines-pro-type-static-cast-downcast]
|
||||
// CHECK-FIXES: auto RPC2 = dynamic_cast<const PolymorphicDerived&>(ConstRefToPolymorphicBase);
|
||||
|
||||
PolymorphicDerived d1;
|
||||
auto RP2 = static_cast<PolymorphicBase&>(d1); // OK, upcast to a public base
|
||||
}
|
||||
|
||||
template<class B, class D>
|
||||
void templ() {
|
||||
auto B0 = static_cast<B*>(new D());
|
||||
}
|
||||
|
||||
void templ_bad_call() {
|
||||
templ<Derived, Base>(); //FIXME: this should trigger a warning
|
||||
}
|
||||
|
||||
void templ_good_call() {
|
||||
templ<Base, Derived>(); // OK, upcast to a public base
|
||||
}
|
Loading…
Reference in New Issue