llvm/clang/test/SemaTemplate/typo-template-name.cpp

// RUN: %clang_cc1 -std=c++1z %s -verify -Wno-unused

namespace InExpr {
  namespace A {
    void typo_first_a(); // expected-note {{found}}
    template<typename T> void typo_first_b(); // expected-note 2{{declared here}}
  }
  void testA() { A::typo_first_a<int>(); } // expected-error {{'typo_first_a' does not name a template but is followed by template arguments; did you mean 'typo_first_b'?}}

  namespace B {
    void typo_first_b(); // expected-note {{found}}
  }
  void testB() { B::typo_first_b<int>(); } // expected-error {{'typo_first_b' does not name a template but is followed by template arguments; did you mean 'A::typo_first_b'?}}

  struct Base {
    template<typename T> static void foo(); // expected-note 4{{declared here}}
    int n;
  };
  struct Derived : Base {
    void foo(); // expected-note {{found}}
  };
  // We probably don't want to suggest correcting to .Base::foo<int>
  void testMember() { Derived().foo<int>(); } // expected-error-re {{does not name a template but is followed by template arguments{{$}}}}

  struct Derived2 : Base {
    void goo(); // expected-note {{found}}
  };
  void testMember2() { Derived2().goo<int>(); } // expected-error {{member 'goo' of 'InExpr::Derived2' is not a template; did you mean 'foo'?}}

  void no_correction() {
    int foo; // expected-note 3{{found}}

    foo<int>(); // expected-error {{'foo' does not name a template but is followed by template arguments; did you mean 'Base::foo'?}}
    foo<>(); // expected-error {{'foo' does not name a template but is followed by template arguments; did you mean 'Base::foo'?}}
    foo<Base *>(); // expected-error {{'foo' does not name a template but is followed by template arguments; did you mean 'Base::foo'?}}

    // These are valid expressions.
    foo<foo; // expected-warning {{self-comparison}}
    foo<int()>(0); // expected-warning {{comparisons like 'X<=Y<=Z' don't have their mathematical meaning}}
    foo<int(), true>(false);
    foo<Base{}.n;
  }
}