aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMark Mitchell <mark@codesourcery.com>2003-12-03 19:05:12 +0000
committerMark Mitchell <mmitchel@gcc.gnu.org>2003-12-03 19:05:12 +0000
commitee43dab5e00224b3c0ec40112e65802c966e3343 (patch)
tree706e464c3361e136a06a7c457742151daf6c4595
parentbee6ab3e7241334f1a3550c4e0ac9fe8be715a6b (diff)
downloadgcc-ee43dab5e00224b3c0ec40112e65802c966e3343.zip
gcc-ee43dab5e00224b3c0ec40112e65802c966e3343.tar.gz
gcc-ee43dab5e00224b3c0ec40112e65802c966e3343.tar.bz2
re PR c++/10771 (Unreadable error message for illegal specialization)
PR c++/10771 * parser.c (cp_parser_check_for_invalid_template_id): New function. (cp_parser_simple_type_specifier): Use it. (cp_parser_elaborated_type_specifier): Likewise. (cp_parser_class_head): Likewise. PR c++/10771 * g++.dg/template/error5.C: New test. From-SVN: r74239
-rw-r--r--gcc/cp/ChangeLog9
-rw-r--r--gcc/cp/parser.c70
-rw-r--r--gcc/testsuite/ChangeLog5
-rw-r--r--gcc/testsuite/g++.dg/template/error5.C6
4 files changed, 79 insertions, 11 deletions
diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog
index 7b7f63a..de5c714 100644
--- a/gcc/cp/ChangeLog
+++ b/gcc/cp/ChangeLog
@@ -1,3 +1,12 @@
+2003-12-03 Mark Mitchell <mark@codesourcery.com>
+
+ PR c++/10771
+ * parser.c (cp_parser_check_for_invalid_template_id): New
+ function.
+ (cp_parser_simple_type_specifier): Use it.
+ (cp_parser_elaborated_type_specifier): Likewise.
+ (cp_parser_class_head): Likewise.
+
2003-12-02 Giovanni Bajo <giovannibajo@gcc.gnu.org>
PR c++/10126
diff --git a/gcc/cp/parser.c b/gcc/cp/parser.c
index c157173..9ec128c 100644
--- a/gcc/cp/parser.c
+++ b/gcc/cp/parser.c
@@ -1702,6 +1702,8 @@ static void cp_parser_check_type_definition
(cp_parser *);
static void cp_parser_check_for_definition_in_return_type
(tree, int);
+static void cp_parser_check_for_invalid_template_id
+ (cp_parser *, tree);
static tree cp_parser_non_constant_expression
(const char *);
static bool cp_parser_diagnose_invalid_type_name
@@ -1813,6 +1815,54 @@ cp_parser_check_for_definition_in_return_type (tree declarator,
error ("new types may not be defined in a return type");
}
+/* A type-specifier (TYPE) has been parsed which cannot be followed by
+ "<" in any valid C++ program. If the next token is indeed "<",
+ issue a message warning the user about what appears to be an
+ invalid attempt to form a template-id. */
+
+static void
+cp_parser_check_for_invalid_template_id (cp_parser* parser,
+ tree type)
+{
+ ptrdiff_t start;
+ cp_token *token;
+
+ if (cp_lexer_next_token_is (parser->lexer, CPP_LESS))
+ {
+ if (TYPE_P (type))
+ error ("`%T' is not a template", type);
+ else if (TREE_CODE (type) == IDENTIFIER_NODE)
+ error ("`%s' is not a template", IDENTIFIER_POINTER (type));
+ else
+ error ("invalid template-id");
+ /* Remember the location of the invalid "<". */
+ if (cp_parser_parsing_tentatively (parser)
+ && !cp_parser_committed_to_tentative_parse (parser))
+ {
+ token = cp_lexer_peek_token (parser->lexer);
+ token = cp_lexer_prev_token (parser->lexer, token);
+ start = cp_lexer_token_difference (parser->lexer,
+ parser->lexer->first_token,
+ token);
+ }
+ else
+ start = -1;
+ /* Consume the "<". */
+ cp_lexer_consume_token (parser->lexer);
+ /* Parse the template arguments. */
+ cp_parser_enclosed_template_argument_list (parser);
+ /* Permanently remove the invalid template arugments so that
+ this error message is not issued again. */
+ if (start >= 0)
+ {
+ token = cp_lexer_advance_token (parser->lexer,
+ parser->lexer->first_token,
+ start);
+ cp_lexer_purge_tokens_after (parser->lexer, token);
+ }
+ }
+}
+
/* Issue an error message about the fact that THING appeared in a
constant-expression. Returns ERROR_MARK_NODE. */
@@ -8520,17 +8570,8 @@ cp_parser_simple_type_specifier (cp_parser* parser, cp_parser_flags flags,
/* There is no valid C++ program where a non-template type is
followed by a "<". That usually indicates that the user thought
that the type was a template. */
- if (type && cp_lexer_next_token_is (parser->lexer, CPP_LESS))
- {
- error ("`%T' is not a template", TREE_TYPE (type));
- /* Consume the "<". */
- cp_lexer_consume_token (parser->lexer);
- /* Parse the template arguments. */
- cp_parser_enclosed_template_argument_list (parser);
- /* Attempt to recover by using the basic type, ignoring the
- template arguments. */
- return type;
- }
+ if (type)
+ cp_parser_check_for_invalid_template_id (parser, TREE_TYPE (type));
return type;
}
@@ -8827,6 +8868,11 @@ cp_parser_elaborated_type_specifier (cp_parser* parser,
}
if (tag_type != enum_type)
cp_parser_check_class_key (tag_type, type);
+
+ /* A "<" cannot follow an elaborated type specifier. If that
+ happens, the user was probably trying to form a template-id. */
+ cp_parser_check_for_invalid_template_id (parser, type);
+
return type;
}
@@ -11664,6 +11710,8 @@ cp_parser_class_head (cp_parser* parser,
pop_deferring_access_checks ();
+ cp_parser_check_for_invalid_template_id (parser, id);
+
/* If it's not a `:' or a `{' then we can't really be looking at a
class-head, since a class-head only appears as part of a
class-specifier. We have to detect this situation before calling
diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog
index bb142e5..b6883f5 100644
--- a/gcc/testsuite/ChangeLog
+++ b/gcc/testsuite/ChangeLog
@@ -1,3 +1,8 @@
+2003-12-03 Mark Mitchell <mark@codesourcery.com>
+
+ PR c++/10771
+ * g++.dg/template/error5.C: New test.
+
2003-12-02 David Ung <davidu@mips.com>
* gcc.dg/compat/vector-check.h: Corrected type for var
diff --git a/gcc/testsuite/g++.dg/template/error5.C b/gcc/testsuite/g++.dg/template/error5.C
new file mode 100644
index 0000000..0c79350
--- /dev/null
+++ b/gcc/testsuite/g++.dg/template/error5.C
@@ -0,0 +1,6 @@
+template <typename T>
+struct X<T*> { // { dg-error "not a template" }
+ typedef int Y;
+};
+
+extern struct Z<int> s; // { dg-error "not a template" }