2019-05-20 17:26:47 +08:00
|
|
|
// RUN: %check_clang_tidy %s readability-redundant-string-cstr %t
|
2016-04-16 02:12:06 +08:00
|
|
|
|
|
|
|
typedef unsigned __INT16_TYPE__ char16;
|
|
|
|
typedef unsigned __INT32_TYPE__ char32;
|
|
|
|
typedef __SIZE_TYPE__ size;
|
2015-03-16 08:32:25 +08:00
|
|
|
|
|
|
|
namespace std {
|
|
|
|
template <typename T>
|
|
|
|
class allocator {};
|
|
|
|
template <typename T>
|
|
|
|
class char_traits {};
|
|
|
|
template <typename C, typename T, typename A>
|
|
|
|
struct basic_string {
|
2016-04-16 02:12:06 +08:00
|
|
|
typedef basic_string<C, T, A> _Type;
|
2015-03-16 08:32:25 +08:00
|
|
|
basic_string();
|
|
|
|
basic_string(const C *p, const A &a = A());
|
2016-04-16 02:12:06 +08:00
|
|
|
|
2020-07-29 22:35:28 +08:00
|
|
|
~basic_string();
|
|
|
|
|
2015-03-16 08:32:25 +08:00
|
|
|
const C *c_str() const;
|
2016-11-03 20:56:48 +08:00
|
|
|
const C *data() const;
|
2016-04-16 02:12:06 +08:00
|
|
|
|
|
|
|
_Type& append(const C *s);
|
|
|
|
_Type& append(const C *s, size n);
|
|
|
|
_Type& assign(const C *s);
|
|
|
|
_Type& assign(const C *s, size n);
|
|
|
|
|
|
|
|
int compare(const _Type&) const;
|
|
|
|
int compare(const C* s) const;
|
|
|
|
int compare(size pos, size len, const _Type&) const;
|
|
|
|
int compare(size pos, size len, const C* s) const;
|
|
|
|
|
|
|
|
size find(const _Type& str, size pos = 0) const;
|
|
|
|
size find(const C* s, size pos = 0) const;
|
|
|
|
size find(const C* s, size pos, size n) const;
|
|
|
|
|
|
|
|
_Type& insert(size pos, const _Type& str);
|
|
|
|
_Type& insert(size pos, const C* s);
|
|
|
|
_Type& insert(size pos, const C* s, size n);
|
|
|
|
|
|
|
|
_Type& operator+=(const _Type& str);
|
|
|
|
_Type& operator+=(const C* s);
|
|
|
|
_Type& operator=(const _Type& str);
|
|
|
|
_Type& operator=(const C* s);
|
2015-03-16 08:32:25 +08:00
|
|
|
};
|
2016-04-16 02:12:06 +08:00
|
|
|
|
2015-03-16 08:32:25 +08:00
|
|
|
typedef basic_string<char, std::char_traits<char>, std::allocator<char>> string;
|
2016-03-25 03:42:36 +08:00
|
|
|
typedef basic_string<wchar_t, std::char_traits<wchar_t>, std::allocator<wchar_t>> wstring;
|
2016-04-16 02:12:06 +08:00
|
|
|
typedef basic_string<char16, std::char_traits<char16>, std::allocator<char16>> u16string;
|
|
|
|
typedef basic_string<char32, std::char_traits<char32>, std::allocator<char32>> u32string;
|
2015-03-16 08:32:25 +08:00
|
|
|
}
|
2016-03-25 03:42:36 +08:00
|
|
|
|
2016-04-16 02:12:06 +08:00
|
|
|
std::string operator+(const std::string&, const std::string&);
|
|
|
|
std::string operator+(const std::string&, const char*);
|
|
|
|
std::string operator+(const char*, const std::string&);
|
|
|
|
|
|
|
|
bool operator==(const std::string&, const std::string&);
|
|
|
|
bool operator==(const std::string&, const char*);
|
|
|
|
bool operator==(const char*, const std::string&);
|
|
|
|
|
2015-03-16 08:32:25 +08:00
|
|
|
namespace llvm {
|
|
|
|
struct StringRef {
|
|
|
|
StringRef(const char *p);
|
|
|
|
StringRef(const std::string &);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-03-25 03:42:36 +08:00
|
|
|
// Tests for std::string.
|
|
|
|
|
2015-03-16 08:32:25 +08:00
|
|
|
void f1(const std::string &s) {
|
|
|
|
f1(s.c_str());
|
2016-11-03 20:56:48 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: redundant call to 'c_str' [readability-redundant-string-cstr]
|
|
|
|
// CHECK-FIXES: {{^ }}f1(s);{{$}}
|
|
|
|
f1(s.data());
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: redundant call to 'data' [readability-redundant-string-cstr]
|
2015-03-16 08:32:25 +08:00
|
|
|
// CHECK-FIXES: {{^ }}f1(s);{{$}}
|
|
|
|
}
|
|
|
|
void f2(const llvm::StringRef r) {
|
|
|
|
std::string s;
|
|
|
|
f2(s.c_str());
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}std::string s;{{$}}
|
|
|
|
// CHECK-FIXES-NEXT: {{^ }}f2(s);{{$}}
|
|
|
|
}
|
|
|
|
void f3(const llvm::StringRef &r) {
|
|
|
|
std::string s;
|
|
|
|
f3(s.c_str());
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}std::string s;{{$}}
|
|
|
|
// CHECK-FIXES-NEXT: {{^ }}f3(s);{{$}}
|
|
|
|
}
|
2016-03-25 03:42:36 +08:00
|
|
|
void f4(const std::string &s) {
|
|
|
|
const std::string* ptr = &s;
|
|
|
|
f1(ptr->c_str());
|
2016-11-03 20:56:48 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: redundant call to 'c_str' [readability-redundant-string-cstr]
|
2016-03-25 03:42:36 +08:00
|
|
|
// CHECK-FIXES: {{^ }}f1(*ptr);{{$}}
|
|
|
|
}
|
2016-04-16 02:12:06 +08:00
|
|
|
void f5(const std::string &s) {
|
|
|
|
std::string tmp;
|
|
|
|
tmp.append(s.c_str());
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}tmp.append(s);{{$}}
|
|
|
|
tmp.assign(s.c_str());
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}tmp.assign(s);{{$}}
|
|
|
|
|
|
|
|
if (tmp.compare(s.c_str()) == 0) return;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:19: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}if (tmp.compare(s) == 0) return;{{$}}
|
|
|
|
|
|
|
|
if (tmp.compare(1, 2, s.c_str()) == 0) return;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:25: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}if (tmp.compare(1, 2, s) == 0) return;{{$}}
|
|
|
|
|
|
|
|
if (tmp.find(s.c_str()) == 0) return;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:16: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}if (tmp.find(s) == 0) return;{{$}}
|
|
|
|
|
|
|
|
if (tmp.find(s.c_str(), 2) == 0) return;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:16: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}if (tmp.find(s, 2) == 0) return;{{$}}
|
|
|
|
|
|
|
|
if (tmp.find(s.c_str(), 2) == 0) return;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:16: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}if (tmp.find(s, 2) == 0) return;{{$}}
|
|
|
|
|
|
|
|
tmp.insert(1, s.c_str());
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:17: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}tmp.insert(1, s);{{$}}
|
|
|
|
|
|
|
|
tmp = s.c_str();
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}tmp = s;{{$}}
|
|
|
|
|
|
|
|
tmp += s.c_str();
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:10: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}tmp += s;{{$}}
|
|
|
|
|
|
|
|
if (tmp == s.c_str()) return;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:14: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}if (tmp == s) return;{{$}}
|
|
|
|
|
|
|
|
tmp = s + s.c_str();
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:13: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}tmp = s + s;{{$}}
|
|
|
|
|
|
|
|
tmp = s.c_str() + s;
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:9: warning: redundant call {{.*}}
|
|
|
|
// CHECK-FIXES: {{^ }}tmp = s + s;{{$}}
|
|
|
|
}
|
|
|
|
void f6(const std::string &s) {
|
|
|
|
std::string tmp;
|
|
|
|
tmp.append(s.c_str(), 2);
|
|
|
|
tmp.assign(s.c_str(), 2);
|
|
|
|
|
|
|
|
if (tmp.compare(s) == 0) return;
|
|
|
|
if (tmp.compare(1, 2, s) == 0) return;
|
|
|
|
|
|
|
|
tmp = s;
|
|
|
|
tmp += s;
|
|
|
|
|
|
|
|
if (tmp == s)
|
|
|
|
return;
|
|
|
|
|
|
|
|
tmp = s + s;
|
|
|
|
|
|
|
|
if (tmp.find(s.c_str(), 2, 4) == 0) return;
|
|
|
|
|
|
|
|
tmp.insert(1, s);
|
|
|
|
tmp.insert(1, s.c_str(), 2);
|
|
|
|
}
|
2016-03-25 03:42:36 +08:00
|
|
|
|
|
|
|
// Tests for std::wstring.
|
|
|
|
|
|
|
|
void g1(const std::wstring &s) {
|
|
|
|
g1(s.c_str());
|
2016-11-03 20:56:48 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: redundant call to 'c_str' [readability-redundant-string-cstr]
|
2016-03-25 03:42:36 +08:00
|
|
|
// CHECK-FIXES: {{^ }}g1(s);{{$}}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tests for std::u16string.
|
|
|
|
|
|
|
|
void h1(const std::u16string &s) {
|
|
|
|
h1(s.c_str());
|
2016-11-03 20:56:48 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: redundant call to 'c_str' [readability-redundant-string-cstr]
|
2016-03-25 03:42:36 +08:00
|
|
|
// CHECK-FIXES: {{^ }}h1(s);{{$}}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tests for std::u32string.
|
|
|
|
|
|
|
|
void k1(const std::u32string &s) {
|
|
|
|
k1(s.c_str());
|
2016-11-03 20:56:48 +08:00
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:6: warning: redundant call to 'c_str' [readability-redundant-string-cstr]
|
2016-03-25 03:42:36 +08:00
|
|
|
// CHECK-FIXES: {{^ }}k1(s);{{$}}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tests on similar classes that aren't good candidates for this checker.
|
|
|
|
|
|
|
|
struct NotAString {
|
|
|
|
NotAString();
|
|
|
|
NotAString(const NotAString&);
|
|
|
|
const char *c_str() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
void dummy(const char*) {}
|
|
|
|
|
|
|
|
void invalid(const NotAString &s) {
|
|
|
|
dummy(s.c_str());
|
|
|
|
}
|
2020-02-19 04:32:03 +08:00
|
|
|
|
|
|
|
// Test for rvalue std::string.
|
|
|
|
void m1(std::string&&) {
|
|
|
|
std::string s;
|
|
|
|
|
|
|
|
m1(s.c_str());
|
|
|
|
|
|
|
|
void (*m1p1)(std::string&&);
|
|
|
|
m1p1 = m1;
|
|
|
|
m1p1(s.c_str());
|
|
|
|
|
|
|
|
using m1tp = void (*)(std::string &&);
|
|
|
|
m1tp m1p2 = m1;
|
|
|
|
m1p2(s.c_str());
|
|
|
|
}
|
2020-03-26 03:22:09 +08:00
|
|
|
|
|
|
|
namespace PR45286 {
|
|
|
|
struct Foo {
|
|
|
|
void func(const std::string &) {}
|
|
|
|
void func2(std::string &&) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
void bar() {
|
|
|
|
std::string Str{"aaa"};
|
|
|
|
Foo Foo;
|
|
|
|
Foo.func(Str.c_str());
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:12: warning: redundant call to 'c_str' [readability-redundant-string-cstr]
|
|
|
|
// CHECK-FIXES: {{^ }}Foo.func(Str);{{$}}
|
|
|
|
|
|
|
|
// Ensure it doesn't transform Binding to r values
|
|
|
|
Foo.func2(Str.c_str());
|
|
|
|
|
|
|
|
// Ensure its not confused by parens
|
|
|
|
Foo.func((Str.c_str()));
|
|
|
|
// CHECK-MESSAGES: :[[@LINE-1]]:13: warning: redundant call to 'c_str' [readability-redundant-string-cstr]
|
|
|
|
// CHECK-FIXES: {{^ }}Foo.func((Str));{{$}}
|
|
|
|
Foo.func2((Str.c_str()));
|
|
|
|
}
|
|
|
|
} // namespace PR45286
|