
Warn when a lambda specifies a default capture and captures ``this``. Offer FixIts to correct the code. Reviewed By: njames93, carlosgalvezp Differential Revision: https://reviews.llvm.org/D141133
99 lines
3.5 KiB
C++
99 lines
3.5 KiB
C++
//===--- AvoidCaptureDefaultWhenCapturingThisCheck.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.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "AvoidCaptureDefaultWhenCapturingThisCheck.h"
|
|
#include "../utils/LexerUtils.h"
|
|
#include "clang/AST/ASTContext.h"
|
|
#include "clang/ASTMatchers/ASTMatchFinder.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include <algorithm>
|
|
|
|
using namespace clang::ast_matchers;
|
|
|
|
namespace clang::tidy::cppcoreguidelines {
|
|
|
|
void AvoidCaptureDefaultWhenCapturingThisCheck::registerMatchers(
|
|
MatchFinder *Finder) {
|
|
Finder->addMatcher(lambdaExpr(hasAnyCapture(capturesThis())).bind("lambda"),
|
|
this);
|
|
}
|
|
|
|
static SourceLocation findDefaultCaptureEnd(const LambdaExpr *Lambda,
|
|
ASTContext &Context) {
|
|
for (const LambdaCapture &Capture : Lambda->explicit_captures()) {
|
|
if (Capture.isExplicit()) {
|
|
if (Capture.getCaptureKind() == LCK_ByRef) {
|
|
const SourceManager &SourceMgr = Context.getSourceManager();
|
|
SourceLocation AddressofLoc = utils::lexer::findPreviousTokenKind(
|
|
Capture.getLocation(), SourceMgr, Context.getLangOpts(), tok::amp);
|
|
return AddressofLoc;
|
|
} else {
|
|
return Capture.getLocation();
|
|
}
|
|
}
|
|
}
|
|
return Lambda->getIntroducerRange().getEnd();
|
|
}
|
|
|
|
static std::string createReplacementText(const LambdaExpr *Lambda) {
|
|
std::string Replacement;
|
|
llvm::raw_string_ostream Stream(Replacement);
|
|
|
|
auto AppendName = [&](llvm::StringRef Name) {
|
|
if (Replacement.size() != 0) {
|
|
Stream << ", ";
|
|
}
|
|
if (Lambda->getCaptureDefault() == LCD_ByRef && Name != "this") {
|
|
Stream << "&" << Name;
|
|
} else {
|
|
Stream << Name;
|
|
}
|
|
};
|
|
|
|
for (const LambdaCapture &Capture : Lambda->implicit_captures()) {
|
|
assert(Capture.isImplicit());
|
|
if (Capture.capturesVariable() && Capture.isImplicit()) {
|
|
AppendName(Capture.getCapturedVar()->getName());
|
|
} else if (Capture.capturesThis()) {
|
|
AppendName("this");
|
|
}
|
|
}
|
|
if (Replacement.size() &&
|
|
Lambda->explicit_capture_begin() != Lambda->explicit_capture_end()) {
|
|
// Add back separator if we are adding explicit capture variables.
|
|
Stream << ", ";
|
|
}
|
|
return Replacement;
|
|
}
|
|
|
|
void AvoidCaptureDefaultWhenCapturingThisCheck::check(
|
|
const MatchFinder::MatchResult &Result) {
|
|
if (const auto *Lambda = Result.Nodes.getNodeAs<LambdaExpr>("lambda")) {
|
|
if (Lambda->getCaptureDefault() != LCD_None) {
|
|
bool IsThisImplicitlyCaptured = std::any_of(
|
|
Lambda->implicit_capture_begin(), Lambda->implicit_capture_end(),
|
|
[](const LambdaCapture &Capture) { return Capture.capturesThis(); });
|
|
auto Diag = diag(Lambda->getCaptureDefaultLoc(),
|
|
"lambdas that %select{|implicitly }0capture 'this' "
|
|
"should not specify a capture default")
|
|
<< IsThisImplicitlyCaptured;
|
|
|
|
std::string ReplacementText = createReplacementText(Lambda);
|
|
SourceLocation DefaultCaptureEnd =
|
|
findDefaultCaptureEnd(Lambda, *Result.Context);
|
|
Diag << FixItHint::CreateReplacement(
|
|
CharSourceRange::getCharRange(Lambda->getCaptureDefaultLoc(),
|
|
DefaultCaptureEnd),
|
|
ReplacementText);
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace clang::tidy::cppcoreguidelines
|