]> granicus.if.org Git - clang/commitdiff
PR4364: fix parsing 'typename' in an expression.
authorEli Friedman <eli.friedman@gmail.com>
Thu, 11 Jun 2009 00:33:41 +0000 (00:33 +0000)
committerEli Friedman <eli.friedman@gmail.com>
Thu, 11 Jun 2009 00:33:41 +0000 (00:33 +0000)
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@73177 91177308-0d34-0410-b5e6-96231b3b80d8

lib/Parse/ParseExpr.cpp
test/SemaCXX/typename-expression.cpp [new file with mode: 0644]

index cd62c64276b3df4a282f2c6350ca1da2309334c5..3fee78bb719f674aafd5743f48cb854c741e9be2 100644 (file)
@@ -749,7 +749,13 @@ Parser::OwningExprResult Parser::ParseCastExpression(bool isUnaryExpression,
       Diag(Tok, diag::err_expected_expression);
       return ExprError();
     }
-    
+
+    if (SavedKind == tok::kw_typename) {
+      // postfix-expression: typename-specifier '(' expression-list[opt] ')'
+      if (!TryAnnotateTypeOrScopeToken())
+        return ExprError();
+    }
+
     // postfix-expression: simple-type-specifier '(' expression-list[opt] ')'
     //
     DeclSpec DS;
diff --git a/test/SemaCXX/typename-expression.cpp b/test/SemaCXX/typename-expression.cpp
new file mode 100644 (file)
index 0000000..8dde609
--- /dev/null
@@ -0,0 +1,19 @@
+// RUN: clang-cc -fsyntax-only -verify %s
+
+// PR4364
+template<class T> struct a {
+  T b() {
+    return typename T::x();
+  }
+};
+struct B {
+  typedef B x;
+};
+B c() {
+  a<B> x;
+  return x.b();
+}
+
+// Some extra tests for invalid cases
+template<class T> struct test2 { T b() { return typename T::a; } }; // expected-error{{expected '(' for function-style cast or type construction}}
+template<class T> struct test3 { T b() { return typename a; } }; // expected-error{{expected a qualified name after 'typename'}}