1aba8f320SVy Nguyen //===--- AssertEquals.cpp - clang-tidy --------------------------*- C++ -*-===//
2aba8f320SVy Nguyen //
3aba8f320SVy Nguyen // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4aba8f320SVy Nguyen // See https://llvm.org/LICENSE.txt for license information.
5aba8f320SVy Nguyen // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6aba8f320SVy Nguyen //
7aba8f320SVy Nguyen //===----------------------------------------------------------------------===//
8aba8f320SVy Nguyen
9aba8f320SVy Nguyen #include "AssertEquals.h"
10aba8f320SVy Nguyen
11aba8f320SVy Nguyen #include <map>
12aba8f320SVy Nguyen #include <string>
13aba8f320SVy Nguyen
14aba8f320SVy Nguyen using namespace clang::ast_matchers;
15aba8f320SVy Nguyen
16*7d2ea6c4SCarlos Galvez namespace clang::tidy::objc {
17aba8f320SVy Nguyen
18aba8f320SVy Nguyen // Mapping from `XCTAssert*Equal` to `XCTAssert*EqualObjects` name.
NameMap()19aba8f320SVy Nguyen static const std::map<std::string, std::string> &NameMap() {
20aba8f320SVy Nguyen static std::map<std::string, std::string> map{
21aba8f320SVy Nguyen {"XCTAssertEqual", "XCTAssertEqualObjects"},
22aba8f320SVy Nguyen {"XCTAssertNotEqual", "XCTAssertNotEqualObjects"},
23aba8f320SVy Nguyen
24aba8f320SVy Nguyen };
25aba8f320SVy Nguyen return map;
26aba8f320SVy Nguyen }
27aba8f320SVy Nguyen
registerMatchers(MatchFinder * finder)28aba8f320SVy Nguyen void AssertEquals::registerMatchers(MatchFinder *finder) {
29aba8f320SVy Nguyen for (const auto &pair : NameMap()) {
30aba8f320SVy Nguyen finder->addMatcher(
31aba8f320SVy Nguyen binaryOperator(anyOf(hasOperatorName("!="), hasOperatorName("==")),
32aba8f320SVy Nguyen isExpandedFromMacro(pair.first),
33aba8f320SVy Nguyen anyOf(hasLHS(hasType(qualType(
34aba8f320SVy Nguyen hasCanonicalType(asString("NSString *"))))),
35aba8f320SVy Nguyen hasRHS(hasType(qualType(
36aba8f320SVy Nguyen hasCanonicalType(asString("NSString *"))))))
37aba8f320SVy Nguyen
38aba8f320SVy Nguyen )
39aba8f320SVy Nguyen .bind(pair.first),
40aba8f320SVy Nguyen this);
41aba8f320SVy Nguyen }
42aba8f320SVy Nguyen }
43aba8f320SVy Nguyen
check(const ast_matchers::MatchFinder::MatchResult & result)44aba8f320SVy Nguyen void AssertEquals::check(const ast_matchers::MatchFinder::MatchResult &result) {
45aba8f320SVy Nguyen for (const auto &pair : NameMap()) {
46aba8f320SVy Nguyen if (const auto *root = result.Nodes.getNodeAs<BinaryOperator>(pair.first)) {
47aba8f320SVy Nguyen SourceManager *sm = result.SourceManager;
48aba8f320SVy Nguyen // The macros are nested two levels, so going up twice.
49aba8f320SVy Nguyen auto macro_callsite = sm->getImmediateMacroCallerLoc(
50aba8f320SVy Nguyen sm->getImmediateMacroCallerLoc(root->getBeginLoc()));
51aba8f320SVy Nguyen diag(macro_callsite, "use " + pair.second + " for comparing objects")
52aba8f320SVy Nguyen << FixItHint::CreateReplacement(
53aba8f320SVy Nguyen clang::CharSourceRange::getCharRange(
54aba8f320SVy Nguyen macro_callsite,
55aba8f320SVy Nguyen macro_callsite.getLocWithOffset(pair.first.length())),
56aba8f320SVy Nguyen pair.second);
57aba8f320SVy Nguyen }
58aba8f320SVy Nguyen }
59aba8f320SVy Nguyen }
60aba8f320SVy Nguyen
61*7d2ea6c4SCarlos Galvez } // namespace clang::tidy::objc
62