mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-01-01 13:20:25 +00:00
[clang-tidy] remove duplicate fixes of alias checkers
when both a check and its alias are enabled, we should only take the fixes of one of them and not both. This patch fixes bug 45577 https://bugs.llvm.org/show_bug.cgi?id=45577 Reviewed By: aaron.ballman, njames93 Differential Revision: https://reviews.llvm.org/D80753
This commit is contained in:
parent
216a37bb46
commit
af4f2eb476
@ -122,6 +122,8 @@ public:
|
|||||||
{
|
{
|
||||||
auto Level = static_cast<DiagnosticsEngine::Level>(Error.DiagLevel);
|
auto Level = static_cast<DiagnosticsEngine::Level>(Error.DiagLevel);
|
||||||
std::string Name = Error.DiagnosticName;
|
std::string Name = Error.DiagnosticName;
|
||||||
|
if (!Error.EnabledDiagnosticAliases.empty())
|
||||||
|
Name += "," + llvm::join(Error.EnabledDiagnosticAliases, ",");
|
||||||
if (Error.IsWarningAsError) {
|
if (Error.IsWarningAsError) {
|
||||||
Name += ",-warnings-as-errors";
|
Name += ",-warnings-as-errors";
|
||||||
Level = DiagnosticsEngine::Error;
|
Level = DiagnosticsEngine::Error;
|
||||||
|
@ -27,6 +27,7 @@
|
|||||||
#include "llvm/ADT/STLExtras.h"
|
#include "llvm/ADT/STLExtras.h"
|
||||||
#include "llvm/ADT/SmallString.h"
|
#include "llvm/ADT/SmallString.h"
|
||||||
#include "llvm/Support/Regex.h"
|
#include "llvm/Support/Regex.h"
|
||||||
|
#include "llvm/Support/FormatVariadic.h"
|
||||||
#include <tuple>
|
#include <tuple>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
using namespace clang;
|
using namespace clang;
|
||||||
@ -634,6 +635,8 @@ void ClangTidyDiagnosticConsumer::removeIncompatibleErrors() {
|
|||||||
std::tuple<unsigned, EventType, int, int, unsigned> Priority;
|
std::tuple<unsigned, EventType, int, int, unsigned> Priority;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
removeDuplicatedDiagnosticsOfAliasCheckers();
|
||||||
|
|
||||||
// Compute error sizes.
|
// Compute error sizes.
|
||||||
std::vector<int> Sizes;
|
std::vector<int> Sizes;
|
||||||
std::vector<
|
std::vector<
|
||||||
@ -728,3 +731,59 @@ std::vector<ClangTidyError> ClangTidyDiagnosticConsumer::take() {
|
|||||||
removeIncompatibleErrors();
|
removeIncompatibleErrors();
|
||||||
return std::move(Errors);
|
return std::move(Errors);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
struct LessClangTidyErrorWithoutDiagnosticName {
|
||||||
|
bool operator()(const ClangTidyError *LHS, const ClangTidyError *RHS) const {
|
||||||
|
const tooling::DiagnosticMessage &M1 = LHS->Message;
|
||||||
|
const tooling::DiagnosticMessage &M2 = RHS->Message;
|
||||||
|
|
||||||
|
return std::tie(M1.FilePath, M1.FileOffset, M1.Message) <
|
||||||
|
std::tie(M2.FilePath, M2.FileOffset, M2.Message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} // end anonymous namespace
|
||||||
|
|
||||||
|
void ClangTidyDiagnosticConsumer::removeDuplicatedDiagnosticsOfAliasCheckers() {
|
||||||
|
using UniqueErrorSet =
|
||||||
|
std::set<ClangTidyError *, LessClangTidyErrorWithoutDiagnosticName>;
|
||||||
|
UniqueErrorSet UniqueErrors;
|
||||||
|
|
||||||
|
auto IT = Errors.begin();
|
||||||
|
while (IT != Errors.end()) {
|
||||||
|
ClangTidyError &Error = *IT;
|
||||||
|
std::pair<UniqueErrorSet::iterator, bool> Inserted =
|
||||||
|
UniqueErrors.insert(&Error);
|
||||||
|
|
||||||
|
// Unique error, we keep it and move along.
|
||||||
|
if (Inserted.second) {
|
||||||
|
++IT;
|
||||||
|
} else {
|
||||||
|
ClangTidyError &ExistingError = **Inserted.first;
|
||||||
|
const llvm::StringMap<tooling::Replacements> &CandidateFix =
|
||||||
|
Error.Message.Fix;
|
||||||
|
const llvm::StringMap<tooling::Replacements> &ExistingFix =
|
||||||
|
(*Inserted.first)->Message.Fix;
|
||||||
|
|
||||||
|
if (CandidateFix != ExistingFix) {
|
||||||
|
|
||||||
|
// In case of a conflict, don't suggest any fix-it.
|
||||||
|
ExistingError.Message.Fix.clear();
|
||||||
|
ExistingError.Notes.emplace_back(
|
||||||
|
llvm::formatv("cannot apply fix-it because an alias checker has "
|
||||||
|
"suggested a different fix-it; please remove one of "
|
||||||
|
"the checkers ('{0}', '{1}') or "
|
||||||
|
"ensure they are both configured the same",
|
||||||
|
ExistingError.DiagnosticName, Error.DiagnosticName)
|
||||||
|
.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Error.IsWarningAsError)
|
||||||
|
ExistingError.IsWarningAsError = true;
|
||||||
|
|
||||||
|
// Since it is the same error, we should take it as alias and remove it.
|
||||||
|
ExistingError.EnabledDiagnosticAliases.emplace_back(Error.DiagnosticName);
|
||||||
|
IT = Errors.erase(IT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -48,6 +48,7 @@ struct ClangTidyError : tooling::Diagnostic {
|
|||||||
bool IsWarningAsError);
|
bool IsWarningAsError);
|
||||||
|
|
||||||
bool IsWarningAsError;
|
bool IsWarningAsError;
|
||||||
|
std::vector<std::string> EnabledDiagnosticAliases;
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Contains displayed and ignored diagnostic counters for a ClangTidy
|
/// Contains displayed and ignored diagnostic counters for a ClangTidy
|
||||||
@ -246,6 +247,7 @@ public:
|
|||||||
private:
|
private:
|
||||||
void finalizeLastError();
|
void finalizeLastError();
|
||||||
void removeIncompatibleErrors();
|
void removeIncompatibleErrors();
|
||||||
|
void removeDuplicatedDiagnosticsOfAliasCheckers();
|
||||||
|
|
||||||
/// Returns the \c HeaderFilter constructed for the options set in the
|
/// Returns the \c HeaderFilter constructed for the options set in the
|
||||||
/// context.
|
/// context.
|
||||||
|
@ -0,0 +1,23 @@
|
|||||||
|
// RUN: %check_clang_tidy %s cppcoreguidelines-pro-type-member-init,hicpp-member-init,modernize-use-emplace,hicpp-use-emplace %t -- \
|
||||||
|
//// RUN: -config='{CheckOptions: [ \
|
||||||
|
//// RUN: {key: cppcoreguidelines-pro-type-member-init.UseAssignment, value: 1}, \
|
||||||
|
//// RUN: ]}'
|
||||||
|
|
||||||
|
class Foo {
|
||||||
|
public:
|
||||||
|
Foo() : _num1(0)
|
||||||
|
// CHECK-MESSAGES: warning: constructor does not initialize these fields: _num2 [cppcoreguidelines-pro-type-member-init,hicpp-member-init]
|
||||||
|
// CHECK-MESSAGES: note: cannot apply fix-it because an alias checker has suggested a different fix-it; please remove one of the checkers ('cppcoreguidelines-pro-type-member-init', 'hicpp-member-init') or ensure they are both configured the same
|
||||||
|
{
|
||||||
|
_num1 = 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
int use_the_members() const {
|
||||||
|
return _num1 + _num2;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
int _num1;
|
||||||
|
int _num2;
|
||||||
|
// CHECK-FIXES: _num2;
|
||||||
|
};
|
@ -0,0 +1,39 @@
|
|||||||
|
// RUN: %check_clang_tidy %s cppcoreguidelines-pro-type-member-init,hicpp-member-init,modernize-use-emplace,hicpp-use-emplace %t
|
||||||
|
|
||||||
|
namespace std {
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
class vector {
|
||||||
|
public:
|
||||||
|
void push_back(const T &) {}
|
||||||
|
void push_back(T &&) {}
|
||||||
|
|
||||||
|
template <typename... Args>
|
||||||
|
void emplace_back(Args &&... args){};
|
||||||
|
};
|
||||||
|
} // namespace std
|
||||||
|
|
||||||
|
class Foo {
|
||||||
|
public:
|
||||||
|
Foo() : _num1(0)
|
||||||
|
// CHECK-MESSAGES: warning: constructor does not initialize these fields: _num2 [cppcoreguidelines-pro-type-member-init,hicpp-member-init]
|
||||||
|
{
|
||||||
|
_num1 = 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
int use_the_members() const {
|
||||||
|
return _num1 + _num2;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
int _num1;
|
||||||
|
int _num2;
|
||||||
|
// CHECK-FIXES: _num2{};
|
||||||
|
};
|
||||||
|
|
||||||
|
int should_use_emplace(std::vector<Foo> &v) {
|
||||||
|
v.push_back(Foo());
|
||||||
|
// CHECK-FIXES: v.emplace_back();
|
||||||
|
// CHECK-MESSAGES: warning: use emplace_back instead of push_back [hicpp-use-emplace,modernize-use-emplace]
|
||||||
|
}
|
||||||
|
|
@ -2,8 +2,7 @@
|
|||||||
|
|
||||||
void alwaysThrows() {
|
void alwaysThrows() {
|
||||||
int ex = 42;
|
int ex = 42;
|
||||||
// CHECK-MESSAGES: warning: throw expression should throw anonymous temporary values instead [cert-err09-cpp]
|
// CHECK-MESSAGES: warning: throw expression should throw anonymous temporary values instead [cert-err09-cpp,cert-err61-cpp]
|
||||||
// CHECK-MESSAGES: warning: throw expression should throw anonymous temporary values instead [cert-err61-cpp]
|
|
||||||
throw ex;
|
throw ex;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -248,6 +248,26 @@ public:
|
|||||||
return count(MapEntry.getKey());
|
return count(MapEntry.getKey());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// equal - check whether both of the containers are equal.
|
||||||
|
bool operator==(const StringMap &RHS) const {
|
||||||
|
if (size() != RHS.size())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
for (const auto &KeyValue : *this) {
|
||||||
|
auto FindInRHS = RHS.find(KeyValue.getKey());
|
||||||
|
|
||||||
|
if (FindInRHS == RHS.end())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!(KeyValue.getValue() == FindInRHS->getValue()))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool operator!=(const StringMap &RHS) const { return !(*this == RHS); }
|
||||||
|
|
||||||
/// insert - Insert the specified key/value pair into the map. If the key
|
/// insert - Insert the specified key/value pair into the map. If the key
|
||||||
/// already exists in the map, return false and ignore the request, otherwise
|
/// already exists in the map, return false and ignore the request, otherwise
|
||||||
/// insert it and return true.
|
/// insert it and return true.
|
||||||
|
@ -387,6 +387,70 @@ TEST_F(StringMapTest, MoveAssignment) {
|
|||||||
ASSERT_EQ(B.count("x"), 0u);
|
ASSERT_EQ(B.count("x"), 0u);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_F(StringMapTest, EqualEmpty) {
|
||||||
|
StringMap<int> A;
|
||||||
|
StringMap<int> B;
|
||||||
|
ASSERT_TRUE(A == B);
|
||||||
|
ASSERT_FALSE(A != B);
|
||||||
|
ASSERT_TRUE(A == A); // self check
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(StringMapTest, EqualWithValues) {
|
||||||
|
StringMap<int> A;
|
||||||
|
A["A"] = 1;
|
||||||
|
A["B"] = 2;
|
||||||
|
A["C"] = 3;
|
||||||
|
A["D"] = 3;
|
||||||
|
|
||||||
|
StringMap<int> B;
|
||||||
|
B["A"] = 1;
|
||||||
|
B["B"] = 2;
|
||||||
|
B["C"] = 3;
|
||||||
|
B["D"] = 3;
|
||||||
|
|
||||||
|
ASSERT_TRUE(A == B);
|
||||||
|
ASSERT_TRUE(B == A);
|
||||||
|
ASSERT_FALSE(A != B);
|
||||||
|
ASSERT_FALSE(B != A);
|
||||||
|
ASSERT_TRUE(A == A); // self check
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(StringMapTest, NotEqualMissingKeys) {
|
||||||
|
StringMap<int> A;
|
||||||
|
A["A"] = 1;
|
||||||
|
A["B"] = 2;
|
||||||
|
|
||||||
|
StringMap<int> B;
|
||||||
|
B["A"] = 1;
|
||||||
|
B["B"] = 2;
|
||||||
|
B["C"] = 3;
|
||||||
|
B["D"] = 3;
|
||||||
|
|
||||||
|
ASSERT_FALSE(A == B);
|
||||||
|
ASSERT_FALSE(B == A);
|
||||||
|
ASSERT_TRUE(A != B);
|
||||||
|
ASSERT_TRUE(B != A);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(StringMapTest, NotEqualWithDifferentValues) {
|
||||||
|
StringMap<int> A;
|
||||||
|
A["A"] = 1;
|
||||||
|
A["B"] = 2;
|
||||||
|
A["C"] = 100;
|
||||||
|
A["D"] = 3;
|
||||||
|
|
||||||
|
StringMap<int> B;
|
||||||
|
B["A"] = 1;
|
||||||
|
B["B"] = 2;
|
||||||
|
B["C"] = 3;
|
||||||
|
B["D"] = 3;
|
||||||
|
|
||||||
|
ASSERT_FALSE(A == B);
|
||||||
|
ASSERT_FALSE(B == A);
|
||||||
|
ASSERT_TRUE(A != B);
|
||||||
|
ASSERT_TRUE(B != A);
|
||||||
|
}
|
||||||
|
|
||||||
struct Countable {
|
struct Countable {
|
||||||
int &InstanceCount;
|
int &InstanceCount;
|
||||||
int Number;
|
int Number;
|
||||||
|
Loading…
Reference in New Issue
Block a user