[docs] Add LICENSE.txt to the root of the mono-repo
[llvm-project.git] / clang-tools-extra / clang-tidy / abseil / DurationUnnecessaryConversionCheck.cpp
blobb357d80f43624d1fa6003d44d9a823c93e329dcf
1 //===--- DurationUnnecessaryConversionCheck.cpp - clang-tidy
2 //-----------------------===//
3 //
4 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5 // See https://llvm.org/LICENSE.txt for license information.
6 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //
8 //===----------------------------------------------------------------------===//
10 #include "DurationUnnecessaryConversionCheck.h"
11 #include "DurationRewriter.h"
12 #include "clang/AST/ASTContext.h"
13 #include "clang/ASTMatchers/ASTMatchFinder.h"
14 #include "clang/Tooling/FixIt.h"
16 using namespace clang::ast_matchers;
18 namespace clang {
19 namespace tidy {
20 namespace abseil {
22 void DurationUnnecessaryConversionCheck::registerMatchers(MatchFinder *Finder) {
23 for (const auto &Scale : {"Hours", "Minutes", "Seconds", "Milliseconds",
24 "Microseconds", "Nanoseconds"}) {
25 std::string DurationFactory = (llvm::Twine("::absl::") + Scale).str();
26 std::string FloatConversion =
27 (llvm::Twine("::absl::ToDouble") + Scale).str();
28 std::string IntegerConversion =
29 (llvm::Twine("::absl::ToInt64") + Scale).str();
31 // Matcher which matches the current scale's factory with a `1` argument,
32 // e.g. `absl::Seconds(1)`.
33 auto FactoryMatcher = ignoringElidableConstructorCall(
34 callExpr(callee(functionDecl(hasName(DurationFactory))),
35 hasArgument(0, ignoringImpCasts(integerLiteral(equals(1))))));
37 // Matcher which matches either inverse function and binds its argument,
38 // e.g. `absl::ToDoubleSeconds(dur)`.
39 auto InverseFunctionMatcher = callExpr(
40 callee(functionDecl(hasAnyName(FloatConversion, IntegerConversion))),
41 hasArgument(0, expr().bind("arg")));
43 // Matcher which matches a duration divided by the factory_matcher above,
44 // e.g. `dur / absl::Seconds(1)`.
45 auto DivisionOperatorMatcher = cxxOperatorCallExpr(
46 hasOverloadedOperatorName("/"), hasArgument(0, expr().bind("arg")),
47 hasArgument(1, FactoryMatcher));
49 // Matcher which matches a duration argument to `FDivDuration`,
50 // e.g. `absl::FDivDuration(dur, absl::Seconds(1))`
51 auto FdivMatcher = callExpr(
52 callee(functionDecl(hasName("::absl::FDivDuration"))),
53 hasArgument(0, expr().bind("arg")), hasArgument(1, FactoryMatcher));
55 // Matcher which matches a duration argument being scaled,
56 // e.g. `absl::ToDoubleSeconds(dur) * 2`
57 auto ScalarMatcher = ignoringImpCasts(
58 binaryOperator(hasOperatorName("*"),
59 hasEitherOperand(expr(ignoringParenImpCasts(
60 callExpr(callee(functionDecl(hasAnyName(
61 FloatConversion, IntegerConversion))),
62 hasArgument(0, expr().bind("arg")))
63 .bind("inner_call")))))
64 .bind("binop"));
66 Finder->addMatcher(
67 callExpr(callee(functionDecl(hasName(DurationFactory))),
68 hasArgument(0, anyOf(InverseFunctionMatcher,
69 DivisionOperatorMatcher, FdivMatcher,
70 ScalarMatcher)))
71 .bind("call"),
72 this);
76 void DurationUnnecessaryConversionCheck::check(
77 const MatchFinder::MatchResult &Result) {
78 const auto *OuterCall = Result.Nodes.getNodeAs<Expr>("call");
80 if (isInMacro(Result, OuterCall))
81 return;
83 FixItHint Hint;
84 if (const auto *Binop = Result.Nodes.getNodeAs<BinaryOperator>("binop")) {
85 const auto *Arg = Result.Nodes.getNodeAs<Expr>("arg");
86 const auto *InnerCall = Result.Nodes.getNodeAs<Expr>("inner_call");
87 const Expr *LHS = Binop->getLHS();
88 const Expr *RHS = Binop->getRHS();
90 if (LHS->IgnoreParenImpCasts() == InnerCall) {
91 Hint = FixItHint::CreateReplacement(
92 OuterCall->getSourceRange(),
93 (llvm::Twine(tooling::fixit::getText(*Arg, *Result.Context)) + " * " +
94 tooling::fixit::getText(*RHS, *Result.Context))
95 .str());
96 } else {
97 assert(RHS->IgnoreParenImpCasts() == InnerCall &&
98 "Inner call should be find on the RHS");
100 Hint = FixItHint::CreateReplacement(
101 OuterCall->getSourceRange(),
102 (llvm::Twine(tooling::fixit::getText(*LHS, *Result.Context)) + " * " +
103 tooling::fixit::getText(*Arg, *Result.Context))
104 .str());
106 } else if (const auto *Arg = Result.Nodes.getNodeAs<Expr>("arg")) {
107 Hint = FixItHint::CreateReplacement(
108 OuterCall->getSourceRange(),
109 tooling::fixit::getText(*Arg, *Result.Context));
111 diag(OuterCall->getBeginLoc(),
112 "remove unnecessary absl::Duration conversions")
113 << Hint;
116 } // namespace abseil
117 } // namespace tidy
118 } // namespace clang