ckennelly created this revision. ckennelly added reviewers: hokein, njames93, alexfh, aaron.ballman. Herald added subscribers: xazax.hun, mgorny. ckennelly requested review of this revision. Herald added a project: clang. Herald added a subscriber: cfe-commits.
std::make_unique_for_overwrite and std::make_shared_for_overwrite were introduced in C++20. These default initialize the values, unlike std::make_unique / std::make_shared, which value initialize. The latter can cause unexpected performance regressions when trivial types (such as int) are subsequently value initialized. Repository: rG LLVM Github Monorepo https://reviews.llvm.org/D92992 Files: clang-tools-extra/clang-tidy/modernize/CMakeLists.txt clang-tools-extra/clang-tidy/modernize/MakeSharedCheck.cpp clang-tools-extra/clang-tidy/modernize/MakeSharedForOverwriteCheck.cpp clang-tools-extra/clang-tidy/modernize/MakeSharedForOverwriteCheck.h clang-tools-extra/clang-tidy/modernize/MakeSmartPtrCheck.cpp clang-tools-extra/clang-tidy/modernize/MakeSmartPtrCheck.h clang-tools-extra/clang-tidy/modernize/MakeUniqueCheck.cpp clang-tools-extra/clang-tidy/modernize/MakeUniqueForOverwriteCheck.cpp clang-tools-extra/clang-tidy/modernize/MakeUniqueForOverwriteCheck.h clang-tools-extra/clang-tidy/modernize/ModernizeTidyModule.cpp clang-tools-extra/docs/ReleaseNotes.rst clang-tools-extra/docs/clang-tidy/checks/list.rst clang-tools-extra/docs/clang-tidy/checks/modernize-make-shared-for-overwrite.rst clang-tools-extra/docs/clang-tidy/checks/modernize-make-unique-for-overwrite.rst clang-tools-extra/test/clang-tidy/checkers/modernize-make-shared-for-overwrite.cpp clang-tools-extra/test/clang-tidy/checkers/modernize-make-unique-for-overwrite.cpp
Index: clang-tools-extra/test/clang-tidy/checkers/modernize-make-unique-for-overwrite.cpp =================================================================== --- /dev/null +++ clang-tools-extra/test/clang-tidy/checkers/modernize-make-unique-for-overwrite.cpp @@ -0,0 +1,476 @@ +// RUN: %check_clang_tidy -std=c++20-or-later %s modernize-make-unique-for-overwrite %t -- -- -I %S/Inputs/modernize-smart-ptr + +#include "initializer_list.h" +#include "unique_ptr.h" +// CHECK-FIXES: #include <memory> + +struct Base { + Base(); + Base(int, int); +}; + +struct Derived : public Base { + Derived(); + Derived(int, int); +}; + +struct APair { + int a, b; +}; + +struct DPair { + DPair() : a(0), b(0) {} + DPair(int x, int y) : a(y), b(x) {} + int a, b; +}; + +template <typename T> +struct MyVector { + MyVector(std::initializer_list<T>); +}; + +struct Empty {}; + +struct E { + E(std::initializer_list<int>); + E(); +}; + +struct F { + F(std::initializer_list<int>); + F(); + int a; +}; + +struct G { + G(std::initializer_list<int>); + G(int); +}; + +struct H { + H(std::vector<int>); + H(std::vector<int> &, double); + H(MyVector<int>, int); +}; + +struct I { + I(G); +}; + +struct J { + J(E e, int); +}; + +namespace { +class Foo {}; +} // namespace + +namespace bar { +class Bar {}; +} // namespace bar + +template <class T> +using unique_ptr_ = std::unique_ptr<T>; + +void *operator new(__SIZE_TYPE__ Count, void *Ptr); + +int g(std::unique_ptr<int> P); + +std::unique_ptr<Base> getPointer() { + return std::unique_ptr<Base>(new Base); +} + +std::unique_ptr<Base> getPointerValue() { + return std::unique_ptr<Base>(new Base()); +} + +void basic() { + std::unique_ptr<int> P1 = std::unique_ptr<int>(new int()); + std::unique_ptr<int> P2 = std::unique_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:29: warning: use std::make_unique_for_overwrite instead [modernize-make-unique-for-overwrite] + // CHECK-FIXES: std::unique_ptr<int> P2 = std::make_unique_for_overwrite<int>(); + + P1.reset(new int()); + P2.reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_unique_for_overwrite instead [modernize-make-unique-for-overwrite] + // CHECK-FIXES: P2 = std::make_unique_for_overwrite<int>(); + + P1 = std::unique_ptr<int>(new int()); + P2 = std::unique_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:8: warning: use std::make_unique_for_overwrite instead [modernize-make-unique-for-overwrite] + // CHECK-FIXES: P2 = std::make_unique_for_overwrite<int>(); + + P1.reset(new int()); + P2.reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_unique_for_overwrite instead [modernize-make-unique-for-overwrite] + // CHECK-FIXES: P2 = std::make_unique_for_overwrite<int>(); + + // With auto. + auto P4 = std::unique_ptr<int>(new int()); + auto P5 = std::unique_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:13: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: auto P5 = std::make_unique_for_overwrite<int>(); + + std::unique_ptr<int> P6 = std::unique_ptr<int>((new int())); + std::unique_ptr<int> P7 = std::unique_ptr<int>((new int)); + // CHECK-MESSAGES: :[[@LINE-1]]:29: warning: use std::make_unique_for_overwrite instead [modernize-make-unique-for-overwrite] + // CHECK-FIXES: std::unique_ptr<int> P7 = std::make_unique_for_overwrite<int>(); + + P4.reset((new int())); + P5.reset((new int)); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_unique_for_overwrite instead [modernize-make-unique-for-overwrite] + // CHECK-FIXES: P5 = std::make_unique_for_overwrite<int>(); + + std::unique_ptr<int> P8 = std::unique_ptr<int>((((new int())))); + std::unique_ptr<int> P9 = std::unique_ptr<int>((((new int)))); + // CHECK-MESSAGES: :[[@LINE-1]]:29: warning: use std::make_unique_for_overwrite instead [modernize-make-unique-for-overwrite] + // CHECK-FIXES: std::unique_ptr<int> P9 = std::make_unique_for_overwrite<int>(); + + P5.reset(((((new int()))))); + P6.reset(((((new int))))); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_unique_for_overwrite instead [modernize-make-unique-for-overwrite] + // CHECK-FIXES: P6 = std::make_unique_for_overwrite<int>(); + + { + // No std. + using namespace std; + unique_ptr<int> Q = unique_ptr<int>(new int()); + unique_ptr<int> P = unique_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:25: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: unique_ptr<int> P = std::make_unique_for_overwrite<int>(); + + Q = unique_ptr<int>(new int()); + P = unique_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: P = std::make_unique_for_overwrite<int>(); + } + + std::unique_ptr<int> R(new int()); + std::unique_ptr<int> S(new int); + + // Create the unique_ptr as a parameter to a function. + int T = g(std::unique_ptr<int>(new int())); + T = g(std::unique_ptr<int>(new int)); + // CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: T = g(std::make_unique_for_overwrite<int>()); + + // Only replace if the type in the template is the same as the type returned + // by the new operator. + auto Pderived = std::unique_ptr<Base>(new Derived()); + auto PderivedNoparen = std::unique_ptr<Base>(new Derived); + + // OK to replace for reset and assign + Pderived.reset(new Derived()); + PderivedNoparen.reset(new Derived); + + Pderived = std::unique_ptr<Derived>(new Derived()); + PderivedNoparen = std::unique_ptr<Derived>(new Derived); + + // FIXME: OK to replace if assigned to unique_ptr<Base> + Pderived = std::unique_ptr<Base>(new Derived()); + Pderived = std::unique_ptr<Base>(new Derived); + + // FIXME: OK to replace when auto is not used + std::unique_ptr<Base> PBase = std::unique_ptr<Base>(new Derived()); + std::unique_ptr<Base> PBaseNoparen = std::unique_ptr<Base>(new Derived); + + // The pointer is returned by the function, nothing to do. + std::unique_ptr<Base> RetPtr = getPointer(); + RetPtr = getPointerValue(); + + // This emulates std::move. + std::unique_ptr<int> Move = static_cast<std::unique_ptr<int> &&>(P1); + + // Placement arguments should not be removed. + int *PInt = new int; + std::unique_ptr<int> Placement = std::unique_ptr<int>(new (PInt) int{3}); + Placement.reset(new (PInt) int{3}); + Placement = std::unique_ptr<int>(new (PInt) int{3}); + + std::unique_ptr<int> PlacementNoparen = std::unique_ptr<int>(new (PInt) int); + PlacementNoparen.reset(new (PInt) int); + PlacementNoparen = std::unique_ptr<int>(new (PInt) int); +} + +// Calling make_smart_ptr from within a member function of a type with a +// private or protected constructor would be ill-formed. +class Private { +private: + Private(int z) {} + +public: + Private() {} + void create() { + auto callsPublic = std::unique_ptr<Private>(new Private); + auto ptr = std::unique_ptr<Private>(new Private(42)); + ptr.reset(new Private(42)); + ptr = std::unique_ptr<Private>(new Private(42)); + } + + virtual ~Private(); +}; + +class Protected { +protected: + Protected() {} + +public: + Protected(int, int) {} + void create() { + auto callsPublic = std::unique_ptr<Protected>(new Protected(1, 2)); + auto ptr = std::unique_ptr<Protected>(new Protected); + ptr.reset(new Protected); + ptr = std::unique_ptr<Protected>(new Protected); + } +}; + +void initialization(int T, Base b) { + // Test different kinds of initialization of the pointee. + + // Direct initialization with parenthesis. + std::unique_ptr<DPair> PDir1 = std::unique_ptr<DPair>(new DPair(1, T)); + PDir1.reset(new DPair(1, T)); + + // Direct initialization with braces. + std::unique_ptr<DPair> PDir2 = std::unique_ptr<DPair>(new DPair{2, T}); + PDir2.reset(new DPair{2, T}); + + // Aggregate initialization. + std::unique_ptr<APair> PAggr = std::unique_ptr<APair>(new APair{T, 1}); + PAggr.reset(new APair{T, 1}); + + // Check aggregate init with intermediate temporaries. + std::unique_ptr<APair> PAggrTemp = std::unique_ptr<APair>(new APair({T, 1})); + PAggrTemp.reset(new APair({T, 1})); + + // Test different kinds of initialization of the pointee, when the unique_ptr + // is initialized with braces. + + // Direct initialization with parenthesis. + std::unique_ptr<DPair> PDir3 = std::unique_ptr<DPair>{new DPair(3, T)}; + + // Direct initialization with braces. + std::unique_ptr<DPair> PDir4 = std::unique_ptr<DPair>{new DPair{4, T}}; + + // Aggregate initialization. + std::unique_ptr<APair> PAggr2 = std::unique_ptr<APair>{new APair{T, 2}}; + + // Direct initialization with parenthesis, without arguments. + std::unique_ptr<DPair> PDir5 = std::unique_ptr<DPair>(new DPair()); + + // Direct initialization with braces, without arguments. + std::unique_ptr<DPair> PDir6 = std::unique_ptr<DPair>(new DPair{}); + + // Aggregate initialization without arguments. + std::unique_ptr<Empty> PEmpty = std::unique_ptr<Empty>(new Empty{}); + + // Initialization with default constructor. + std::unique_ptr<E> PE1 = std::unique_ptr<E>(new E{}); + PE1.reset(new E{}); + + // No warnings for `auto` new expression. + PE1.reset(new auto(E())); + + //============================================================================ + // NOTE: For initializer-list constructors, the check only gives warnings, + // and no fixes are generated. + //============================================================================ + + // Initialization with the initializer-list constructor. + std::unique_ptr<E> PE2 = std::unique_ptr<E>(new E{1, 2}); + PE2.reset(new E{1, 2}); + + // Initialization with default constructor. + std::unique_ptr<F> PF1 = std::unique_ptr<F>(new F()); + PF1.reset(new F()); + + // Initialization with default constructor. + std::unique_ptr<F> PF2 = std::unique_ptr<F>(new F{}); + PF2.reset(new F()); + + // Initialization with the initializer-list constructor. + std::unique_ptr<F> PF3 = std::unique_ptr<F>(new F{1}); + PF3.reset(new F{1}); + + // Initialization with the initializer-list constructor. + std::unique_ptr<F> PF4 = std::unique_ptr<F>(new F{1, 2}); + + // Initialization with the initializer-list constructor. + std::unique_ptr<F> PF5 = std::unique_ptr<F>(new F({1, 2})); + + // Initialization with the initializer-list constructor as the default + // constructor is not present. + std::unique_ptr<G> PG1 = std::unique_ptr<G>(new G{}); + PG1.reset(new G{}); + + // Initialization with the initializer-list constructor. + std::unique_ptr<G> PG2 = std::unique_ptr<G>(new G{1}); + + // Initialization with the initializer-list constructor. + std::unique_ptr<G> PG3 = std::unique_ptr<G>(new G{1, 2}); + + std::unique_ptr<H> PH1 = std::unique_ptr<H>(new H({1, 2, 3})); + PH1.reset(new H({1, 2, 3})); + + std::unique_ptr<H> PH2 = std::unique_ptr<H>(new H({1, 2, 3}, 1)); + PH2.reset(new H({1, 2, 3}, 1)); + + std::unique_ptr<H> PH3 = std::unique_ptr<H>(new H({1, 2, 3}, 1.0)); + PH3.reset(new H({1, 2, 3}, 1.0)); + + std::unique_ptr<I> PI1 = std::unique_ptr<I>(new I(G({1, 2, 3}))); + PI1.reset(new I(G({1, 2, 3}))); + + std::unique_ptr<J> PJ1 = std::unique_ptr<J>(new J({1, 2}, 1)); + PJ1.reset(new J({1, 2}, 1)); + + std::unique_ptr<J> PJ2 = std::unique_ptr<J>(new J(E{1, 2}, 1)); + PJ2.reset(new J(E{1, 2}, 1)); + + std::unique_ptr<J> PJ3 = std::unique_ptr<J>(new J{{1, 2}, 1}); + PJ3.reset(new J{{1, 2}, 1}); + + std::unique_ptr<J> PJ4 = std::unique_ptr<J>(new J{E{1, 2}, 1}); + PJ4.reset(new J{E{1, 2}, 1}); + + std::unique_ptr<Foo> FF = std::unique_ptr<Foo>(new Foo()); + FF.reset(new Foo()); + + std::unique_ptr<bar::Bar> BB = std::unique_ptr<bar::Bar>(new bar::Bar()); + BB.reset(new bar::Bar()); + + std::unique_ptr<Foo[]> FFs; + FFs.reset(new Foo[5]); + FFs.reset(new Foo[5]()); + const int Num = 1; + FFs.reset(new Foo[Num]); + int Num2 = 1; + + // The check doesn't give warnings and fixes for cases where the original new + // expression does value initialization. + std::unique_ptr<int[]> FI; + FI.reset(new int[5]()); + + FI.reset(new int[5]); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: + // CHECK-FIXES: FI = std::make_unique_for_overwrite<int[]>(5); + FI.reset(new int[Num]); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: + // CHECK-FIXES: FI = std::make_unique_for_overwrite<int[]>(Num); + FI.reset(new int[Num2]); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: + // CHECK-FIXES: FI = std::make_unique_for_overwrite<int[]>(Num2); +} + +void aliases() { + typedef std::unique_ptr<int> IntPtr; + IntPtr Typedef = IntPtr(new int()); + IntPtr Typedef2 = IntPtr(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:21: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: IntPtr Typedef2 = std::make_unique_for_overwrite<int>(); + + // We use 'bool' instead of '_Bool'. + typedef std::unique_ptr<bool> BoolPtr; + BoolPtr BoolType = BoolPtr(new bool()); + BoolPtr BoolType2 = BoolPtr(new bool); + // CHECK-MESSAGES: :[[@LINE-1]]:23: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: BoolPtr BoolType2 = std::make_unique_for_overwrite<bool>(); + + // We use 'Base' instead of 'struct Base'. + typedef std::unique_ptr<Base> BasePtr; + BasePtr StructType = BasePtr(new Base); + +#define PTR unique_ptr<int> + std::unique_ptr<int> Macro = std::PTR(new int()); + std::unique_ptr<int> Macro2 = std::PTR(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:33: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: std::unique_ptr<int> Macro2 = std::make_unique_for_overwrite<int>(); +#undef PTR + + std::unique_ptr<int> Using = unique_ptr_<int>(new int()); + std::unique_ptr<int> Using2 = unique_ptr_<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:33: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: std::unique_ptr<int> Using2 = std::make_unique_for_overwrite<int>(); +} + +void whitespaces() { + // clang-format off + auto Space = std::unique_ptr <int>(new int()); + auto Space2 = std::unique_ptr <int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:17: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: auto Space2 = std::make_unique_for_overwrite<int>(); + + auto Spaces = std :: unique_ptr <int>(new int()); + auto Spaces2 = std :: unique_ptr <int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:18: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: auto Spaces2 = std::make_unique_for_overwrite<int>(); + // clang-format on +} + +void nesting() { + auto Nest = std::unique_ptr<std::unique_ptr<int>>(new std::unique_ptr<int>(new int)); + Nest.reset(new std::unique_ptr<int>(new int)); + Nest->reset(new int()); + Nest->reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: *Nest = std::make_unique_for_overwrite<int>(); +} + +void reset() { + std::unique_ptr<int> P; + P.reset(); + P.reset(nullptr); + P.reset(new int()); + P.reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:5: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: P = std::make_unique_for_overwrite<int>(); + + auto Q = &P; + Q->reset(new int()); + Q->reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_unique_for_overwrite instead + // CHECK-FIXES: *Q = std::make_unique_for_overwrite<int>(); +} + +#define DEFINE(...) __VA_ARGS__ +template <typename T> +void g2(std::unique_ptr<Foo> *t) { + DEFINE(auto p = std::unique_ptr<Foo>(new Foo); t->reset(new Foo);); +} +void macro() { + std::unique_ptr<Foo> *t; + g2<bar::Bar>(t); +} +#undef DEFINE + +class UniqueFoo : public std::unique_ptr<Foo> { +public: + void foo() { + reset(new Foo); + this->reset(new Foo); + this->reset(new Foo()); + (*this).reset(new Foo); + (*this).reset(new Foo()); + } +}; + +// Ignore statements inside a template instantiation. +template <typename T> +void template_fun(T *t) { + std::unique_ptr<T> t2 = std::unique_ptr<T>(new T); + std::unique_ptr<T> t3 = std::unique_ptr<T>(new T()); + t2.reset(new T); + t3.reset(new T()); +} + +void invoke_template() { + Foo *foo; + template_fun(foo); +} + +void no_fix_for_invalid_new_loc() { + // FIXME: Although the code is valid, the end location of `new struct Base` is + // invalid. Correct it once https://bugs.llvm.org/show_bug.cgi?id=35952 is + // fixed. + auto T = std::unique_ptr<Base>(new struct Base); +} Index: clang-tools-extra/test/clang-tidy/checkers/modernize-make-shared-for-overwrite.cpp =================================================================== --- /dev/null +++ clang-tools-extra/test/clang-tidy/checkers/modernize-make-shared-for-overwrite.cpp @@ -0,0 +1,288 @@ +// RUN: %check_clang_tidy -std=c++20-or-later %s modernize-make-shared-for-overwrite %t -- -- -I %S/Inputs/modernize-smart-ptr + +#include "shared_ptr.h" +// CHECK-FIXES: #include <memory> + +struct Base { + Base(); + Base(int, int); +}; + +struct Derived : public Base { + Derived(); + Derived(int, int); +}; + +struct APair { + int a, b; +}; + +struct DPair { + DPair() : a(0), b(0) {} + DPair(int x, int y) : a(y), b(x) {} + int a, b; +}; + +struct Empty {}; + +template <class T> +using shared_ptr_ = std::shared_ptr<T>; + +void *operator new(__SIZE_TYPE__ Count, void *Ptr); + +int g(std::shared_ptr<int> P); + +std::shared_ptr<Base> getPointer() { + return std::shared_ptr<Base>(new Base); +} + +std::shared_ptr<Base> getPointerValue() { + return std::shared_ptr<Base>(new Base()); +} + +void basic() { + std::shared_ptr<int> P1 = std::shared_ptr<int>(new int()); + std::shared_ptr<int> P2 = std::shared_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:29: warning: use std::make_shared_for_overwrite instead [modernize-make-shared-for-overwrite] + // CHECK-FIXES: std::shared_ptr<int> P2 = std::make_shared_for_overwrite<int>(); + + P1.reset(new int()); + P2.reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_shared_for_overwrite instead [modernize-make-shared-for-overwrite] + // CHECK-FIXES: P2 = std::make_shared_for_overwrite<int>(); + + P1 = std::shared_ptr<int>(new int()); + P2 = std::shared_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:8: warning: use std::make_shared_for_overwrite instead [modernize-make-shared-for-overwrite] + // CHECK-FIXES: P2 = std::make_shared_for_overwrite<int>(); + + P2.reset(new int()); + P2.reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_shared_for_overwrite instead [modernize-make-shared-for-overwrite] + // CHECK-FIXES: P2 = std::make_shared_for_overwrite<int>(); + + P2 = std::shared_ptr<int>(new int()); + P2 = std::shared_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:8: warning: use std::make_shared_for_overwrite instead [modernize-make-shared-for-overwrite] + // CHECK-FIXES: P2 = std::make_shared_for_overwrite<int>(); + + // With auto. + auto P4 = std::shared_ptr<int>(new int()); + auto P5 = std::shared_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:13: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: auto P5 = std::make_shared_for_overwrite<int>(); + + std::shared_ptr<int> P6 = std::shared_ptr<int>((new int())); + std::shared_ptr<int> P7 = std::shared_ptr<int>((new int)); + // CHECK-MESSAGES: :[[@LINE-1]]:29: warning: use std::make_shared_for_overwrite instead [modernize-make-shared-for-overwrite] + // CHECK-FIXES: std::shared_ptr<int> P7 = std::make_shared_for_overwrite<int>(); + + P4.reset((((new int())))); + P4.reset((((new int)))); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_shared_for_overwrite instead [modernize-make-shared-for-overwrite] + // CHECK-FIXES: P4 = std::make_shared_for_overwrite<int>(); + + P4 = std::shared_ptr<int>(((new int()))); + P4 = std::shared_ptr<int>(((new int))); + // CHECK-MESSAGES: :[[@LINE-1]]:8: warning: use std::make_shared_for_overwrite instead [modernize-make-shared-for-overwrite] + // CHECK-FIXES: P4 = std::make_shared_for_overwrite<int>(); + + { + // No std. + using namespace std; + shared_ptr<int> Q = shared_ptr<int>(new int()); + shared_ptr<int> P = shared_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:25: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: shared_ptr<int> P = std::make_shared_for_overwrite<int>(); + + Q = shared_ptr<int>(new int()); + Q = shared_ptr<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: Q = std::make_shared_for_overwrite<int>(); + } + + std::shared_ptr<int> R(new int()); + std::shared_ptr<int> S(new int); + + // Create the shared_ptr as a parameter to a function. + int T = g(std::shared_ptr<int>(new int())); + T = g(std::shared_ptr<int>(new int)); + // CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: T = g(std::make_shared_for_overwrite<int>()); + + // Only replace if the type in the template is the same as the type returned + // by the new operator. + auto Pderived = std::shared_ptr<Base>(new Derived()); + auto PderivedNoparen = std::shared_ptr<Base>(new Derived); + + // OK to replace for reset and assign + Pderived.reset(new Derived()); + Pderived.reset(new Derived); + + Pderived = std::shared_ptr<Derived>(new Derived()); + Pderived = std::shared_ptr<Derived>(new Derived); + + // FIXME: OK to replace if assigned to shared_ptr<Base> + Pderived = std::shared_ptr<Base>(new Derived()); + Pderived = std::shared_ptr<Base>(new Derived); + + // FIXME: OK to replace when auto is not used + std::shared_ptr<Base> PBase = std::shared_ptr<Base>(new Derived()); + std::shared_ptr<Base> PBase2 = std::shared_ptr<Base>(new Derived); + + // The pointer is returned by the function, nothing to do. + std::shared_ptr<Base> RetPtr = getPointer(); + std::shared_ptr<Base> RetPtr2 = getPointerValue(); + + // This emulates std::move. + std::shared_ptr<int> Move = static_cast<std::shared_ptr<int> &&>(P1); + + // Placement arguments should not be removed. + int *PInt = new int; + std::shared_ptr<int> Placement = std::shared_ptr<int>(new (PInt) int{3}); + Placement.reset(new (PInt) int{3}); + Placement = std::shared_ptr<int>(new (PInt) int{3}); + + std::shared_ptr<int> PlacementNoparen = std::shared_ptr<int>(new (PInt) int); + PlacementNoparen.reset(new (PInt) int); + PlacementNoparen = std::shared_ptr<int>(new (PInt) int); +} + +// Calling make_smart_ptr from within a member function of a type with a +// private or protected constructor would be ill-formed. +class Private { +private: + Private(int z) {} + +public: + Private() {} + void create() { + auto callsPublic = std::shared_ptr<Private>(new Private); + auto ptr = std::shared_ptr<Private>(new Private(42)); + ptr.reset(new Private(42)); + ptr = std::shared_ptr<Private>(new Private(42)); + } + + virtual ~Private(); +}; + +class Protected { +protected: + Protected() {} + +public: + Protected(int, int) {} + void create() { + auto callsPublic = std::shared_ptr<Protected>(new Protected(1, 2)); + auto ptr = std::shared_ptr<Protected>(new Protected); + ptr.reset(new Protected); + ptr = std::shared_ptr<Protected>(new Protected); + } +}; + +void initialization(int T, Base b) { + // Test different kinds of initialization of the pointee. + + // Direct initialization with parenthesis. + std::shared_ptr<DPair> PDir1 = std::shared_ptr<DPair>(new DPair(1, T)); + PDir1.reset(new DPair(1, T)); + + // Direct initialization with braces. + std::shared_ptr<DPair> PDir2 = std::shared_ptr<DPair>(new DPair{2, T}); + PDir2.reset(new DPair{2, T}); + + // Aggregate initialization. + std::shared_ptr<APair> PAggr = std::shared_ptr<APair>(new APair{T, 1}); + PAggr.reset(new APair{T, 1}); + + // Test different kinds of initialization of the pointee, when the shared_ptr + // is initialized with braces. + + // Direct initialization with parenthesis. + std::shared_ptr<DPair> PDir3 = std::shared_ptr<DPair>{new DPair(3, T)}; + + // Direct initialization with braces. + std::shared_ptr<DPair> PDir4 = std::shared_ptr<DPair>{new DPair{4, T}}; + + // Aggregate initialization. + std::shared_ptr<APair> PAggr2 = std::shared_ptr<APair>{new APair{T, 2}}; + + // Direct initialization with parenthesis, without arguments. + std::shared_ptr<DPair> PDir5 = std::shared_ptr<DPair>(new DPair()); + + // Direct initialization with braces, without arguments. + std::shared_ptr<DPair> PDir6 = std::shared_ptr<DPair>(new DPair{}); + + // Aggregate initialization without arguments. + std::shared_ptr<Empty> PEmpty = std::shared_ptr<Empty>(new Empty{}); +} + +void aliases() { + typedef std::shared_ptr<int> IntPtr; + IntPtr Typedef = IntPtr(new int()); + IntPtr Typedef2 = IntPtr(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:21: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: IntPtr Typedef2 = std::make_shared_for_overwrite<int>(); + + // We use 'bool' instead of '_Bool'. + typedef std::shared_ptr<bool> BoolPtr; + BoolPtr BoolType = BoolPtr(new bool()); + BoolPtr BoolType2 = BoolPtr(new bool); + // CHECK-MESSAGES: :[[@LINE-1]]:23: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: BoolPtr BoolType2 = std::make_shared_for_overwrite<bool>(); + + // We use 'Base' instead of 'struct Base'. + typedef std::shared_ptr<Base> BasePtr; + BasePtr StructType = BasePtr(new Base); + +#define PTR shared_ptr<int> + std::shared_ptr<int> Macro = std::PTR(new int()); + std::shared_ptr<int> Macro2 = std::PTR(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:33: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: std::shared_ptr<int> Macro2 = std::make_shared_for_overwrite<int>(); +#undef PTR + + std::shared_ptr<int> Using = shared_ptr_<int>(new int()); + std::shared_ptr<int> Using2 = shared_ptr_<int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:33: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: std::shared_ptr<int> Using2 = std::make_shared_for_overwrite<int>(); +} + +void whitespaces() { + // clang-format off + auto Space = std::shared_ptr <int>(new int()); + auto Space2 = std::shared_ptr <int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:17: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: auto Space2 = std::make_shared_for_overwrite<int>(); + + auto Spaces = std :: shared_ptr <int>(new int()); + auto Spaces2 = std :: shared_ptr <int>(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:18: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: auto Spaces2 = std::make_shared_for_overwrite<int>(); + // clang-format on +} + +void nesting() { + auto Nest = std::shared_ptr<std::shared_ptr<int>>(new std::shared_ptr<int>(new int)); + Nest.reset(new std::shared_ptr<int>(new int)); + Nest->reset(new int()); + Nest->reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: *Nest = std::make_shared_for_overwrite<int>(); +} + +void reset() { + std::shared_ptr<int> P; + P.reset(); + P.reset(nullptr); + P.reset(new int()); + P.reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:5: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: P = std::make_shared_for_overwrite<int>(); + + auto Q = &P; + Q->reset(new int()); + Q->reset(new int); + // CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_shared_for_overwrite instead + // CHECK-FIXES: *Q = std::make_shared_for_overwrite<int>(); +} Index: clang-tools-extra/docs/clang-tidy/checks/modernize-make-unique-for-overwrite.rst =================================================================== --- /dev/null +++ clang-tools-extra/docs/clang-tidy/checks/modernize-make-unique-for-overwrite.rst @@ -0,0 +1,51 @@ +.. title:: clang-tidy - modernize-make-unique-for-overwrite + +modernize-make-unique-for-overwrite +===================== + +This check finds the creation of ``std::unique_ptr`` objects by explicitly +calling the constructor and a ``new`` expression where default initialization +occurs, and replaces it with a call to ``std::make_unique_for_overwrite``, +introduced in C++20. + +.. code-block:: c++ + + auto my_ptr = std::unique_ptr<int>(new int); + + // becomes + + auto my_ptr = std::make_unique_for_overwrite<int>(); + +This check also finds calls to ``std::unique_ptr::reset()`` with a ``new`` +expression, and replaces it with a call to ``std::make_unique``. + +.. code-block:: c++ + + my_ptr.reset(new int); + + // becomes + + my_ptr = std::make_unique_for_overwrite<int>(); + +Options +------- + +.. option:: MakeSmartPtrFunction + + A string specifying the name of make-unique-ptr-for-overwrite function. Default is + `std::make_unique_for_overwrite`. + +.. option:: MakeSmartPtrFunctionHeader + + A string specifying the corresponding header of make-unique-ptr-for-overwrite function. + Default is `<memory>`. + +.. option:: IncludeStyle + + A string specifying which include-style is used, `llvm` or `google`. Default + is `llvm`. + +.. option:: IgnoreMacros + + If set to `true`, the check will not give warnings inside macros. Default + is `true`. Index: clang-tools-extra/docs/clang-tidy/checks/modernize-make-shared-for-overwrite.rst =================================================================== --- /dev/null +++ clang-tools-extra/docs/clang-tidy/checks/modernize-make-shared-for-overwrite.rst @@ -0,0 +1,51 @@ +.. title:: clang-tidy - modernize-make-shared-for-overwrite + +modernize-make-shared-for-overwrite +===================== + +This check finds the creation of ``std::shared_ptr`` objects by explicitly +calling the constructor and a ``new`` expression where default initialization +occurs, and replaces it with a call to ``std::make_shared_for_overwrite``, +introduced in C++20. + +.. code-block:: c++ + + auto my_ptr = std::shared_ptr<int>(new int); + + // becomes + + auto my_ptr = std::make_shared_for_overwrite<int>(); + +This check also finds calls to ``std::shared_ptr::reset()`` with a ``new`` +expression, and replaces it with a call to ``std::make_shared``. + +.. code-block:: c++ + + my_ptr.reset(new int); + + // becomes + + my_ptr = std::make_shared_for_overwrite<int>(); + +Options +------- + +.. option:: MakeSmartPtrFunction + + A string specifying the name of make-shared-ptr-for-overwrite function. Default is + `std::make_shared_for_overwrite`. + +.. option:: MakeSmartPtrFunctionHeader + + A string specifying the corresponding header of make-shared-ptr-for-overwrite function. + Default is `<memory>`. + +.. option:: IncludeStyle + + A string specifying which include-style is used, `llvm` or `google`. Default + is `llvm`. + +.. option:: IgnoreMacros + + If set to `true`, the check will not give warnings inside macros. Default + is `true`. Index: clang-tools-extra/docs/clang-tidy/checks/list.rst =================================================================== --- clang-tools-extra/docs/clang-tidy/checks/list.rst +++ clang-tools-extra/docs/clang-tidy/checks/list.rst @@ -221,7 +221,9 @@ `modernize-deprecated-ios-base-aliases <modernize-deprecated-ios-base-aliases.html>`_, "Yes" `modernize-loop-convert <modernize-loop-convert.html>`_, "Yes" `modernize-make-shared <modernize-make-shared.html>`_, "Yes" + `modernize-make-shared-for-overwrite <modernize-make-shared-for-overwrite.html>`_, "Yes" `modernize-make-unique <modernize-make-unique.html>`_, "Yes" + `modernize-make-unique-for-overwrite <modernize-make-unique-for-overwrite.html>`_, "Yes" `modernize-pass-by-value <modernize-pass-by-value.html>`_, "Yes" `modernize-raw-string-literal <modernize-raw-string-literal.html>`_, "Yes" `modernize-redundant-void-arg <modernize-redundant-void-arg.html>`_, "Yes" Index: clang-tools-extra/docs/ReleaseNotes.rst =================================================================== --- clang-tools-extra/docs/ReleaseNotes.rst +++ clang-tools-extra/docs/ReleaseNotes.rst @@ -140,6 +140,18 @@ Diagnoses every integer to pointer cast. +- New :doc:`modernize-make-shared-for-overwrite + <clang-tidy/checks/modernize-make-shared-for-overwrite>` check. + + Replaces default initialized objects in `new` expressions with a call to + `std::make_shared_for_overwrite` introduced in C++20. + +- New :doc:`modernize-make-unique-for-overwrite + <clang-tidy/checks/modernize-make-unique-for-overwrite>` check. + + Replaces default initialized objects in `new` expressions with a call to + `std::make_unique_for_overwrite` introduced in C++20. + - New :doc:`readability-function-cognitive-complexity <clang-tidy/checks/readability-function-cognitive-complexity>` check. Index: clang-tools-extra/clang-tidy/modernize/ModernizeTidyModule.cpp =================================================================== --- clang-tools-extra/clang-tidy/modernize/ModernizeTidyModule.cpp +++ clang-tools-extra/clang-tidy/modernize/ModernizeTidyModule.cpp @@ -16,7 +16,9 @@ #include "DeprecatedIosBaseAliasesCheck.h" #include "LoopConvertCheck.h" #include "MakeSharedCheck.h" +#include "MakeSharedForOverwriteCheck.h" #include "MakeUniqueCheck.h" +#include "MakeUniqueForOverwriteCheck.h" #include "PassByValueCheck.h" #include "RawStringLiteralCheck.h" #include "RedundantVoidArgCheck.h" @@ -60,7 +62,11 @@ "modernize-deprecated-ios-base-aliases"); CheckFactories.registerCheck<LoopConvertCheck>("modernize-loop-convert"); CheckFactories.registerCheck<MakeSharedCheck>("modernize-make-shared"); + CheckFactories.registerCheck<MakeSharedForOverwriteCheck>( + "modernize-make-shared-for-overwrite"); CheckFactories.registerCheck<MakeUniqueCheck>("modernize-make-unique"); + CheckFactories.registerCheck<MakeUniqueForOverwriteCheck>( + "modernize-make-unique-for-overwrite"); CheckFactories.registerCheck<PassByValueCheck>("modernize-pass-by-value"); CheckFactories.registerCheck<RawStringLiteralCheck>( "modernize-raw-string-literal"); Index: clang-tools-extra/clang-tidy/modernize/MakeUniqueForOverwriteCheck.h =================================================================== --- /dev/null +++ clang-tools-extra/clang-tidy/modernize/MakeUniqueForOverwriteCheck.h @@ -0,0 +1,44 @@ +//===--- MakeUniqueForOverwriteCheck.h - clang-tidy -------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_MAKEUNIQUEFOROVERWRITECHECK_H +#define LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_MAKEUNIQUEFOROVERWRITECHECK_H + +#include "MakeSmartPtrCheck.h" + +namespace clang { +namespace tidy { +namespace modernize { + +/// Replace the pattern: +/// \code +/// std::unique_ptr<type>(new type) +/// \endcode +/// +/// With the C++20 version: +/// \code +/// std::make_unique_for_overwrite<type>() +/// \endcode +class MakeUniqueForOverwriteCheck : public MakeSmartPtrCheck { +public: + MakeUniqueForOverwriteCheck(StringRef Name, ClangTidyContext *Context); + +protected: + SmartPtrTypeMatcher getSmartPointerTypeMatcher() const override; + + bool isLanguageVersionSupported(const LangOptions &LangOpts) const override; + +private: + const bool RequireCPlusPlus20; +}; + +} // namespace modernize +} // namespace tidy +} // namespace clang + +#endif // LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_MAKEUNIQUEFOROVERWRITECHECK_H Index: clang-tools-extra/clang-tidy/modernize/MakeUniqueForOverwriteCheck.cpp =================================================================== --- clang-tools-extra/clang-tidy/modernize/MakeUniqueForOverwriteCheck.cpp +++ clang-tools-extra/clang-tidy/modernize/MakeUniqueForOverwriteCheck.cpp @@ -1,4 +1,4 @@ -//===--- MakeUniqueCheck.cpp - clang-tidy----------------------------------===// +//===--- MakeUniqueForOverwriteCheck.cpp - clang-tidy ---------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. @@ -6,7 +6,9 @@ // //===----------------------------------------------------------------------===// -#include "MakeUniqueCheck.h" +#include "MakeUniqueForOverwriteCheck.h" +#include "clang/AST/ASTContext.h" +#include "clang/ASTMatchers/ASTMatchFinder.h" using namespace clang::ast_matchers; @@ -14,13 +16,13 @@ namespace tidy { namespace modernize { -MakeUniqueCheck::MakeUniqueCheck(StringRef Name, - clang::tidy::ClangTidyContext *Context) - : MakeSmartPtrCheck(Name, Context, "std::make_unique"), - RequireCPlusPlus14(Options.get("MakeSmartPtrFunction", "").empty()) {} +MakeUniqueForOverwriteCheck::MakeUniqueForOverwriteCheck( + StringRef Name, clang::tidy::ClangTidyContext *Context) + : MakeSmartPtrCheck(Name, Context, "std::make_unique_for_overwrite", false), + RequireCPlusPlus20(Options.get("MakeSmartPtrFunction", "").empty()) {} -MakeUniqueCheck::SmartPtrTypeMatcher -MakeUniqueCheck::getSmartPointerTypeMatcher() const { +MakeUniqueForOverwriteCheck::SmartPtrTypeMatcher +MakeUniqueForOverwriteCheck::getSmartPointerTypeMatcher() const { return qualType(hasUnqualifiedDesugaredType( recordType(hasDeclaration(classTemplateSpecializationDecl( hasName("::std::unique_ptr"), templateArgumentCountIs(2), @@ -36,9 +38,9 @@ equalsBoundNode(PointerType)))))))))))))))); } -bool MakeUniqueCheck::isLanguageVersionSupported( +bool MakeUniqueForOverwriteCheck::isLanguageVersionSupported( const LangOptions &LangOpts) const { - return RequireCPlusPlus14 ? LangOpts.CPlusPlus14 : LangOpts.CPlusPlus11; + return RequireCPlusPlus20 ? LangOpts.CPlusPlus20 : LangOpts.CPlusPlus11; } // FixItHint is done by MakeSmartPtrCheck Index: clang-tools-extra/clang-tidy/modernize/MakeUniqueCheck.cpp =================================================================== --- clang-tools-extra/clang-tidy/modernize/MakeUniqueCheck.cpp +++ clang-tools-extra/clang-tidy/modernize/MakeUniqueCheck.cpp @@ -16,7 +16,7 @@ MakeUniqueCheck::MakeUniqueCheck(StringRef Name, clang::tidy::ClangTidyContext *Context) - : MakeSmartPtrCheck(Name, Context, "std::make_unique"), + : MakeSmartPtrCheck(Name, Context, "std::make_unique", true), RequireCPlusPlus14(Options.get("MakeSmartPtrFunction", "").empty()) {} MakeUniqueCheck::SmartPtrTypeMatcher Index: clang-tools-extra/clang-tidy/modernize/MakeSmartPtrCheck.h =================================================================== --- clang-tools-extra/clang-tidy/modernize/MakeSmartPtrCheck.h +++ clang-tools-extra/clang-tidy/modernize/MakeSmartPtrCheck.h @@ -24,7 +24,7 @@ class MakeSmartPtrCheck : public ClangTidyCheck { public: MakeSmartPtrCheck(StringRef Name, ClangTidyContext *Context, - StringRef MakeSmartPtrFunctionName); + StringRef MakeSmartPtrFunctionName, bool ValueInitializes); void registerMatchers(ast_matchers::MatchFinder *Finder) final; void registerPPCallbacks(const SourceManager &SM, Preprocessor *PP, Preprocessor *ModuleExpanderPP) override; @@ -51,6 +51,7 @@ const std::string MakeSmartPtrFunctionName; const bool IgnoreMacros; const bool IgnoreDefaultInitialization; + const bool ValueInitializes; void checkConstruct(SourceManager &SM, ASTContext *Ctx, const CXXConstructExpr *Construct, const QualType *Type, Index: clang-tools-extra/clang-tidy/modernize/MakeSmartPtrCheck.cpp =================================================================== --- clang-tools-extra/clang-tidy/modernize/MakeSmartPtrCheck.cpp +++ clang-tools-extra/clang-tidy/modernize/MakeSmartPtrCheck.cpp @@ -42,7 +42,8 @@ const char MakeSmartPtrCheck::PointerType[] = "pointerType"; MakeSmartPtrCheck::MakeSmartPtrCheck(StringRef Name, ClangTidyContext *Context, - StringRef MakeSmartPtrFunctionName) + StringRef MakeSmartPtrFunctionName, + bool ValueInitializes) : ClangTidyCheck(Name, Context), Inserter(Options.getLocalOrGlobal("IncludeStyle", utils::IncludeSorter::IS_LLVM)), @@ -52,7 +53,9 @@ Options.get("MakeSmartPtrFunction", MakeSmartPtrFunctionName)), IgnoreMacros(Options.getLocalOrGlobal("IgnoreMacros", true)), IgnoreDefaultInitialization( - Options.get("IgnoreDefaultInitialization", true)) {} + !ValueInitializes || + Options.get("IgnoreDefaultInitialization", true)), + ValueInitializes(ValueInitializes) {} void MakeSmartPtrCheck::storeOptions(ClangTidyOptions::OptionMap &Opts) { Options.store(Opts, "IncludeStyle", Inserter.getStyle()); @@ -136,7 +139,7 @@ bool Initializes = New->hasInitializer() || !utils::type_traits::isTriviallyDefaultConstructible( New->getAllocatedType(), *Result.Context); - if (!Initializes && IgnoreDefaultInitialization) + if (Initializes != ValueInitializes && IgnoreDefaultInitialization) return; if (Construct) checkConstruct(SM, Result.Context, Construct, Type, New); Index: clang-tools-extra/clang-tidy/modernize/MakeSharedForOverwriteCheck.h =================================================================== --- /dev/null +++ clang-tools-extra/clang-tidy/modernize/MakeSharedForOverwriteCheck.h @@ -0,0 +1,44 @@ +//===--- MakeSharedForOverwriteCheck.h - clang-tidy -------------*- C++ -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_MAKESHAREDFOROVERWRITECHECK_H +#define LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_MAKESHAREDFOROVERWRITECHECK_H + +#include "MakeSmartPtrCheck.h" + +namespace clang { +namespace tidy { +namespace modernize { + +/// Replace the pattern: +/// \code +/// std::shared_ptr<type>(new type) +/// \endcode +/// +/// With the C++20 version: +/// \code +/// std::make_shared_for_overwrite<type>() +/// \endcode +class MakeSharedForOverwriteCheck : public MakeSmartPtrCheck { +public: + MakeSharedForOverwriteCheck(StringRef Name, ClangTidyContext *Context); + +protected: + SmartPtrTypeMatcher getSmartPointerTypeMatcher() const override; + + bool isLanguageVersionSupported(const LangOptions &LangOpts) const override; + +private: + const bool RequireCPlusPlus20; +}; + +} // namespace modernize +} // namespace tidy +} // namespace clang + +#endif // LLVM_CLANG_TOOLS_EXTRA_CLANG_TIDY_MODERNIZE_MAKESHAREDFOROVERWRITECHECK_H Index: clang-tools-extra/clang-tidy/modernize/MakeSharedForOverwriteCheck.cpp =================================================================== --- clang-tools-extra/clang-tidy/modernize/MakeSharedForOverwriteCheck.cpp +++ clang-tools-extra/clang-tidy/modernize/MakeSharedForOverwriteCheck.cpp @@ -1,4 +1,4 @@ -//===--- MakeSharedCheck.cpp - clang-tidy----------------------------------===// +//===--- MakeSharedForOverwriteCheck.cpp - clang-tidy ---------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. @@ -6,10 +6,9 @@ // //===----------------------------------------------------------------------===// -#include "MakeSharedCheck.h" - -// FixItHint - Hint to check documentation script to mark this check as -// providing a FixIt. +#include "MakeSharedForOverwriteCheck.h" +#include "clang/AST/ASTContext.h" +#include "clang/ASTMatchers/ASTMatchFinder.h" using namespace clang::ast_matchers; @@ -17,11 +16,13 @@ namespace tidy { namespace modernize { -MakeSharedCheck::MakeSharedCheck(StringRef Name, ClangTidyContext *Context) - : MakeSmartPtrCheck(Name, Context, "std::make_shared") {} +MakeSharedForOverwriteCheck::MakeSharedForOverwriteCheck( + StringRef Name, ClangTidyContext *Context) + : MakeSmartPtrCheck(Name, Context, "std::make_shared_for_overwrite", false), + RequireCPlusPlus20(Options.get("MakeSmartPtrFunction", "").empty()) {} -MakeSharedCheck::SmartPtrTypeMatcher -MakeSharedCheck::getSmartPointerTypeMatcher() const { +MakeSharedForOverwriteCheck::SmartPtrTypeMatcher +MakeSharedForOverwriteCheck::getSmartPointerTypeMatcher() const { return qualType(hasUnqualifiedDesugaredType( recordType(hasDeclaration(classTemplateSpecializationDecl( hasName("::std::shared_ptr"), templateArgumentCountIs(1), @@ -29,6 +30,11 @@ qualType().bind(PointerType))))))))); } +bool MakeSharedForOverwriteCheck::isLanguageVersionSupported( + const LangOptions &LangOpts) const { + return RequireCPlusPlus20 ? LangOpts.CPlusPlus20 : LangOpts.CPlusPlus11; +} + } // namespace modernize } // namespace tidy } // namespace clang Index: clang-tools-extra/clang-tidy/modernize/MakeSharedCheck.cpp =================================================================== --- clang-tools-extra/clang-tidy/modernize/MakeSharedCheck.cpp +++ clang-tools-extra/clang-tidy/modernize/MakeSharedCheck.cpp @@ -18,7 +18,7 @@ namespace modernize { MakeSharedCheck::MakeSharedCheck(StringRef Name, ClangTidyContext *Context) - : MakeSmartPtrCheck(Name, Context, "std::make_shared") {} + : MakeSmartPtrCheck(Name, Context, "std::make_shared", true) {} MakeSharedCheck::SmartPtrTypeMatcher MakeSharedCheck::getSmartPointerTypeMatcher() const { Index: clang-tools-extra/clang-tidy/modernize/CMakeLists.txt =================================================================== --- clang-tools-extra/clang-tidy/modernize/CMakeLists.txt +++ clang-tools-extra/clang-tidy/modernize/CMakeLists.txt @@ -12,8 +12,10 @@ LoopConvertCheck.cpp LoopConvertUtils.cpp MakeSharedCheck.cpp + MakeSharedForOverwriteCheck.cpp MakeSmartPtrCheck.cpp MakeUniqueCheck.cpp + MakeUniqueForOverwriteCheck.cpp ModernizeTidyModule.cpp PassByValueCheck.cpp RawStringLiteralCheck.cpp
_______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits