forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			1085 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			1085 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			C++
		
	
	
	
//===- unittest/Tooling/ASTSelectionTest.cpp ------------------------------===//
 | 
						|
//
 | 
						|
// 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 "TestVisitor.h"
 | 
						|
#include "clang/Basic/SourceManager.h"
 | 
						|
#include "clang/Tooling/Refactoring/ASTSelection.h"
 | 
						|
 | 
						|
using namespace clang;
 | 
						|
using namespace tooling;
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
struct FileLocation {
 | 
						|
  unsigned Line, Column;
 | 
						|
 | 
						|
  SourceLocation translate(const SourceManager &SM) {
 | 
						|
    return SM.translateLineCol(SM.getMainFileID(), Line, Column);
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
using FileRange = std::pair<FileLocation, FileLocation>;
 | 
						|
 | 
						|
class SelectionFinderVisitor : public TestVisitor<SelectionFinderVisitor> {
 | 
						|
  FileLocation Location;
 | 
						|
  Optional<FileRange> SelectionRange;
 | 
						|
  llvm::function_ref<void(SourceRange SelectionRange,
 | 
						|
                          Optional<SelectedASTNode>)>
 | 
						|
      Consumer;
 | 
						|
 | 
						|
public:
 | 
						|
  SelectionFinderVisitor(FileLocation Location,
 | 
						|
                         Optional<FileRange> SelectionRange,
 | 
						|
                         llvm::function_ref<void(SourceRange SelectionRange,
 | 
						|
                                                 Optional<SelectedASTNode>)>
 | 
						|
                             Consumer)
 | 
						|
      : Location(Location), SelectionRange(SelectionRange), Consumer(Consumer) {
 | 
						|
  }
 | 
						|
 | 
						|
  bool VisitTranslationUnitDecl(const TranslationUnitDecl *TU) {
 | 
						|
    const ASTContext &Context = TU->getASTContext();
 | 
						|
    const SourceManager &SM = Context.getSourceManager();
 | 
						|
 | 
						|
    SourceRange SelRange;
 | 
						|
    if (SelectionRange) {
 | 
						|
      SelRange = SourceRange(SelectionRange->first.translate(SM),
 | 
						|
                             SelectionRange->second.translate(SM));
 | 
						|
    } else {
 | 
						|
      SourceLocation Loc = Location.translate(SM);
 | 
						|
      SelRange = SourceRange(Loc, Loc);
 | 
						|
    }
 | 
						|
    Consumer(SelRange, findSelectedASTNodes(Context, SelRange));
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
/// This is a test utility function that computes the AST selection at the
 | 
						|
/// given location with an optional selection range.
 | 
						|
///
 | 
						|
/// A location roughly corresponds to a cursor location in an editor, while
 | 
						|
/// the optional range corresponds to the selection range in an editor.
 | 
						|
void findSelectedASTNodesWithRange(
 | 
						|
    StringRef Source, FileLocation Location, Optional<FileRange> SelectionRange,
 | 
						|
    llvm::function_ref<void(SourceRange SelectionRange,
 | 
						|
                            Optional<SelectedASTNode>)>
 | 
						|
        Consumer,
 | 
						|
    SelectionFinderVisitor::Language Language =
 | 
						|
        SelectionFinderVisitor::Lang_CXX11) {
 | 
						|
  SelectionFinderVisitor Visitor(Location, SelectionRange, Consumer);
 | 
						|
  EXPECT_TRUE(Visitor.runOver(Source, Language));
 | 
						|
}
 | 
						|
 | 
						|
void findSelectedASTNodes(
 | 
						|
    StringRef Source, FileLocation Location, Optional<FileRange> SelectionRange,
 | 
						|
    llvm::function_ref<void(Optional<SelectedASTNode>)> Consumer,
 | 
						|
    SelectionFinderVisitor::Language Language =
 | 
						|
        SelectionFinderVisitor::Lang_CXX11) {
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, Location, SelectionRange,
 | 
						|
      [&](SourceRange, Optional<SelectedASTNode> Selection) {
 | 
						|
        Consumer(std::move(Selection));
 | 
						|
      },
 | 
						|
      Language);
 | 
						|
}
 | 
						|
 | 
						|
void checkNodeImpl(bool IsTypeMatched, const SelectedASTNode &Node,
 | 
						|
                   SourceSelectionKind SelectionKind, unsigned NumChildren) {
 | 
						|
  ASSERT_TRUE(IsTypeMatched);
 | 
						|
  EXPECT_EQ(Node.Children.size(), NumChildren);
 | 
						|
  ASSERT_EQ(Node.SelectionKind, SelectionKind);
 | 
						|
}
 | 
						|
 | 
						|
void checkDeclName(const SelectedASTNode &Node, StringRef Name) {
 | 
						|
  const auto *ND = Node.Node.get<NamedDecl>();
 | 
						|
  EXPECT_TRUE(!!ND);
 | 
						|
  ASSERT_EQ(ND->getName(), Name);
 | 
						|
}
 | 
						|
 | 
						|
template <typename T>
 | 
						|
const SelectedASTNode &checkNode(
 | 
						|
    const SelectedASTNode &StmtNode, SourceSelectionKind SelectionKind,
 | 
						|
    unsigned NumChildren = 0,
 | 
						|
    std::enable_if_t<std::is_base_of<Stmt, T>::value, T> *StmtOverloadChecker =
 | 
						|
        nullptr) {
 | 
						|
  checkNodeImpl(isa<T>(StmtNode.Node.get<Stmt>()), StmtNode, SelectionKind,
 | 
						|
                NumChildren);
 | 
						|
  return StmtNode;
 | 
						|
}
 | 
						|
 | 
						|
template <typename T>
 | 
						|
const SelectedASTNode &checkNode(
 | 
						|
    const SelectedASTNode &DeclNode, SourceSelectionKind SelectionKind,
 | 
						|
    unsigned NumChildren = 0, StringRef Name = "",
 | 
						|
    std::enable_if_t<std::is_base_of<Decl, T>::value, T> *DeclOverloadChecker =
 | 
						|
        nullptr) {
 | 
						|
  checkNodeImpl(isa<T>(DeclNode.Node.get<Decl>()), DeclNode, SelectionKind,
 | 
						|
                NumChildren);
 | 
						|
  if (!Name.empty())
 | 
						|
    checkDeclName(DeclNode, Name);
 | 
						|
  return DeclNode;
 | 
						|
}
 | 
						|
 | 
						|
struct ForAllChildrenOf {
 | 
						|
  const SelectedASTNode &Node;
 | 
						|
 | 
						|
  static void childKindVerifier(const SelectedASTNode &Node,
 | 
						|
                                SourceSelectionKind SelectionKind) {
 | 
						|
    for (const SelectedASTNode &Child : Node.Children) {
 | 
						|
      ASSERT_EQ(Node.SelectionKind, SelectionKind);
 | 
						|
      childKindVerifier(Child, SelectionKind);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
public:
 | 
						|
  ForAllChildrenOf(const SelectedASTNode &Node) : Node(Node) {}
 | 
						|
 | 
						|
  void shouldHaveSelectionKind(SourceSelectionKind Kind) {
 | 
						|
    childKindVerifier(Node, Kind);
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
ForAllChildrenOf allChildrenOf(const SelectedASTNode &Node) {
 | 
						|
  return ForAllChildrenOf(Node);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, CursorNoSelection) {
 | 
						|
  findSelectedASTNodes(
 | 
						|
      " void f() { }", {1, 1}, None,
 | 
						|
      [](Optional<SelectedASTNode> Node) { EXPECT_FALSE(Node); });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, CursorAtStartOfFunction) {
 | 
						|
  findSelectedASTNodes(
 | 
						|
      "void f() { }", {1, 1}, None, [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        checkNode<TranslationUnitDecl>(*Node, SourceSelectionKind::None,
 | 
						|
                                       /*NumChildren=*/1);
 | 
						|
        checkNode<FunctionDecl>(Node->Children[0],
 | 
						|
                                SourceSelectionKind::ContainsSelection,
 | 
						|
                                /*NumChildren=*/0, /*Name=*/"f");
 | 
						|
 | 
						|
        // Check that the dumping works.
 | 
						|
        std::string DumpValue;
 | 
						|
        llvm::raw_string_ostream OS(DumpValue);
 | 
						|
        Node->Children[0].dump(OS);
 | 
						|
        ASSERT_EQ(OS.str(), "FunctionDecl \"f\" contains-selection\n");
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, RangeNoSelection) {
 | 
						|
  findSelectedASTNodes(
 | 
						|
      " void f() { }", {1, 1}, FileRange{{1, 1}, {1, 1}},
 | 
						|
      [](Optional<SelectedASTNode> Node) { EXPECT_FALSE(Node); });
 | 
						|
  findSelectedASTNodes(
 | 
						|
      "  void f() { }", {1, 1}, FileRange{{1, 1}, {1, 2}},
 | 
						|
      [](Optional<SelectedASTNode> Node) { EXPECT_FALSE(Node); });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, EmptyRangeFallbackToCursor) {
 | 
						|
  findSelectedASTNodes("void f() { }", {1, 1}, FileRange{{1, 1}, {1, 1}},
 | 
						|
                       [](Optional<SelectedASTNode> Node) {
 | 
						|
                         EXPECT_TRUE(Node);
 | 
						|
                         checkNode<FunctionDecl>(
 | 
						|
                             Node->Children[0],
 | 
						|
                             SourceSelectionKind::ContainsSelection,
 | 
						|
                             /*NumChildren=*/0, /*Name=*/"f");
 | 
						|
                       });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, WholeFunctionSelection) {
 | 
						|
  StringRef Source = "int f(int x) { return x;\n}\nvoid f2() { }";
 | 
						|
  // From 'int' until just after '}':
 | 
						|
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {1, 1}, FileRange{{1, 1}, {2, 2}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/2, /*Name=*/"f");
 | 
						|
        checkNode<ParmVarDecl>(Fn.Children[0],
 | 
						|
                               SourceSelectionKind::InsideSelection);
 | 
						|
        const auto &Body = checkNode<CompoundStmt>(
 | 
						|
            Fn.Children[1], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &Return = checkNode<ReturnStmt>(
 | 
						|
            Body.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<ImplicitCastExpr>(Return.Children[0],
 | 
						|
                                    SourceSelectionKind::InsideSelection,
 | 
						|
                                    /*NumChildren=*/1);
 | 
						|
        checkNode<DeclRefExpr>(Return.Children[0].Children[0],
 | 
						|
                               SourceSelectionKind::InsideSelection);
 | 
						|
      });
 | 
						|
 | 
						|
  // From 'int' until just before '}':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {2, 1}, FileRange{{1, 1}, {2, 1}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/2, /*Name=*/"f");
 | 
						|
        const auto &Body = checkNode<CompoundStmt>(
 | 
						|
            Fn.Children[1], SourceSelectionKind::ContainsSelectionEnd,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<ReturnStmt>(Body.Children[0],
 | 
						|
                              SourceSelectionKind::InsideSelection,
 | 
						|
                              /*NumChildren=*/1);
 | 
						|
      });
 | 
						|
  // From '{' until just after '}':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {1, 14}, FileRange{{1, 14}, {2, 2}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"f");
 | 
						|
        const auto &Body = checkNode<CompoundStmt>(
 | 
						|
            Fn.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<ReturnStmt>(Body.Children[0],
 | 
						|
                              SourceSelectionKind::InsideSelection,
 | 
						|
                              /*NumChildren=*/1);
 | 
						|
      });
 | 
						|
  // From 'x' until just after '}':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {2, 2}, FileRange{{1, 11}, {2, 2}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/2, /*Name=*/"f");
 | 
						|
        checkNode<ParmVarDecl>(Fn.Children[0],
 | 
						|
                               SourceSelectionKind::ContainsSelectionStart);
 | 
						|
        const auto &Body = checkNode<CompoundStmt>(
 | 
						|
            Fn.Children[1], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<ReturnStmt>(Body.Children[0],
 | 
						|
                              SourceSelectionKind::InsideSelection,
 | 
						|
                              /*NumChildren=*/1);
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, MultipleFunctionSelection) {
 | 
						|
  StringRef Source = R"(void f0() {
 | 
						|
}
 | 
						|
void f1() { }
 | 
						|
void f2() { }
 | 
						|
void f3() { }
 | 
						|
)";
 | 
						|
  auto SelectedF1F2 = [](Optional<SelectedASTNode> Node) {
 | 
						|
    EXPECT_TRUE(Node);
 | 
						|
    EXPECT_EQ(Node->Children.size(), 2u);
 | 
						|
    checkNode<FunctionDecl>(Node->Children[0],
 | 
						|
                            SourceSelectionKind::InsideSelection,
 | 
						|
                            /*NumChildren=*/1, /*Name=*/"f1");
 | 
						|
    checkNode<FunctionDecl>(Node->Children[1],
 | 
						|
                            SourceSelectionKind::InsideSelection,
 | 
						|
                            /*NumChildren=*/1, /*Name=*/"f2");
 | 
						|
  };
 | 
						|
  // Just after '}' of f0 and just before 'void' of f3:
 | 
						|
  findSelectedASTNodes(Source, {2, 2}, FileRange{{2, 2}, {5, 1}}, SelectedF1F2);
 | 
						|
  // Just before 'void' of f1 and just after '}' of f2:
 | 
						|
  findSelectedASTNodes(Source, {3, 1}, FileRange{{3, 1}, {4, 14}},
 | 
						|
                       SelectedF1F2);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, MultipleStatementSelection) {
 | 
						|
  StringRef Source = R"(void f(int x, int y) {
 | 
						|
  int z = x;
 | 
						|
  f(2, 3);
 | 
						|
  if (x == 0) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  x = 1;
 | 
						|
  return;
 | 
						|
})";
 | 
						|
  // From 'f(2,3)' until just before 'x = 1;':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {3, 2}, FileRange{{3, 2}, {7, 1}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"f");
 | 
						|
        const auto &Body = checkNode<CompoundStmt>(
 | 
						|
            Fn.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/2);
 | 
						|
        allChildrenOf(checkNode<CallExpr>(Body.Children[0],
 | 
						|
                                          SourceSelectionKind::InsideSelection,
 | 
						|
                                          /*NumChildren=*/3))
 | 
						|
            .shouldHaveSelectionKind(SourceSelectionKind::InsideSelection);
 | 
						|
        allChildrenOf(checkNode<IfStmt>(Body.Children[1],
 | 
						|
                                        SourceSelectionKind::InsideSelection,
 | 
						|
                                        /*NumChildren=*/2))
 | 
						|
            .shouldHaveSelectionKind(SourceSelectionKind::InsideSelection);
 | 
						|
      });
 | 
						|
  // From 'f(2,3)' until just before ';' in 'x = 1;':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {3, 2}, FileRange{{3, 2}, {7, 8}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"f");
 | 
						|
        const auto &Body = checkNode<CompoundStmt>(
 | 
						|
            Fn.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/3);
 | 
						|
        checkNode<CallExpr>(Body.Children[0],
 | 
						|
                            SourceSelectionKind::InsideSelection,
 | 
						|
                            /*NumChildren=*/3);
 | 
						|
        checkNode<IfStmt>(Body.Children[1],
 | 
						|
                          SourceSelectionKind::InsideSelection,
 | 
						|
                          /*NumChildren=*/2);
 | 
						|
        checkNode<BinaryOperator>(Body.Children[2],
 | 
						|
                                  SourceSelectionKind::InsideSelection,
 | 
						|
                                  /*NumChildren=*/2);
 | 
						|
      });
 | 
						|
  // From the middle of 'int z = 3' until the middle of 'x = 1;':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {2, 10}, FileRange{{2, 10}, {7, 5}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"f");
 | 
						|
        const auto &Body = checkNode<CompoundStmt>(
 | 
						|
            Fn.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/4);
 | 
						|
        checkNode<DeclStmt>(Body.Children[0],
 | 
						|
                            SourceSelectionKind::ContainsSelectionStart,
 | 
						|
                            /*NumChildren=*/1);
 | 
						|
        checkNode<CallExpr>(Body.Children[1],
 | 
						|
                            SourceSelectionKind::InsideSelection,
 | 
						|
                            /*NumChildren=*/3);
 | 
						|
        checkNode<IfStmt>(Body.Children[2],
 | 
						|
                          SourceSelectionKind::InsideSelection,
 | 
						|
                          /*NumChildren=*/2);
 | 
						|
        checkNode<BinaryOperator>(Body.Children[3],
 | 
						|
                                  SourceSelectionKind::ContainsSelectionEnd,
 | 
						|
                                  /*NumChildren=*/1);
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, SelectionInFunctionInObjCImplementation) {
 | 
						|
  StringRef Source = R"(
 | 
						|
@interface I
 | 
						|
@end
 | 
						|
@implementation I
 | 
						|
 | 
						|
int notSelected() { }
 | 
						|
 | 
						|
int selected(int x) {
 | 
						|
  return x;
 | 
						|
}
 | 
						|
 | 
						|
@end
 | 
						|
@implementation I(Cat)
 | 
						|
 | 
						|
void catF() { }
 | 
						|
 | 
						|
@end
 | 
						|
 | 
						|
void outerFunction() { }
 | 
						|
)";
 | 
						|
  // Just the 'x' expression in 'selected':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {9, 10}, FileRange{{9, 10}, {9, 11}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Impl = checkNode<ObjCImplementationDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"I");
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Impl.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"selected");
 | 
						|
        allChildrenOf(Fn).shouldHaveSelectionKind(
 | 
						|
            SourceSelectionKind::ContainsSelection);
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // The entire 'catF':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {15, 1}, FileRange{{15, 1}, {15, 16}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Impl = checkNode<ObjCCategoryImplDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"Cat");
 | 
						|
        const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
            Impl.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"catF");
 | 
						|
        allChildrenOf(Fn).shouldHaveSelectionKind(
 | 
						|
            SourceSelectionKind::ContainsSelection);
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // From the line before 'selected' to the line after 'catF':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {16, 1}, FileRange{{7, 1}, {16, 1}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 2u);
 | 
						|
        const auto &Impl = checkNode<ObjCImplementationDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelectionStart,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"I");
 | 
						|
        const auto &Selected = checkNode<FunctionDecl>(
 | 
						|
            Impl.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/2, /*Name=*/"selected");
 | 
						|
        allChildrenOf(Selected).shouldHaveSelectionKind(
 | 
						|
            SourceSelectionKind::InsideSelection);
 | 
						|
        const auto &Cat = checkNode<ObjCCategoryImplDecl>(
 | 
						|
            Node->Children[1], SourceSelectionKind::ContainsSelectionEnd,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"Cat");
 | 
						|
        const auto &CatF = checkNode<FunctionDecl>(
 | 
						|
            Cat.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"catF");
 | 
						|
        allChildrenOf(CatF).shouldHaveSelectionKind(
 | 
						|
            SourceSelectionKind::InsideSelection);
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // Just the 'outer' function:
 | 
						|
  findSelectedASTNodes(Source, {19, 1}, FileRange{{19, 1}, {19, 25}},
 | 
						|
                       [](Optional<SelectedASTNode> Node) {
 | 
						|
                         EXPECT_TRUE(Node);
 | 
						|
                         EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
                         checkNode<FunctionDecl>(
 | 
						|
                             Node->Children[0],
 | 
						|
                             SourceSelectionKind::ContainsSelection,
 | 
						|
                             /*NumChildren=*/1, /*Name=*/"outerFunction");
 | 
						|
                       },
 | 
						|
                       SelectionFinderVisitor::Lang_OBJC);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, FunctionInObjCImplementationCarefulWithEarlyExit) {
 | 
						|
  StringRef Source = R"(
 | 
						|
@interface I
 | 
						|
@end
 | 
						|
@implementation I
 | 
						|
 | 
						|
void selected() {
 | 
						|
}
 | 
						|
 | 
						|
- (void) method { }
 | 
						|
 | 
						|
@end
 | 
						|
)";
 | 
						|
  // Just 'selected'
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {6, 1}, FileRange{{6, 1}, {7, 2}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Impl = checkNode<ObjCImplementationDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"I");
 | 
						|
        checkNode<FunctionDecl>(Impl.Children[0],
 | 
						|
                                SourceSelectionKind::ContainsSelection,
 | 
						|
                                /*NumChildren=*/1, /*Name=*/"selected");
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, AvoidImplicitDeclarations) {
 | 
						|
  StringRef Source = R"(
 | 
						|
struct Copy {
 | 
						|
  int x;
 | 
						|
};
 | 
						|
void foo() {
 | 
						|
  Copy x;
 | 
						|
  Copy y = x;
 | 
						|
}
 | 
						|
)";
 | 
						|
  // The entire struct 'Copy':
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {2, 1}, FileRange{{2, 1}, {4, 3}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
        const auto &Record = checkNode<CXXRecordDecl>(
 | 
						|
            Node->Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1, /*Name=*/"Copy");
 | 
						|
        checkNode<FieldDecl>(Record.Children[0],
 | 
						|
                             SourceSelectionKind::InsideSelection);
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, CorrectEndForObjectiveCImplementation) {
 | 
						|
  StringRef Source = R"(
 | 
						|
@interface I
 | 
						|
@end
 | 
						|
@implementation I
 | 
						|
@ end
 | 
						|
)";
 | 
						|
  // Just after '@ end'
 | 
						|
  findSelectedASTNodes(Source, {5, 6}, None,
 | 
						|
                       [](Optional<SelectedASTNode> Node) {
 | 
						|
                         EXPECT_TRUE(Node);
 | 
						|
                         EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
                         checkNode<ObjCImplementationDecl>(
 | 
						|
                             Node->Children[0],
 | 
						|
                             SourceSelectionKind::ContainsSelection);
 | 
						|
                       },
 | 
						|
                       SelectionFinderVisitor::Lang_OBJC);
 | 
						|
}
 | 
						|
 | 
						|
const SelectedASTNode &checkFnBody(const Optional<SelectedASTNode> &Node,
 | 
						|
                                   StringRef Name) {
 | 
						|
  EXPECT_TRUE(Node);
 | 
						|
  EXPECT_EQ(Node->Children.size(), 1u);
 | 
						|
  const auto &Fn = checkNode<FunctionDecl>(
 | 
						|
      Node->Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
      /*NumChildren=*/1, Name);
 | 
						|
  return checkNode<CompoundStmt>(Fn.Children[0],
 | 
						|
                                 SourceSelectionKind::ContainsSelection,
 | 
						|
                                 /*NumChildren=*/1);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, SelectObjectiveCPseudoObjectExprs) {
 | 
						|
  StringRef Source = R"(
 | 
						|
@interface I
 | 
						|
@property(readwrite) int prop;
 | 
						|
@end
 | 
						|
void selectProp(I *i) {
 | 
						|
(void)i.prop;
 | 
						|
i.prop = 21;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
@interface NSMutableArray
 | 
						|
- (id)objectAtIndexedSubscript:(unsigned int)index;
 | 
						|
- (void)setObject:(id)object atIndexedSubscript:(unsigned int)index;
 | 
						|
@end
 | 
						|
 | 
						|
void selectSubscript(NSMutableArray *array, I *i) {
 | 
						|
  (void)array[10];
 | 
						|
  array[i.prop] = i;
 | 
						|
}
 | 
						|
)";
 | 
						|
  // Just 'i.prop'.
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {6, 7}, FileRange{{6, 7}, {6, 13}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        const auto &CS = checkFnBody(Node, /*Name=*/"selectProp");
 | 
						|
        const auto &CCast = checkNode<CStyleCastExpr>(
 | 
						|
            CS.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &POE = checkNode<PseudoObjectExpr>(
 | 
						|
            CCast.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &PRE = checkNode<ObjCPropertyRefExpr>(
 | 
						|
            POE.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &Cast = checkNode<ImplicitCastExpr>(
 | 
						|
            PRE.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<DeclRefExpr>(Cast.Children[0],
 | 
						|
                               SourceSelectionKind::InsideSelection);
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // Just 'i.prop = 21'
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {7, 1}, FileRange{{7, 1}, {7, 12}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        const auto &CS = checkFnBody(Node, /*Name=*/"selectProp");
 | 
						|
        const auto &POE = checkNode<PseudoObjectExpr>(
 | 
						|
            CS.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &BinOp = checkNode<BinaryOperator>(
 | 
						|
            POE.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/2);
 | 
						|
        const auto &PRE = checkNode<ObjCPropertyRefExpr>(
 | 
						|
            BinOp.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &Cast = checkNode<ImplicitCastExpr>(
 | 
						|
            PRE.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<DeclRefExpr>(Cast.Children[0],
 | 
						|
                               SourceSelectionKind::InsideSelection);
 | 
						|
        checkNode<IntegerLiteral>(BinOp.Children[1],
 | 
						|
                                  SourceSelectionKind::InsideSelection);
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // Just 'array[10]'
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {17, 9}, FileRange{{17, 9}, {17, 18}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        const auto &CS = checkFnBody(Node, /*Name=*/"selectSubscript");
 | 
						|
        const auto &CCast = checkNode<CStyleCastExpr>(
 | 
						|
            CS.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &POE = checkNode<PseudoObjectExpr>(
 | 
						|
            CCast.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &SRE = checkNode<ObjCSubscriptRefExpr>(
 | 
						|
            POE.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/2);
 | 
						|
        const auto &Cast = checkNode<ImplicitCastExpr>(
 | 
						|
            SRE.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<DeclRefExpr>(Cast.Children[0],
 | 
						|
                               SourceSelectionKind::InsideSelection);
 | 
						|
        checkNode<IntegerLiteral>(SRE.Children[1],
 | 
						|
                                  SourceSelectionKind::InsideSelection);
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // Just 'array[i.prop] = array'
 | 
						|
  findSelectedASTNodes(
 | 
						|
      Source, {18, 3}, FileRange{{18, 3}, {18, 20}},
 | 
						|
      [](Optional<SelectedASTNode> Node) {
 | 
						|
        const auto &CS = checkFnBody(Node, /*Name=*/"selectSubscript");
 | 
						|
        const auto &POE = checkNode<PseudoObjectExpr>(
 | 
						|
            CS.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &BinOp = checkNode<BinaryOperator>(
 | 
						|
            POE.Children[0], SourceSelectionKind::ContainsSelection,
 | 
						|
            /*NumChildren=*/2);
 | 
						|
        const auto &SRE = checkNode<ObjCSubscriptRefExpr>(
 | 
						|
            BinOp.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/2);
 | 
						|
        const auto &Cast = checkNode<ImplicitCastExpr>(
 | 
						|
            SRE.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<DeclRefExpr>(Cast.Children[0],
 | 
						|
                               SourceSelectionKind::InsideSelection);
 | 
						|
        const auto &POE2 = checkNode<PseudoObjectExpr>(
 | 
						|
            SRE.Children[1], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &PRE = checkNode<ObjCPropertyRefExpr>(
 | 
						|
            POE2.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        const auto &Cast2 = checkNode<ImplicitCastExpr>(
 | 
						|
            PRE.Children[0], SourceSelectionKind::InsideSelection,
 | 
						|
            /*NumChildren=*/1);
 | 
						|
        checkNode<DeclRefExpr>(Cast2.Children[0],
 | 
						|
                               SourceSelectionKind::InsideSelection);
 | 
						|
        checkNode<DeclRefExpr>(BinOp.Children[1],
 | 
						|
                               SourceSelectionKind::InsideSelection);
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, SimpleCodeRangeASTSelection) {
 | 
						|
  StringRef Source = R"(void f(int x, int y) {
 | 
						|
  int z = x;
 | 
						|
  f(2, 3);
 | 
						|
  if (x == 0) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  x = 1;
 | 
						|
  return;
 | 
						|
}
 | 
						|
void f2() {
 | 
						|
  int m = 0;
 | 
						|
}
 | 
						|
)";
 | 
						|
  // No selection range.
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {2, 2}, None,
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_FALSE(SelectedCode);
 | 
						|
      });
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {2, 2}, FileRange{{2, 2}, {2, 2}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_FALSE(SelectedCode);
 | 
						|
      });
 | 
						|
  // Range that spans multiple functions is an invalid code range.
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {2, 2}, FileRange{{7, 2}, {12, 1}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_FALSE(SelectedCode);
 | 
						|
      });
 | 
						|
  // Just 'z = x;':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {2, 2}, FileRange{{2, 2}, {2, 13}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<DeclStmt>((*SelectedCode)[0]));
 | 
						|
        ArrayRef<SelectedASTNode::ReferenceType> Parents =
 | 
						|
            SelectedCode->getParents();
 | 
						|
        EXPECT_EQ(Parents.size(), 3u);
 | 
						|
        EXPECT_TRUE(
 | 
						|
            isa<TranslationUnitDecl>(Parents[0].get().Node.get<Decl>()));
 | 
						|
        // Function 'f' definition.
 | 
						|
        EXPECT_TRUE(isa<FunctionDecl>(Parents[1].get().Node.get<Decl>()));
 | 
						|
        // Function body of function 'F'.
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(Parents[2].get().Node.get<Stmt>()));
 | 
						|
      });
 | 
						|
  // From 'f(2,3)' until just before 'x = 1;':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {3, 2}, FileRange{{3, 2}, {7, 1}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 2u);
 | 
						|
        EXPECT_TRUE(isa<CallExpr>((*SelectedCode)[0]));
 | 
						|
        EXPECT_TRUE(isa<IfStmt>((*SelectedCode)[1]));
 | 
						|
        ArrayRef<SelectedASTNode::ReferenceType> Parents =
 | 
						|
            SelectedCode->getParents();
 | 
						|
        EXPECT_EQ(Parents.size(), 3u);
 | 
						|
        EXPECT_TRUE(
 | 
						|
            isa<TranslationUnitDecl>(Parents[0].get().Node.get<Decl>()));
 | 
						|
        // Function 'f' definition.
 | 
						|
        EXPECT_TRUE(isa<FunctionDecl>(Parents[1].get().Node.get<Decl>()));
 | 
						|
        // Function body of function 'F'.
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(Parents[2].get().Node.get<Stmt>()));
 | 
						|
      });
 | 
						|
  // From 'f(2,3)' until just before ';' in 'x = 1;':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {3, 2}, FileRange{{3, 2}, {7, 8}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 3u);
 | 
						|
        EXPECT_TRUE(isa<CallExpr>((*SelectedCode)[0]));
 | 
						|
        EXPECT_TRUE(isa<IfStmt>((*SelectedCode)[1]));
 | 
						|
        EXPECT_TRUE(isa<BinaryOperator>((*SelectedCode)[2]));
 | 
						|
      });
 | 
						|
  // From the middle of 'int z = 3' until the middle of 'x = 1;':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {2, 10}, FileRange{{2, 10}, {7, 5}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 4u);
 | 
						|
        EXPECT_TRUE(isa<DeclStmt>((*SelectedCode)[0]));
 | 
						|
        EXPECT_TRUE(isa<CallExpr>((*SelectedCode)[1]));
 | 
						|
        EXPECT_TRUE(isa<IfStmt>((*SelectedCode)[2]));
 | 
						|
        EXPECT_TRUE(isa<BinaryOperator>((*SelectedCode)[3]));
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, OutOfBodyCodeRange) {
 | 
						|
  StringRef Source = R"(
 | 
						|
int codeRange = 2 + 3;
 | 
						|
)";
 | 
						|
  // '2+3' expression.
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {2, 17}, FileRange{{2, 17}, {2, 22}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<BinaryOperator>((*SelectedCode)[0]));
 | 
						|
        ArrayRef<SelectedASTNode::ReferenceType> Parents =
 | 
						|
            SelectedCode->getParents();
 | 
						|
        EXPECT_EQ(Parents.size(), 2u);
 | 
						|
        EXPECT_TRUE(
 | 
						|
            isa<TranslationUnitDecl>(Parents[0].get().Node.get<Decl>()));
 | 
						|
        // Variable 'codeRange'.
 | 
						|
        EXPECT_TRUE(isa<VarDecl>(Parents[1].get().Node.get<Decl>()));
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, SelectVarDeclStmt) {
 | 
						|
  StringRef Source = R"(
 | 
						|
void f() {
 | 
						|
   {
 | 
						|
       int a;
 | 
						|
   }
 | 
						|
}
 | 
						|
)";
 | 
						|
  // 'int a'
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {4, 8}, FileRange{{4, 8}, {4, 14}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<DeclStmt>((*SelectedCode)[0]));
 | 
						|
        ArrayRef<SelectedASTNode::ReferenceType> Parents =
 | 
						|
            SelectedCode->getParents();
 | 
						|
        EXPECT_EQ(Parents.size(), 4u);
 | 
						|
        EXPECT_TRUE(
 | 
						|
            isa<TranslationUnitDecl>(Parents[0].get().Node.get<Decl>()));
 | 
						|
        // Function 'f' definition.
 | 
						|
        EXPECT_TRUE(isa<FunctionDecl>(Parents[1].get().Node.get<Decl>()));
 | 
						|
        // Function body of function 'F'.
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(Parents[2].get().Node.get<Stmt>()));
 | 
						|
        // Compound statement in body of 'F'.
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(Parents[3].get().Node.get<Stmt>()));
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, SelectEntireDeclStmtRange) {
 | 
						|
  StringRef Source = R"(
 | 
						|
void f(int x, int y) {
 | 
						|
   int a = x * y;
 | 
						|
}
 | 
						|
)";
 | 
						|
  // 'int a = x * y'
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {3, 4}, FileRange{{3, 4}, {3, 17}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<DeclStmt>((*SelectedCode)[0]));
 | 
						|
        ArrayRef<SelectedASTNode::ReferenceType> Parents =
 | 
						|
            SelectedCode->getParents();
 | 
						|
        EXPECT_EQ(Parents.size(), 3u);
 | 
						|
        EXPECT_TRUE(
 | 
						|
            isa<TranslationUnitDecl>(Parents[0].get().Node.get<Decl>()));
 | 
						|
        // Function 'f' definition.
 | 
						|
        EXPECT_TRUE(isa<FunctionDecl>(Parents[1].get().Node.get<Decl>()));
 | 
						|
        // Function body of function 'F'.
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(Parents[2].get().Node.get<Stmt>()));
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, SelectEntireDeclStmtRangeWithMultipleDecls) {
 | 
						|
  StringRef Source = R"(
 | 
						|
void f(int x, int y) {
 | 
						|
   int a = x * y, b = x - y;
 | 
						|
}
 | 
						|
)";
 | 
						|
  // 'b = x - y'
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {3, 19}, FileRange{{3, 19}, {3, 28}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<DeclStmt>((*SelectedCode)[0]));
 | 
						|
        ArrayRef<SelectedASTNode::ReferenceType> Parents =
 | 
						|
            SelectedCode->getParents();
 | 
						|
        EXPECT_EQ(Parents.size(), 3u);
 | 
						|
        EXPECT_TRUE(
 | 
						|
            isa<TranslationUnitDecl>(Parents[0].get().Node.get<Decl>()));
 | 
						|
        // Function 'f' definition.
 | 
						|
        EXPECT_TRUE(isa<FunctionDecl>(Parents[1].get().Node.get<Decl>()));
 | 
						|
        // Function body of function 'F'.
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(Parents[2].get().Node.get<Stmt>()));
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, SimpleCodeRangeASTSelectionInObjCMethod) {
 | 
						|
  StringRef Source = R"(@interface I @end
 | 
						|
@implementation I
 | 
						|
- (void) f:(int)x with:(int) y {
 | 
						|
  int z = x;
 | 
						|
  [self f: 2 with: 3];
 | 
						|
  if (x == 0) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  x = 1;
 | 
						|
  return;
 | 
						|
}
 | 
						|
- (void)f2 {
 | 
						|
  int m = 0;
 | 
						|
}
 | 
						|
@end
 | 
						|
)";
 | 
						|
  // Range that spans multiple methods is an invalid code range.
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {9, 2}, FileRange{{9, 2}, {13, 1}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_FALSE(SelectedCode);
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // Just 'z = x;':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {4, 2}, FileRange{{4, 2}, {4, 13}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<DeclStmt>((*SelectedCode)[0]));
 | 
						|
        ArrayRef<SelectedASTNode::ReferenceType> Parents =
 | 
						|
            SelectedCode->getParents();
 | 
						|
        EXPECT_EQ(Parents.size(), 4u);
 | 
						|
        EXPECT_TRUE(
 | 
						|
            isa<TranslationUnitDecl>(Parents[0].get().Node.get<Decl>()));
 | 
						|
        // 'I' @implementation.
 | 
						|
        EXPECT_TRUE(isa<ObjCImplDecl>(Parents[1].get().Node.get<Decl>()));
 | 
						|
        // Function 'f' definition.
 | 
						|
        EXPECT_TRUE(isa<ObjCMethodDecl>(Parents[2].get().Node.get<Decl>()));
 | 
						|
        // Function body of function 'F'.
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(Parents[3].get().Node.get<Stmt>()));
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // From '[self f: 2 with: 3]' until just before 'x = 1;':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {5, 2}, FileRange{{5, 2}, {9, 1}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 2u);
 | 
						|
        EXPECT_TRUE(isa<ObjCMessageExpr>((*SelectedCode)[0]));
 | 
						|
        EXPECT_TRUE(isa<IfStmt>((*SelectedCode)[1]));
 | 
						|
        ArrayRef<SelectedASTNode::ReferenceType> Parents =
 | 
						|
            SelectedCode->getParents();
 | 
						|
        EXPECT_EQ(Parents.size(), 4u);
 | 
						|
        EXPECT_TRUE(
 | 
						|
            isa<TranslationUnitDecl>(Parents[0].get().Node.get<Decl>()));
 | 
						|
        // 'I' @implementation.
 | 
						|
        EXPECT_TRUE(isa<ObjCImplDecl>(Parents[1].get().Node.get<Decl>()));
 | 
						|
        // Function 'f' definition.
 | 
						|
        EXPECT_TRUE(isa<ObjCMethodDecl>(Parents[2].get().Node.get<Decl>()));
 | 
						|
        // Function body of function 'F'.
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(Parents[3].get().Node.get<Stmt>()));
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, CanonicalizeObjCStringLiteral) {
 | 
						|
  StringRef Source = R"(
 | 
						|
void foo() {
 | 
						|
  (void)@"test";
 | 
						|
}
 | 
						|
      )";
 | 
						|
  // Just '"test"':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {3, 10}, FileRange{{3, 10}, {3, 16}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<ObjCStringLiteral>((*SelectedCode)[0]));
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
  // Just 'test':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {3, 11}, FileRange{{3, 11}, {3, 15}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<ObjCStringLiteral>((*SelectedCode)[0]));
 | 
						|
      },
 | 
						|
      SelectionFinderVisitor::Lang_OBJC);
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, CanonicalizeMemberCalleeToCall) {
 | 
						|
  StringRef Source = R"(
 | 
						|
class AClass { public:
 | 
						|
  void method();
 | 
						|
  int afield;
 | 
						|
  void selectWholeCallWhenJustMethodSelected(int &i) {
 | 
						|
    method();
 | 
						|
  }
 | 
						|
};
 | 
						|
void selectWholeCallWhenJustMethodSelected() {
 | 
						|
  AClass a;
 | 
						|
  a.method();
 | 
						|
}
 | 
						|
void dontSelectArgument(AClass &a) {
 | 
						|
  a.selectWholeCallWhenJustMethodSelected(a.afield);
 | 
						|
}
 | 
						|
     )";
 | 
						|
  // Just 'method' with implicit 'this':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {6, 5}, FileRange{{6, 5}, {6, 11}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<CXXMemberCallExpr>((*SelectedCode)[0]));
 | 
						|
      });
 | 
						|
  // Just 'method':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {11, 5}, FileRange{{11, 5}, {11, 11}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<CXXMemberCallExpr>((*SelectedCode)[0]));
 | 
						|
      });
 | 
						|
  // Just 'afield', which should not select the call.
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {14, 5}, FileRange{{14, 45}, {14, 51}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_FALSE(isa<CXXMemberCallExpr>((*SelectedCode)[0]));
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
TEST(ASTSelectionFinder, CanonicalizeFuncCalleeToCall) {
 | 
						|
  StringRef Source = R"(
 | 
						|
void function();
 | 
						|
 | 
						|
void test() {
 | 
						|
  function();
 | 
						|
}
 | 
						|
     )";
 | 
						|
  // Just 'function':
 | 
						|
  findSelectedASTNodesWithRange(
 | 
						|
      Source, {5, 3}, FileRange{{5, 3}, {5, 11}},
 | 
						|
      [](SourceRange SelectionRange, Optional<SelectedASTNode> Node) {
 | 
						|
        EXPECT_TRUE(Node);
 | 
						|
        Node->dump();
 | 
						|
        Optional<CodeRangeASTSelection> SelectedCode =
 | 
						|
            CodeRangeASTSelection::create(SelectionRange, std::move(*Node));
 | 
						|
        EXPECT_TRUE(SelectedCode);
 | 
						|
        EXPECT_EQ(SelectedCode->size(), 1u);
 | 
						|
        EXPECT_TRUE(isa<CallExpr>((*SelectedCode)[0]));
 | 
						|
        EXPECT_TRUE(isa<CompoundStmt>(
 | 
						|
            SelectedCode->getParents()[SelectedCode->getParents().size() - 1]
 | 
						|
                .get()
 | 
						|
                .Node.get<Stmt>()));
 | 
						|
      });
 | 
						|
}
 | 
						|
 | 
						|
} // end anonymous namespace
 |