xref: /llvm-project/clang-tools-extra/test/clang-tidy/checkers/bugprone/string-constructor.cpp (revision 2463a0ed70754524efb6e05b09d632dcacf9b51f)
189a1d03eSRichard // RUN: %check_clang_tidy %s bugprone-string-constructor %t
289a1d03eSRichard 
389a1d03eSRichard namespace std {
489a1d03eSRichard template <typename T>
589a1d03eSRichard class allocator {};
689a1d03eSRichard template <typename T>
789a1d03eSRichard class char_traits {};
889a1d03eSRichard template <typename C, typename T = std::char_traits<C>, typename A = std::allocator<C> >
989a1d03eSRichard struct basic_string {
1089a1d03eSRichard   basic_string();
1189a1d03eSRichard   basic_string(const C*, unsigned int size);
1289a1d03eSRichard   basic_string(const C *, const A &allocator = A());
1389a1d03eSRichard   basic_string(unsigned int size, C c);
1489a1d03eSRichard };
1589a1d03eSRichard typedef basic_string<char> string;
1689a1d03eSRichard typedef basic_string<wchar_t> wstring;
1789a1d03eSRichard 
1889a1d03eSRichard template <typename C, typename T = std::char_traits<C>>
1989a1d03eSRichard struct basic_string_view {
2089a1d03eSRichard   basic_string_view();
2189a1d03eSRichard   basic_string_view(const C *, unsigned int size);
2289a1d03eSRichard   basic_string_view(const C *);
2389a1d03eSRichard };
2489a1d03eSRichard typedef basic_string_view<char> string_view;
2589a1d03eSRichard typedef basic_string_view<wchar_t> wstring_view;
2689a1d03eSRichard }
2789a1d03eSRichard 
2889a1d03eSRichard const char* kText = "";
2989a1d03eSRichard const char kText2[] = "";
3089a1d03eSRichard extern const char kText3[];
3189a1d03eSRichard 
Test()3289a1d03eSRichard void Test() {
3389a1d03eSRichard   std::string str('x', 4);
3489a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: string constructor parameters are probably swapped; expecting string(count, character) [bugprone-string-constructor]
3589a1d03eSRichard   // CHECK-FIXES: std::string str(4, 'x');
3689a1d03eSRichard   std::wstring wstr(L'x', 4);
3789a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:16: warning: string constructor parameters are probably swapped
3889a1d03eSRichard   // CHECK-FIXES: std::wstring wstr(4, L'x');
3989a1d03eSRichard   std::string s0(0, 'x');
4089a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: constructor creating an empty string
4189a1d03eSRichard   std::string s1(-4, 'x');
4289a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: negative value used as length parameter
4389a1d03eSRichard   std::string s2(0x1000000, 'x');
4489a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: suspicious large length parameter
4589a1d03eSRichard 
4689a1d03eSRichard   std::string q0("test", 0);
4789a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: constructor creating an empty string
4889a1d03eSRichard   std::string q1(kText, -4);
4989a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: negative value used as length parameter
5089a1d03eSRichard   std::string q2("test", 200);
5189a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: length is bigger than string literal size
52*2463a0edS   std::string t1("test", 5);
53*2463a0edS   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: length is bigger than string literal size
5489a1d03eSRichard   std::string q3(kText, 200);
5589a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: length is bigger than string literal size
5689a1d03eSRichard   std::string q4(kText2, 200);
5789a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: length is bigger than string literal size
5889a1d03eSRichard   std::string q5(kText3,  0x1000000);
5989a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: suspicious large length parameter
6089a1d03eSRichard   std::string q6(nullptr);
6189a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:15: warning: constructing string from nullptr is undefined behaviour
6289a1d03eSRichard   std::string q7 = 0;
6389a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:20: warning: constructing string from nullptr is undefined behaviour
6489a1d03eSRichard }
6589a1d03eSRichard 
TestView()6689a1d03eSRichard void TestView() {
6789a1d03eSRichard   std::string_view q0("test", 0);
6889a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:20: warning: constructor creating an empty string
6989a1d03eSRichard   std::string_view q1(kText, -4);
7089a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:20: warning: negative value used as length parameter
7189a1d03eSRichard   std::string_view q2("test", 200);
7289a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:20: warning: length is bigger than string literal size
7389a1d03eSRichard   std::string_view q3(kText, 200);
7489a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:20: warning: length is bigger than string literal size
7589a1d03eSRichard   std::string_view q4(kText2, 200);
7689a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:20: warning: length is bigger than string literal size
7789a1d03eSRichard   std::string_view q5(kText3, 0x1000000);
7889a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:20: warning: suspicious large length parameter
7989a1d03eSRichard   std::string_view q6(nullptr);
8089a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:20: warning: constructing string from nullptr is undefined behaviour
8189a1d03eSRichard   std::string_view q7 = 0;
8289a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:25: warning: constructing string from nullptr is undefined behaviour
8389a1d03eSRichard }
8489a1d03eSRichard 
StringFromZero()8589a1d03eSRichard std::string StringFromZero() {
8689a1d03eSRichard   return 0;
8789a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:10: warning: constructing string from nullptr is undefined behaviour
8889a1d03eSRichard }
8989a1d03eSRichard 
StringViewFromZero()9089a1d03eSRichard std::string_view StringViewFromZero() {
9189a1d03eSRichard   return 0;
9289a1d03eSRichard   // CHECK-MESSAGES: [[@LINE-1]]:10: warning: constructing string from nullptr is undefined behaviour
9389a1d03eSRichard }
9489a1d03eSRichard 
Valid()9589a1d03eSRichard void Valid() {
9689a1d03eSRichard   std::string empty();
9789a1d03eSRichard   std::string str(4, 'x');
9889a1d03eSRichard   std::wstring wstr(4, L'x');
9989a1d03eSRichard   std::string s1("test", 4);
10089a1d03eSRichard   std::string s2("test", 3);
10189a1d03eSRichard   std::string s3("test");
102*2463a0edS   std::string s4("test\000", 5);
103*2463a0edS   std::string s6("te" "st", 4);
10489a1d03eSRichard 
10589a1d03eSRichard   std::string_view emptyv();
10689a1d03eSRichard   std::string_view sv1("test", 4);
10789a1d03eSRichard   std::string_view sv2("test", 3);
10889a1d03eSRichard   std::string_view sv3("test");
10989a1d03eSRichard }
11089a1d03eSRichard 
11189a1d03eSRichard namespace instantiation_dependent_exprs {
11289a1d03eSRichard template<typename T>
11389a1d03eSRichard struct S {
11489a1d03eSRichard   bool x;
finstantiation_dependent_exprs::S11589a1d03eSRichard   std::string f() { return x ? "a" : "b"; }
ginstantiation_dependent_exprs::S11689a1d03eSRichard   std::string_view g() { return x ? "a" : "b"; }
11789a1d03eSRichard };
11889a1d03eSRichard }
119