forked from OSchip/llvm-project
418 lines
16 KiB
C++
418 lines
16 KiB
C++
// RUN: %check_clang_tidy %s modernize-make-unique %t -- -- -std=c++11 \
|
|
// RUN: -I%S/Inputs/modernize-smart-ptr
|
|
|
|
#include "unique_ptr.h"
|
|
#include "initializer_list.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 {};
|
|
|
|
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);
|
|
};
|
|
|
|
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);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:10: warning: use std::make_unique instead
|
|
// CHECK-FIXES: return std::make_unique<Base>();
|
|
}
|
|
|
|
void basic() {
|
|
std::unique_ptr<int> P1 = std::unique_ptr<int>(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:29: warning: use std::make_unique instead [modernize-make-unique]
|
|
// CHECK-FIXES: std::unique_ptr<int> P1 = std::make_unique<int>();
|
|
|
|
P1.reset(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_unique instead [modernize-make-unique]
|
|
// CHECK-FIXES: P1 = std::make_unique<int>();
|
|
|
|
P1 = std::unique_ptr<int>(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:8: warning: use std::make_unique instead [modernize-make-unique]
|
|
// CHECK-FIXES: P1 = std::make_unique<int>();
|
|
|
|
// Without parenthesis.
|
|
std::unique_ptr<int> P2 = std::unique_ptr<int>(new int);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:29: warning: use std::make_unique instead [modernize-make-unique]
|
|
// CHECK-FIXES: std::unique_ptr<int> P2 = std::make_unique<int>();
|
|
|
|
P2.reset(new int);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_unique instead [modernize-make-unique]
|
|
// CHECK-FIXES: P2 = std::make_unique<int>();
|
|
|
|
P2 = std::unique_ptr<int>(new int);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:8: warning: use std::make_unique instead [modernize-make-unique]
|
|
// CHECK-FIXES: P2 = std::make_unique<int>();
|
|
|
|
// With auto.
|
|
auto P3 = std::unique_ptr<int>(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:13: warning: use std::make_unique instead
|
|
// CHECK-FIXES: auto P3 = std::make_unique<int>();
|
|
|
|
{
|
|
// No std.
|
|
using namespace std;
|
|
unique_ptr<int> Q = unique_ptr<int>(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:25: warning: use std::make_unique instead
|
|
// CHECK-FIXES: unique_ptr<int> Q = std::make_unique<int>();
|
|
|
|
Q = unique_ptr<int>(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_unique instead
|
|
// CHECK-FIXES: Q = std::make_unique<int>();
|
|
}
|
|
|
|
std::unique_ptr<int> R(new int());
|
|
|
|
// Create the unique_ptr as a parameter to a function.
|
|
int T = g(std::unique_ptr<int>(new int()));
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:13: warning: use std::make_unique instead
|
|
// CHECK-FIXES: int T = g(std::make_unique<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());
|
|
|
|
// OK to replace for reset and assign
|
|
Pderived.reset(new Derived());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: use std::make_unique instead
|
|
// CHECK-FIXES: Pderived = std::make_unique<Derived>();
|
|
|
|
Pderived = std::unique_ptr<Derived>(new Derived());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: use std::make_unique instead
|
|
// CHECK-FIXES: Pderived = std::make_unique<Derived>();
|
|
|
|
// FIXME: OK to replace if assigned to unique_ptr<Base>
|
|
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());
|
|
|
|
// The pointer is returned by the function, nothing to do.
|
|
std::unique_ptr<Base> RetPtr = getPointer();
|
|
|
|
// 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});
|
|
}
|
|
|
|
// 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);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:24: warning: use std::make_unique instead
|
|
// CHECK-FIXES: auto callsPublic = std::make_unique<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));
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:24: warning: use std::make_unique instead
|
|
// CHECK-FIXES: auto callsPublic = std::make_unique<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));
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:34: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<DPair> PDir1 = std::make_unique<DPair>(1, T);
|
|
PDir1.reset(new DPair(1, T));
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_unique instead
|
|
// CHECK-FIXES: PDir1 = std::make_unique<DPair>(1, T);
|
|
|
|
// Direct initialization with braces.
|
|
std::unique_ptr<DPair> PDir2 = std::unique_ptr<DPair>(new DPair{2, T});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:34: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<DPair> PDir2 = std::make_unique<DPair>(2, T);
|
|
PDir2.reset(new DPair{2, T});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_unique instead
|
|
// CHECK-FIXES: PDir2 = std::make_unique<DPair>(2, T);
|
|
|
|
// Aggregate initialization.
|
|
std::unique_ptr<APair> PAggr = std::unique_ptr<APair>(new APair{T, 1});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:34: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<APair> PAggr = std::make_unique<APair>(APair{T, 1});
|
|
PAggr.reset(new APair{T, 1});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::make_unique<APair>(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)};
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:34: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<DPair> PDir3 = std::make_unique<DPair>(3, T);
|
|
|
|
// Direct initialization with braces.
|
|
std::unique_ptr<DPair> PDir4 = std::unique_ptr<DPair>{new DPair{4, T}};
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:34: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<DPair> PDir4 = std::make_unique<DPair>(4, T);
|
|
|
|
// Aggregate initialization.
|
|
std::unique_ptr<APair> PAggr2 = std::unique_ptr<APair>{new APair{T, 2}};
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:35: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<APair> PAggr2 = std::make_unique<APair>(APair{T, 2});
|
|
|
|
// Direct initialization with parenthesis, without arguments.
|
|
std::unique_ptr<DPair> PDir5 = std::unique_ptr<DPair>(new DPair());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:34: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<DPair> PDir5 = std::make_unique<DPair>();
|
|
|
|
// Direct initialization with braces, without arguments.
|
|
std::unique_ptr<DPair> PDir6 = std::unique_ptr<DPair>(new DPair{});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:34: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<DPair> PDir6 = std::make_unique<DPair>();
|
|
|
|
// Aggregate initialization without arguments.
|
|
std::unique_ptr<Empty> PEmpty = std::unique_ptr<Empty>(new Empty{});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:35: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<Empty> PEmpty = std::make_unique<Empty>(Empty{});
|
|
|
|
// Initialization with default constructor.
|
|
std::unique_ptr<E> PE1 = std::unique_ptr<E>(new E{});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<E> PE1 = std::make_unique<E>();
|
|
|
|
// Initialization with the initializer-list constructor.
|
|
std::unique_ptr<E> PE2 = std::unique_ptr<E>(new E{1, 2});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<E> PE2 = std::make_unique<E>({1, 2});
|
|
|
|
// Initialization with default constructor.
|
|
std::unique_ptr<F> PF1 = std::unique_ptr<F>(new F());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<F> PF1 = std::make_unique<F>();
|
|
|
|
// Initialization with default constructor.
|
|
std::unique_ptr<F> PF2 = std::unique_ptr<F>(new F{});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<F> PF2 = std::make_unique<F>();
|
|
|
|
// Initialization with the initializer-list constructor.
|
|
std::unique_ptr<F> PF3 = std::unique_ptr<F>(new F{1});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<F> PF3 = std::make_unique<F>({1});
|
|
|
|
// Initialization with the initializer-list constructor.
|
|
std::unique_ptr<F> PF4 = std::unique_ptr<F>(new F{1, 2});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<F> PF4 = std::make_unique<F>({1, 2});
|
|
|
|
// Initialization with the initializer-list constructor.
|
|
std::unique_ptr<F> PF5 = std::unique_ptr<F>(new F({1, 2}));
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<F> PF5 = std::make_unique<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{});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<G> PG1 = std::make_unique<G>({});
|
|
|
|
// Initialization with the initializer-list constructor.
|
|
std::unique_ptr<G> PG2 = std::unique_ptr<G>(new G{1});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<G> PG2 = std::make_unique<G>({1});
|
|
|
|
// Initialization with the initializer-list constructor.
|
|
std::unique_ptr<G> PG3 = std::unique_ptr<G>(new G{1, 2});
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:28: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<G> PG3 = std::make_unique<G>({1, 2});
|
|
|
|
std::unique_ptr<Foo> FF = std::unique_ptr<Foo>(new Foo());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:29: warning:
|
|
// CHECK-FIXES: std::unique_ptr<Foo> FF = std::make_unique<Foo>();
|
|
FF.reset(new Foo());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning:
|
|
// CHECK-FIXES: FF = std::make_unique<Foo>();
|
|
|
|
std::unique_ptr<bar::Bar> BB = std::unique_ptr<bar::Bar>(new bar::Bar());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:34: warning:
|
|
// CHECK-FIXES: std::unique_ptr<bar::Bar> BB = std::make_unique<bar::Bar>();
|
|
BB.reset(new bar::Bar());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning:
|
|
// CHECK-FIXES: BB = std::make_unique<bar::Bar>();
|
|
|
|
std::unique_ptr<Foo[]> FFs;
|
|
FFs.reset(new Foo[5]);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:7: warning:
|
|
// CHECK-FIXES: FFs = std::make_unique<Foo[]>(5);
|
|
FFs.reset(new Foo[5]());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:7: warning:
|
|
// CHECK-FIXES: FFs = std::make_unique<Foo[]>(5);
|
|
const int Num = 1;
|
|
FFs.reset(new Foo[Num]);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:7: warning:
|
|
// CHECK-FIXES: FFs = std::make_unique<Foo[]>(Num);
|
|
int Num2 = 1;
|
|
FFs.reset(new Foo[Num2]);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:7: warning:
|
|
// CHECK-FIXES: FFs = std::make_unique<Foo[]>(Num2);
|
|
|
|
std::unique_ptr<int[]> FI;
|
|
FI.reset(new int[5]);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning:
|
|
// CHECK-FIXES: FI = std::make_unique<int[]>(5);
|
|
FI.reset(new int[5]());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning:
|
|
// CHECK-FIXES: FI = std::make_unique<int[]>(5);
|
|
FI.reset(new int[Num]);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning:
|
|
// CHECK-FIXES: FI = std::make_unique<int[]>(Num);
|
|
FI.reset(new int[Num2]);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning:
|
|
// CHECK-FIXES: FI = std::make_unique<int[]>(Num2);
|
|
}
|
|
|
|
void aliases() {
|
|
typedef std::unique_ptr<int> IntPtr;
|
|
IntPtr Typedef = IntPtr(new int);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:20: warning: use std::make_unique instead
|
|
// CHECK-FIXES: IntPtr Typedef = std::make_unique<int>();
|
|
|
|
// We use 'bool' instead of '_Bool'.
|
|
typedef std::unique_ptr<bool> BoolPtr;
|
|
BoolPtr BoolType = BoolPtr(new bool);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:22: warning: use std::make_unique instead
|
|
// CHECK-FIXES: BoolPtr BoolType = std::make_unique<bool>();
|
|
|
|
// We use 'Base' instead of 'struct Base'.
|
|
typedef std::unique_ptr<Base> BasePtr;
|
|
BasePtr StructType = BasePtr(new Base);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:24: warning: use std::make_unique instead
|
|
// CHECK-FIXES: BasePtr StructType = std::make_unique<Base>();
|
|
|
|
#define PTR unique_ptr<int>
|
|
std::unique_ptr<int> Macro = std::PTR(new int);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:32: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<int> Macro = std::make_unique<int>();
|
|
#undef PTR
|
|
|
|
std::unique_ptr<int> Using = unique_ptr_<int>(new int);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:32: warning: use std::make_unique instead
|
|
// CHECK-FIXES: std::unique_ptr<int> Using = std::make_unique<int>();
|
|
}
|
|
|
|
void whitespaces() {
|
|
// clang-format off
|
|
auto Space = std::unique_ptr <int>(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:16: warning: use std::make_unique instead
|
|
// CHECK-FIXES: auto Space = std::make_unique<int>();
|
|
|
|
auto Spaces = std :: unique_ptr <int>(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:17: warning: use std::make_unique instead
|
|
// CHECK-FIXES: auto Spaces = std::make_unique<int>();
|
|
// clang-format on
|
|
}
|
|
|
|
void nesting() {
|
|
auto Nest = std::unique_ptr<std::unique_ptr<int>>(new std::unique_ptr<int>(new int));
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:15: warning: use std::make_unique instead
|
|
// CHECK-FIXES: auto Nest = std::make_unique<std::unique_ptr<int>>(new int);
|
|
Nest.reset(new std::unique_ptr<int>(new int));
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:8: warning: use std::make_unique instead
|
|
// CHECK-FIXES: Nest = std::make_unique<std::unique_ptr<int>>(new int);
|
|
Nest->reset(new int);
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: use std::make_unique instead
|
|
// CHECK-FIXES: *Nest = std::make_unique<int>();
|
|
}
|
|
|
|
void reset() {
|
|
std::unique_ptr<int> P;
|
|
P.reset();
|
|
P.reset(nullptr);
|
|
P.reset(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:5: warning: use std::make_unique instead
|
|
// CHECK-FIXES: P = std::make_unique<int>();
|
|
|
|
auto Q = &P;
|
|
Q->reset(new int());
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: use std::make_unique instead
|
|
// CHECK-FIXES: *Q = std::make_unique<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
|