|  | //===- unittests/AST/DeclPrinterTest.cpp --- Declaration printer tests ----===// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // This file contains tests for Decl::print() and related methods. | 
|  | // | 
|  | // Search this file for WRONG to see test cases that are producing something | 
|  | // completely wrong, invalid C++ or just misleading. | 
|  | // | 
|  | // These tests have a coding convention: | 
|  | // * declaration to be printed is named 'A' unless it should have some special | 
|  | // name (e.g., 'operator+'); | 
|  | // * additional helper declarations are 'Z', 'Y', 'X' and so on. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "clang/AST/ASTContext.h" | 
|  | #include "clang/ASTMatchers/ASTMatchFinder.h" | 
|  | #include "clang/Tooling/Tooling.h" | 
|  | #include "llvm/ADT/SmallString.h" | 
|  | #include "gtest/gtest.h" | 
|  |  | 
|  | using namespace clang; | 
|  | using namespace ast_matchers; | 
|  | using namespace tooling; | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | void PrintDecl(raw_ostream &Out, const ASTContext *Context, const Decl *D) { | 
|  | PrintingPolicy Policy = Context->getPrintingPolicy(); | 
|  | Policy.TerseOutput = true; | 
|  | D->print(Out, Policy, /*Indentation*/ 0, /*PrintInstantiation*/ false); | 
|  | } | 
|  |  | 
|  | class PrintMatch : public MatchFinder::MatchCallback { | 
|  | SmallString<1024> Printed; | 
|  | unsigned NumFoundDecls; | 
|  |  | 
|  | public: | 
|  | PrintMatch() : NumFoundDecls(0) {} | 
|  |  | 
|  | void run(const MatchFinder::MatchResult &Result) override { | 
|  | const Decl *D = Result.Nodes.getDeclAs<Decl>("id"); | 
|  | if (!D || D->isImplicit()) | 
|  | return; | 
|  | NumFoundDecls++; | 
|  | if (NumFoundDecls > 1) | 
|  | return; | 
|  |  | 
|  | llvm::raw_svector_ostream Out(Printed); | 
|  | PrintDecl(Out, Result.Context, D); | 
|  | } | 
|  |  | 
|  | StringRef getPrinted() const { | 
|  | return Printed; | 
|  | } | 
|  |  | 
|  | unsigned getNumFoundDecls() const { | 
|  | return NumFoundDecls; | 
|  | } | 
|  | }; | 
|  |  | 
|  | ::testing::AssertionResult PrintedDeclMatches( | 
|  | StringRef Code, | 
|  | const std::vector<std::string> &Args, | 
|  | const DeclarationMatcher &NodeMatch, | 
|  | StringRef ExpectedPrinted, | 
|  | StringRef FileName) { | 
|  | PrintMatch Printer; | 
|  | MatchFinder Finder; | 
|  | Finder.addMatcher(NodeMatch, &Printer); | 
|  | std::unique_ptr<FrontendActionFactory> Factory( | 
|  | newFrontendActionFactory(&Finder)); | 
|  |  | 
|  | if (!runToolOnCodeWithArgs(Factory->create(), Code, Args, FileName)) | 
|  | return testing::AssertionFailure() | 
|  | << "Parsing error in \"" << Code.str() << "\""; | 
|  |  | 
|  | if (Printer.getNumFoundDecls() == 0) | 
|  | return testing::AssertionFailure() | 
|  | << "Matcher didn't find any declarations"; | 
|  |  | 
|  | if (Printer.getNumFoundDecls() > 1) | 
|  | return testing::AssertionFailure() | 
|  | << "Matcher should match only one declaration " | 
|  | "(found " << Printer.getNumFoundDecls() << ")"; | 
|  |  | 
|  | if (Printer.getPrinted() != ExpectedPrinted) | 
|  | return ::testing::AssertionFailure() | 
|  | << "Expected \"" << ExpectedPrinted.str() << "\", " | 
|  | "got \"" << Printer.getPrinted().str() << "\""; | 
|  |  | 
|  | return ::testing::AssertionSuccess(); | 
|  | } | 
|  |  | 
|  | ::testing::AssertionResult PrintedDeclCXX98Matches(StringRef Code, | 
|  | StringRef DeclName, | 
|  | StringRef ExpectedPrinted) { | 
|  | std::vector<std::string> Args(1, "-std=c++98"); | 
|  | return PrintedDeclMatches(Code, | 
|  | Args, | 
|  | namedDecl(hasName(DeclName)).bind("id"), | 
|  | ExpectedPrinted, | 
|  | "input.cc"); | 
|  | } | 
|  |  | 
|  | ::testing::AssertionResult PrintedDeclCXX98Matches( | 
|  | StringRef Code, | 
|  | const DeclarationMatcher &NodeMatch, | 
|  | StringRef ExpectedPrinted) { | 
|  | std::vector<std::string> Args(1, "-std=c++98"); | 
|  | return PrintedDeclMatches(Code, | 
|  | Args, | 
|  | NodeMatch, | 
|  | ExpectedPrinted, | 
|  | "input.cc"); | 
|  | } | 
|  |  | 
|  | ::testing::AssertionResult PrintedDeclCXX11Matches(StringRef Code, | 
|  | StringRef DeclName, | 
|  | StringRef ExpectedPrinted) { | 
|  | std::vector<std::string> Args(1, "-std=c++11"); | 
|  | return PrintedDeclMatches(Code, | 
|  | Args, | 
|  | namedDecl(hasName(DeclName)).bind("id"), | 
|  | ExpectedPrinted, | 
|  | "input.cc"); | 
|  | } | 
|  |  | 
|  | ::testing::AssertionResult PrintedDeclCXX11Matches( | 
|  | StringRef Code, | 
|  | const DeclarationMatcher &NodeMatch, | 
|  | StringRef ExpectedPrinted) { | 
|  | std::vector<std::string> Args(1, "-std=c++11"); | 
|  | return PrintedDeclMatches(Code, | 
|  | Args, | 
|  | NodeMatch, | 
|  | ExpectedPrinted, | 
|  | "input.cc"); | 
|  | } | 
|  |  | 
|  | ::testing::AssertionResult PrintedDeclCXX11nonMSCMatches( | 
|  | StringRef Code, | 
|  | const DeclarationMatcher &NodeMatch, | 
|  | StringRef ExpectedPrinted) { | 
|  | std::vector<std::string> Args(1, "-std=c++11"); | 
|  | Args.push_back("-fno-delayed-template-parsing"); | 
|  | return PrintedDeclMatches(Code, | 
|  | Args, | 
|  | NodeMatch, | 
|  | ExpectedPrinted, | 
|  | "input.cc"); | 
|  | } | 
|  |  | 
|  | ::testing::AssertionResult | 
|  | PrintedDeclCXX1ZMatches(StringRef Code, const DeclarationMatcher &NodeMatch, | 
|  | StringRef ExpectedPrinted) { | 
|  | std::vector<std::string> Args(1, "-std=c++1z"); | 
|  | return PrintedDeclMatches(Code, | 
|  | Args, | 
|  | NodeMatch, | 
|  | ExpectedPrinted, | 
|  | "input.cc"); | 
|  | } | 
|  |  | 
|  | ::testing::AssertionResult PrintedDeclObjCMatches( | 
|  | StringRef Code, | 
|  | const DeclarationMatcher &NodeMatch, | 
|  | StringRef ExpectedPrinted) { | 
|  | std::vector<std::string> Args(1, ""); | 
|  | return PrintedDeclMatches(Code, | 
|  | Args, | 
|  | NodeMatch, | 
|  | ExpectedPrinted, | 
|  | "input.m"); | 
|  | } | 
|  |  | 
|  | } // unnamed namespace | 
|  |  | 
|  | TEST(DeclPrinter, TestTypedef1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "typedef int A;", | 
|  | "A", | 
|  | "typedef int A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTypedef2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "typedef const char *A;", | 
|  | "A", | 
|  | "typedef const char *A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTypedef3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template <typename Y> class X {};" | 
|  | "typedef X<int> A;", | 
|  | "A", | 
|  | "typedef X<int> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTypedef4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "namespace X { class Y {}; }" | 
|  | "typedef X::Y A;", | 
|  | "A", | 
|  | "typedef X::Y A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestNamespace1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "namespace A { int B; }", | 
|  | "A", | 
|  | "namespace A {\n}")); | 
|  | // Should be: with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestNamespace2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "inline namespace A { int B; }", | 
|  | "A", | 
|  | "inline namespace A {\n}")); | 
|  | // Should be: with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestNamespaceAlias1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "namespace Z { }" | 
|  | "namespace A = Z;", | 
|  | "A", | 
|  | "namespace A = Z")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestNamespaceAlias2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "namespace X { namespace Y {} }" | 
|  | "namespace A = X::Y;", | 
|  | "A", | 
|  | "namespace A = X::Y")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class A { int a; };", | 
|  | "A", | 
|  | "class A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A { int a; };", | 
|  | "A", | 
|  | "struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "union A { int a; };", | 
|  | "A", | 
|  | "union A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class Z { int a; };" | 
|  | "class A : Z { int b; };", | 
|  | "A", | 
|  | "class A : Z {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z { int a; };" | 
|  | "struct A : Z { int b; };", | 
|  | "A", | 
|  | "struct A : Z {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl6) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class Z { int a; };" | 
|  | "class A : public Z { int b; };", | 
|  | "A", | 
|  | "class A : public Z {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl7) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class Z { int a; };" | 
|  | "class A : protected Z { int b; };", | 
|  | "A", | 
|  | "class A : protected Z {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl8) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class Z { int a; };" | 
|  | "class A : private Z { int b; };", | 
|  | "A", | 
|  | "class A : private Z {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl9) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class Z { int a; };" | 
|  | "class A : virtual Z { int b; };", | 
|  | "A", | 
|  | "class A : virtual Z {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl10) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class Z { int a; };" | 
|  | "class A : virtual public Z { int b; };", | 
|  | "A", | 
|  | "class A : virtual public Z {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXRecordDecl11) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class Z { int a; };" | 
|  | "class Y : virtual public Z { int b; };" | 
|  | "class A : virtual public Z, private Y { int c; };", | 
|  | "A", | 
|  | "class A : virtual public Z, private Y {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void A();", | 
|  | "A", | 
|  | "void A()")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void A() {}", | 
|  | "A", | 
|  | "void A()")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void Z();" | 
|  | "void A() { Z(); }", | 
|  | "A", | 
|  | "void A()")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "extern void A();", | 
|  | "A", | 
|  | "extern void A()")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "static void A();", | 
|  | "A", | 
|  | "static void A()")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl6) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "inline void A();", | 
|  | "A", | 
|  | "inline void A()")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl7) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "constexpr int A(int a);", | 
|  | "A", | 
|  | "constexpr int A(int a)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl8) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void A(int a);", | 
|  | "A", | 
|  | "void A(int a)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl9) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void A(...);", | 
|  | "A", | 
|  | "void A(...)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl10) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void A(int a, ...);", | 
|  | "A", | 
|  | "void A(int a, ...)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl11) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "typedef long ssize_t;" | 
|  | "typedef int *pInt;" | 
|  | "void A(int a, pInt b, ssize_t c);", | 
|  | "A", | 
|  | "void A(int a, pInt b, ssize_t c)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl12) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void A(int a, int b = 0);", | 
|  | "A", | 
|  | "void A(int a, int b = 0)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl13) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void (*A(int a))(int b);", | 
|  | "A", | 
|  | "void (*A(int a))(int)")); | 
|  | // Should be: with semicolon, with parameter name (?) | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl14) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T>" | 
|  | "void A(T t) { }" | 
|  | "template<>" | 
|  | "void A(int N) { }", | 
|  | functionDecl(hasName("A"), isExplicitTemplateSpecialization()).bind("id"), | 
|  | "void A(int N)")); | 
|  | // WRONG; Should be: "template <> void A(int N);")); | 
|  | } | 
|  |  | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  A();" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A()")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  A(int a);" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A(int a)")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  A(const A &a);" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A(const A &a)")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  A(const A &a, int = 0);" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A(const A &a, int = 0)")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct A {" | 
|  | "  A(const A &&a);" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A(const A &&a)")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl6) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  explicit A(int a);" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "explicit A(int a)")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl7) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct A {" | 
|  | "  constexpr A();" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "constexpr A()")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl8) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct A {" | 
|  | "  A() = default;" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A() = default")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl9) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct A {" | 
|  | "  A() = delete;" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A() = delete")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl10) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<typename... T>" | 
|  | "struct A {" | 
|  | "  A(const A &a);" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A<T...>(const A<T...> &a)")); | 
|  | // WRONG; Should be: "A(const A<T...> &a);" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConstructorDecl11) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11nonMSCMatches( | 
|  | "template<typename... T>" | 
|  | "struct A : public T... {" | 
|  | "  A(T&&... ts) : T(ts)... {}" | 
|  | "};", | 
|  | cxxConstructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "A<T...>(T &&...ts) : T(ts)...")); | 
|  | // WRONG; Should be: "A(T &&...ts) : T(ts)... {}" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXDestructorDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  ~A();" | 
|  | "};", | 
|  | cxxDestructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "~A()")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXDestructorDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  virtual ~A();" | 
|  | "};", | 
|  | cxxDestructorDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "virtual ~A()")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConversionDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  operator int();" | 
|  | "};", | 
|  | cxxMethodDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "operator int()")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConversionDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct A {" | 
|  | "  operator bool();" | 
|  | "};", | 
|  | cxxMethodDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "operator bool()")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXConversionDecl3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {};" | 
|  | "struct A {" | 
|  | "  operator Z();" | 
|  | "};", | 
|  | cxxMethodDecl(ofClass(hasName("A"))).bind("id"), | 
|  | "operator Z()")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_AllocationFunction1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "namespace std { typedef decltype(sizeof(int)) size_t; }" | 
|  | "struct Z {" | 
|  | "  void *operator new(std::size_t);" | 
|  | "};", | 
|  | cxxMethodDecl(ofClass(hasName("Z"))).bind("id"), | 
|  | "void *operator new(std::size_t)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_AllocationFunction2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "namespace std { typedef decltype(sizeof(int)) size_t; }" | 
|  | "struct Z {" | 
|  | "  void *operator new[](std::size_t);" | 
|  | "};", | 
|  | cxxMethodDecl(ofClass(hasName("Z"))).bind("id"), | 
|  | "void *operator new[](std::size_t)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_AllocationFunction3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct Z {" | 
|  | "  void operator delete(void *);" | 
|  | "};", | 
|  | cxxMethodDecl(ofClass(hasName("Z"))).bind("id"), | 
|  | "void operator delete(void *) noexcept")); | 
|  | // Should be: with semicolon, without noexcept? | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_AllocationFunction4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  void operator delete(void *);" | 
|  | "};", | 
|  | cxxMethodDecl(ofClass(hasName("Z"))).bind("id"), | 
|  | "void operator delete(void *)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_AllocationFunction5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct Z {" | 
|  | "  void operator delete[](void *);" | 
|  | "};", | 
|  | cxxMethodDecl(ofClass(hasName("Z"))).bind("id"), | 
|  | "void operator delete[](void *) noexcept")); | 
|  | // Should be: with semicolon, without noexcept? | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_Operator1) { | 
|  | const char *OperatorNames[] = { | 
|  | "+",  "-",  "*",  "/",  "%",  "^",   "&",   "|", | 
|  | "=",  "<",  ">",  "+=", "-=", "*=",  "/=",  "%=", | 
|  | "^=", "&=", "|=", "<<", ">>", ">>=", "<<=", "==",  "!=", | 
|  | "<=", ">=", "&&", "||",  ",", "->*", | 
|  | "()", "[]" | 
|  | }; | 
|  |  | 
|  | for (unsigned i = 0, e = llvm::array_lengthof(OperatorNames); i != e; ++i) { | 
|  | SmallString<128> Code; | 
|  | Code.append("struct Z { void operator"); | 
|  | Code.append(OperatorNames[i]); | 
|  | Code.append("(Z z); };"); | 
|  |  | 
|  | SmallString<128> Expected; | 
|  | Expected.append("void operator"); | 
|  | Expected.append(OperatorNames[i]); | 
|  | Expected.append("(Z z)"); | 
|  | // Should be: with semicolon | 
|  |  | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | Code, | 
|  | cxxMethodDecl(ofClass(hasName("Z"))).bind("id"), | 
|  | Expected)); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_Operator2) { | 
|  | const char *OperatorNames[] = { | 
|  | "~", "!", "++", "--", "->" | 
|  | }; | 
|  |  | 
|  | for (unsigned i = 0, e = llvm::array_lengthof(OperatorNames); i != e; ++i) { | 
|  | SmallString<128> Code; | 
|  | Code.append("struct Z { void operator"); | 
|  | Code.append(OperatorNames[i]); | 
|  | Code.append("(); };"); | 
|  |  | 
|  | SmallString<128> Expected; | 
|  | Expected.append("void operator"); | 
|  | Expected.append(OperatorNames[i]); | 
|  | Expected.append("()"); | 
|  | // Should be: with semicolon | 
|  |  | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | Code, | 
|  | cxxMethodDecl(ofClass(hasName("Z"))).bind("id"), | 
|  | Expected)); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a);" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  virtual void A(int a);" | 
|  | "};", | 
|  | "A", | 
|  | "virtual void A(int a)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  virtual void A(int a);" | 
|  | "};" | 
|  | "struct ZZ : Z {" | 
|  | "  void A(int a);" | 
|  | "};", | 
|  | "ZZ::A", | 
|  | "void A(int a)")); | 
|  | // Should be: with semicolon | 
|  | // TODO: should we print "virtual"? | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  inline void A(int a);" | 
|  | "};", | 
|  | "A", | 
|  | "inline void A(int a)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  virtual void A(int a) = 0;" | 
|  | "};", | 
|  | "A", | 
|  | "virtual void A(int a) = 0")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_CVQualifier1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) const;" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) const")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_CVQualifier2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) volatile;" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) volatile")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_CVQualifier3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) const volatile;" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) const volatile")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_RefQualifier1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) &;" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) &")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestCXXMethodDecl_RefQualifier2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) &&;" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) &&")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl_ExceptionSpecification1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) throw();" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) throw()")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl_ExceptionSpecification2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) throw(int);" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) throw(int)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl_ExceptionSpecification3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "class ZZ {};" | 
|  | "struct Z {" | 
|  | "  void A(int a) throw(ZZ, int);" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) throw(ZZ, int)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl_ExceptionSpecification4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) noexcept;" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) noexcept")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl_ExceptionSpecification5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) noexcept(true);" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) noexcept(trueA(int a) noexcept(true)")); | 
|  | // WRONG; Should be: "void A(int a) noexcept(true);" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl_ExceptionSpecification6) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "struct Z {" | 
|  | "  void A(int a) noexcept(1 < 2);" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) noexcept(1 < 2A(int a) noexcept(1 < 2)")); | 
|  | // WRONG; Should be: "void A(int a) noexcept(1 < 2);" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionDecl_ExceptionSpecification7) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<int N>" | 
|  | "struct Z {" | 
|  | "  void A(int a) noexcept(N < 2);" | 
|  | "};", | 
|  | "A", | 
|  | "void A(int a) noexcept(N < 2A(int a) noexcept(N < 2)")); | 
|  | // WRONG; Should be: "void A(int a) noexcept(N < 2);" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestVarDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "char *const (*(*A)[5])(int);", | 
|  | "A", | 
|  | "char *const (*(*A)[5])(int)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestVarDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "void (*A)() throw(int);", | 
|  | "A", | 
|  | "void (*A)() throw(int)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestVarDecl3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "void (*A)() noexcept;", | 
|  | "A", | 
|  | "void (*A)() noexcept")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFieldDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T>" | 
|  | "struct Z { T A; };", | 
|  | "A", | 
|  | "T A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFieldDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<int N>" | 
|  | "struct Z { int A[N]; };", | 
|  | "A", | 
|  | "int A[N]")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T>" | 
|  | "struct A { T a; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename T> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T = int>" | 
|  | "struct A { T a; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename T = int> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<class T>" | 
|  | "struct A { T a; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <class T> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T, typename U>" | 
|  | "struct A { T a; U b; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename T, typename U> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<int N>" | 
|  | "struct A { int a[N]; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <int N> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl6) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<int N = 42>" | 
|  | "struct A { int a[N]; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <int N = 42> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl7) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "typedef int MyInt;" | 
|  | "template<MyInt N>" | 
|  | "struct A { int a[N]; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <MyInt N> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl8) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<template<typename U> class T> struct A { };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <template <typename U> class T> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl9) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T> struct Z { };" | 
|  | "template<template<typename U> class T = Z> struct A { };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <template <typename U> class T> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl10) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<typename... T>" | 
|  | "struct A { int a; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename ...T> struct A {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateDecl11) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<typename... T>" | 
|  | "struct A : public T... { int a; };", | 
|  | classTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename ...T> struct A : public T... {\n}")); | 
|  | // Should be: with semicolon, with { ... } | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplatePartialSpecializationDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T, typename U>" | 
|  | "struct A { T a; U b; };" | 
|  | "template<typename T>" | 
|  | "struct A<T, int> { T a; };", | 
|  | classTemplateSpecializationDecl().bind("id"), | 
|  | "struct A {\n}")); | 
|  | // WRONG; Should be: "template<typename T> struct A<T, int> { ... }" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplatePartialSpecializationDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T>" | 
|  | "struct A { T a; };" | 
|  | "template<typename T>" | 
|  | "struct A<T *> { T a; };", | 
|  | classTemplateSpecializationDecl().bind("id"), | 
|  | "struct A {\n}")); | 
|  | // WRONG; Should be: "template<typename T> struct A<T *> { ... }" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestClassTemplateSpecializationDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T>" | 
|  | "struct A { T a; };" | 
|  | "template<>" | 
|  | "struct A<int> { int a; };", | 
|  | classTemplateSpecializationDecl().bind("id"), | 
|  | "struct A {\n}")); | 
|  | // WRONG; Should be: "template<> struct A<int> { ... }" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionTemplateDecl1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T>" | 
|  | "void A(T &t);", | 
|  | functionTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename T> void A(T &t)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionTemplateDecl2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T>" | 
|  | "void A(T &t) { }", | 
|  | functionTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename T> void A(T &t)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionTemplateDecl3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<typename... T>" | 
|  | "void A(T... a);", | 
|  | functionTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename ...T> void A(T ...a)")); | 
|  | // Should be: with semicolon. | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionTemplateDecl4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z { template<typename T> void A(T t); };", | 
|  | functionTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename T> void A(T t)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionTemplateDecl5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "struct Z { template<typename T> void A(T t) {} };", | 
|  | functionTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename T> void A(T t)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestFunctionTemplateDecl6) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T >struct Z {" | 
|  | "  template<typename U> void A(U t) {}" | 
|  | "};", | 
|  | functionTemplateDecl(hasName("A")).bind("id"), | 
|  | "template <typename U> void A(U t)")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T> struct Z {};" | 
|  | "struct X {};" | 
|  | "Z<X> A;", | 
|  | "A", | 
|  | "Z<X> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList2) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T, typename U> struct Z {};" | 
|  | "struct X {};" | 
|  | "typedef int Y;" | 
|  | "Z<X, Y> A;", | 
|  | "A", | 
|  | "Z<X, Y> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList3) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T> struct Z {};" | 
|  | "template<typename T> struct X {};" | 
|  | "Z<X<int> > A;", | 
|  | "A", | 
|  | "Z<X<int> > A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList4) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<typename T> struct Z {};" | 
|  | "template<typename T> struct X {};" | 
|  | "Z<X<int>> A;", | 
|  | "A", | 
|  | "Z<X<int> > A")); | 
|  | // Should be: with semicolon, without extra space in "> >" | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList5) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T> struct Z {};" | 
|  | "template<typename T> struct X { Z<T> A; };", | 
|  | "A", | 
|  | "Z<T> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList6) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<template<typename T> class U> struct Z {};" | 
|  | "template<typename T> struct X {};" | 
|  | "Z<X> A;", | 
|  | "A", | 
|  | "Z<X> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList7) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<template<typename T> class U> struct Z {};" | 
|  | "template<template<typename T> class U> struct Y {" | 
|  | "  Z<U> A;" | 
|  | "};", | 
|  | "A", | 
|  | "Z<U> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList8) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<typename T> struct Z {};" | 
|  | "template<template<typename T> class U> struct Y {" | 
|  | "  Z<U<int> > A;" | 
|  | "};", | 
|  | "A", | 
|  | "Z<U<int> > A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList9) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<unsigned I> struct Z {};" | 
|  | "Z<0> A;", | 
|  | "A", | 
|  | "Z<0> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList10) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<unsigned I> struct Z {};" | 
|  | "template<unsigned I> struct X { Z<I> A; };", | 
|  | "A", | 
|  | "Z<I> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList11) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<int I> struct Z {};" | 
|  | "Z<42 * 10 - 420 / 1> A;", | 
|  | "A", | 
|  | "Z<42 * 10 - 420 / 1> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList12) { | 
|  | ASSERT_TRUE(PrintedDeclCXX98Matches( | 
|  | "template<const char *p> struct Z {};" | 
|  | "extern const char X[] = \"aaa\";" | 
|  | "Z<X> A;", | 
|  | "A", | 
|  | "Z<X> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList13) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<typename... T> struct Z {};" | 
|  | "template<typename... T> struct X {" | 
|  | "  Z<T...> A;" | 
|  | "};", | 
|  | "A", | 
|  | "Z<T...> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList14) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<typename... T> struct Z {};" | 
|  | "template<typename T> struct Y {};" | 
|  | "template<typename... T> struct X {" | 
|  | "  Z<Y<T>...> A;" | 
|  | "};", | 
|  | "A", | 
|  | "Z<Y<T>...> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestTemplateArgumentList15) { | 
|  | ASSERT_TRUE(PrintedDeclCXX11Matches( | 
|  | "template<unsigned I> struct Z {};" | 
|  | "template<typename... T> struct X {" | 
|  | "  Z<sizeof...(T)> A;" | 
|  | "};", | 
|  | "A", | 
|  | "Z<sizeof...(T)> A")); | 
|  | // Should be: with semicolon | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestStaticAssert1) { | 
|  | ASSERT_TRUE(PrintedDeclCXX1ZMatches( | 
|  | "static_assert(true);", | 
|  | staticAssertDecl().bind("id"), | 
|  | "static_assert(true)")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestObjCMethod1) { | 
|  | ASSERT_TRUE(PrintedDeclObjCMatches( | 
|  | "__attribute__((objc_root_class)) @interface X\n" | 
|  | "- (int)A:(id)anObject inRange:(long)range;\n" | 
|  | "@end\n" | 
|  | "@implementation X\n" | 
|  | "- (int)A:(id)anObject inRange:(long)range { int printThis; return 0; }\n" | 
|  | "@end\n", | 
|  | namedDecl(hasName("A:inRange:"), | 
|  | hasDescendant(namedDecl(hasName("printThis")))).bind("id"), | 
|  | "- (int) A:(id)anObject inRange:(long)range")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestObjCProtocol1) { | 
|  | ASSERT_TRUE(PrintedDeclObjCMatches( | 
|  | "@protocol P1, P2;", | 
|  | namedDecl(hasName("P1")).bind("id"), | 
|  | "@protocol P1;\n")); | 
|  | ASSERT_TRUE(PrintedDeclObjCMatches( | 
|  | "@protocol P1, P2;", | 
|  | namedDecl(hasName("P2")).bind("id"), | 
|  | "@protocol P2;\n")); | 
|  | } | 
|  |  | 
|  | TEST(DeclPrinter, TestObjCProtocol2) { | 
|  | ASSERT_TRUE(PrintedDeclObjCMatches( | 
|  | "@protocol P2 @end" | 
|  | "@protocol P1<P2> @end", | 
|  | namedDecl(hasName("P1")).bind("id"), | 
|  | "@protocol P1<P2>\n@end")); | 
|  | } |