11#include "clang/AST/ASTContext.h"
12#include "clang/AST/ExprCXX.h"
13#include "clang/AST/Type.h"
14#include "clang/Sema/DeclSpec.h"
15#include "clang/Tooling/FixIt.h"
21 SourceLocation AmpLocation = Var.getLocation();
23 AmpLocation, Context.getSourceManager(), Context.getLangOpts());
26 if (Token.is(tok::ellipsis))
27 return FixItHint::CreateInsertion(Token.getLocation(),
"&");
29 if (!Token.is(tok::unknown))
30 AmpLocation = Lexer::getLocForEndOfToken(Token.getLocation(), 0,
31 Context.getSourceManager(),
32 Context.getLangOpts());
33 return FixItHint::CreateInsertion(AmpLocation,
"&");
37 return !(isa<PointerType>(T) || isa<ReferenceType>(T) || isa<ArrayType>(T) ||
38 isa<MemberPointerType>(T) || isa<ObjCObjectPointerType>(T));
42 return (QT->isPointerType() && QT->isFunctionPointerType()) ||
43 isa<MemberPointerType>(QT.getTypePtr());
47 return S.isInvalid() || S.isMacroID();
50static std::optional<SourceLocation>
55 auto PreviousTokenLParen = [&Start, &Context]() {
58 Context.getLangOpts());
59 return T.is(tok::l_paren);
62 while (Start.isValid() && PreviousTokenLParen())
64 Context.getLangOpts());
75 return FixItHint::CreateInsertion(Loc, Text);
81 bool WhitespaceBefore =
false) {
83 return (llvm::Twine(
' ') + Qualifiers::fromCVRMask(Qualifier).getAsString())
85 return (llvm::Twine(Qualifiers::fromCVRMask(Qualifier).getAsString()) +
" ")
89static std::optional<FixItHint>
changeValue(
const VarDecl &Var,
90 Qualifiers::TQ Qualifier,
93 const ASTContext &Context) {
99 std::optional<SourceLocation> IgnoredParens =
106 llvm_unreachable(
"Unknown QualifierPolicy enum");
110 Qualifiers::TQ Qualifier,
111 const ASTContext &Context) {
115 std::optional<SourceLocation> IgnoredParens =
122static std::optional<FixItHint>
125 const ASTContext &Context) {
144 Var.getLocation(), Context.getSourceManager(), Context.getLangOpts(),
149 std::optional<SourceLocation> IgnoredParens =
165 Var.getLocation(), Context.getSourceManager(), Context.getLangOpts(),
173static std::optional<FixItHint>
176 const ASTContext &Context) {
182 Var.getLocation(), Context.getSourceManager(), Context.getLangOpts(),
183 tok::amp, tok::ampamp);
184 std::optional<SourceLocation> IgnoredParens =
193 const ASTContext &Context,
194 Qualifiers::TQ Qualifier,
199 "Unexpected Insertion Policy");
202 "Unexpected Target");
204 const QualType ParenStrippedType = Var.getType().IgnoreParens();
206 return changeValue(Var, Qualifier, QualTarget, QualPolicy, Context);
208 if (ParenStrippedType->isReferenceType())
210 QualTarget, QualPolicy, Context);
215 if (ParenStrippedType->isPointerType())
217 ParenStrippedType->getPointeeType().getTypePtr(),
218 QualTarget, QualPolicy, Context);
220 if (ParenStrippedType->isArrayType()) {
221 const Type *AT = ParenStrippedType->getBaseElementTypeUnsafe();
222 assert(AT &&
"Did not retrieve array element type for an array.");
225 return changeValue(Var, Qualifier, QualTarget, QualPolicy, Context);
227 if (AT->isPointerType())
228 return changePointer(Var, Qualifier, AT->getPointeeType().getTypePtr(),
229 QualTarget, QualPolicy, Context);
236 if (isa<ParenExpr>(&Node))
239 if (isa<clang::BinaryOperator>(&Node) || isa<UnaryOperator>(&Node))
242 if (isa<clang::ConditionalOperator>(&Node) ||
243 isa<BinaryConditionalOperator>(&Node))
246 if (
const auto *Op = dyn_cast<CXXOperatorCallExpr>(&Node)) {
247 switch (Op->getOperator()) {
251 return Op->getNumArgs() != 2;
263 if (isa<CStyleCastExpr>(&Node))
273 if (isa<clang::BinaryOperator>(&ExprNode) ||
274 isa<clang::ConditionalOperator>(&ExprNode)) {
277 if (
const auto *Op = dyn_cast<CXXOperatorCallExpr>(&ExprNode)) {
278 return Op->getNumArgs() == 2 && Op->getOperator() != OO_PlusPlus &&
279 Op->getOperator() != OO_MinusMinus && Op->getOperator() != OO_Call &&
280 Op->getOperator() != OO_Subscript;
288 if (
const auto *Op = dyn_cast<clang::UnaryOperator>(&ExprNode)) {
289 if (Op->getOpcode() == UO_AddrOf) {
292 tooling::fixit::getText(*Op->getSubExpr()->IgnoreParens(), Context));
295 StringRef Text = tooling::fixit::getText(ExprNode, Context);
301 Text.consume_back(
"->");
305 return (llvm::Twine(
"*(") + Text +
")").str();
307 return (llvm::Twine(
"*") + Text).str();
static std::string buildQualifier(Qualifiers::TQ Qualifier, bool WhitespaceBefore=false)
bool areParensNeededForStatement(const Stmt &Node)
static std::optional< FixItHint > changePointer(const VarDecl &Var, Qualifiers::TQ Qualifier, const Type *Pointee, QualifierTarget QualTarget, QualifierPolicy QualPolicy, const ASTContext &Context)
static std::optional< FixItHint > changePointerItself(const VarDecl &Var, Qualifiers::TQ Qualifier, const ASTContext &Context)
static std::optional< FixItHint > fixIfNotDangerous(SourceLocation Loc, StringRef Text)
static std::optional< FixItHint > changeValue(const VarDecl &Var, Qualifiers::TQ Qualifier, QualifierTarget QualTarget, QualifierPolicy QualPolicy, const ASTContext &Context)
static bool isMemberOrFunctionPointer(QualType QT)
QualifierTarget
This enum defines which entity is the target for adding the qualifier. This makes only a difference f...
@ Value
Transforming a pointer attaches to the pointee and not the pointer itself. For references and normal ...
static std::optional< FixItHint > changeReferencee(const VarDecl &Var, Qualifiers::TQ Qualifier, QualType Pointee, QualifierTarget QualTarget, QualifierPolicy QualPolicy, const ASTContext &Context)
QualifierPolicy
This enum defines where the qualifier shall be preferably added.
FixItHint changeVarDeclToReference(const VarDecl &Var, ASTContext &Context)
Creates fix to make VarDecl a reference by adding &.
std::optional< FixItHint > addQualifierToVarDecl(const VarDecl &Var, const ASTContext &Context, Qualifiers::TQ Qualifier, QualifierTarget QualTarget, QualifierPolicy QualPolicy)
Creates fix to qualify VarDecl with the specified Qualifier. Requires that Var is isolated in written...
static std::optional< SourceLocation > skipLParensBackwards(SourceLocation Start, const ASTContext &Context)
static bool needParensAfterUnaryOperator(const Expr &ExprNode)
static bool locDangerous(SourceLocation S)
std::string formatDereference(const Expr &ExprNode, const ASTContext &Context)
static bool isValueType(const Type *T)
Token getPreviousToken(SourceLocation Location, const SourceManager &SM, const LangOptions &LangOpts, bool SkipComments)
Returns previous token or tok::unknown if not found.
SourceLocation findPreviousTokenStart(SourceLocation Start, const SourceManager &SM, const LangOptions &LangOpts)
SourceLocation findPreviousTokenKind(SourceLocation Start, const SourceManager &SM, const LangOptions &LangOpts, tok::TokenKind TK)
SourceLocation findPreviousAnyTokenKind(SourceLocation Start, const SourceManager &SM, const LangOptions &LangOpts, TokenKind TK, TokenKinds... TKs)