diff options
author | Jason Merrill <jason@redhat.com> | 2010-11-08 22:54:24 -0500 |
---|---|---|
committer | Jason Merrill <jason@gcc.gnu.org> | 2010-11-08 22:54:24 -0500 |
commit | 9b7d05093738e51274c3b473ede23799db93bf98 (patch) | |
tree | 98fb4d94e9d9d03d85c305dab606ebd564af92c9 /gcc/cp | |
parent | edb7c512888ad65f3804585a43448ada0d10f690 (diff) | |
download | gcc-9b7d05093738e51274c3b473ede23799db93bf98.zip gcc-9b7d05093738e51274c3b473ede23799db93bf98.tar.gz gcc-9b7d05093738e51274c3b473ede23799db93bf98.tar.bz2 |
re PR c++/46382 (constexpr vs. static_assert in constexpr ctors)
PR c++/46382
* semantics.c (check_constexpr_ctor_body): New fn.
* parser.c (cp_parser_ctor_initializer_opt_and_function_body): Call it.
* cp-tree.h: Declare it.
From-SVN: r166471
Diffstat (limited to 'gcc/cp')
-rw-r--r-- | gcc/cp/ChangeLog | 5 | ||||
-rw-r--r-- | gcc/cp/cp-tree.h | 1 | ||||
-rw-r--r-- | gcc/cp/parser.c | 10 | ||||
-rw-r--r-- | gcc/cp/semantics.c | 42 |
4 files changed, 50 insertions, 8 deletions
diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog index 2408adb..7223529d 100644 --- a/gcc/cp/ChangeLog +++ b/gcc/cp/ChangeLog @@ -1,5 +1,10 @@ 2010-11-08 Jason Merrill <jason@redhat.com> + PR c++/46382 + * semantics.c (check_constexpr_ctor_body): New fn. + * parser.c (cp_parser_ctor_initializer_opt_and_function_body): Call it. + * cp-tree.h: Declare it. + PR c++/46335 * tree.c (bot_manip): Check TREE_SIDE_EFFECTS as well. diff --git a/gcc/cp/cp-tree.h b/gcc/cp/cp-tree.h index 241805c..67f4f93 100644 --- a/gcc/cp/cp-tree.h +++ b/gcc/cp/cp-tree.h @@ -5249,6 +5249,7 @@ extern void finish_cleanup (tree, tree); extern bool literal_type_p (tree); extern tree validate_constexpr_fundecl (tree); extern tree register_constexpr_fundef (tree, tree); +extern bool check_constexpr_ctor_body (tree, tree); extern tree ensure_literal_type_for_constexpr_object (tree); extern bool potential_constant_expression (tree, tsubst_flags_t); extern tree cxx_constant_value (tree); diff --git a/gcc/cp/parser.c b/gcc/cp/parser.c index 6a9e4d7..4897941 100644 --- a/gcc/cp/parser.c +++ b/gcc/cp/parser.c @@ -16347,14 +16347,8 @@ cp_parser_ctor_initializer_opt_and_function_body (cp_parser *parser) } /* Parse the function-body. */ cp_parser_function_body (parser); - if (check_body_p - && (TREE_CODE (list) != STATEMENT_LIST - || (last == NULL && STATEMENT_LIST_TAIL (list) != NULL) - || (last != NULL && last != STATEMENT_LIST_TAIL (list)->stmt))) - { - error ("constexpr constructor does not have empty body"); - DECL_DECLARED_CONSTEXPR_P (current_function_decl) = false; - } + if (check_body_p) + check_constexpr_ctor_body (last, list); /* Finish the function body. */ finish_function_body (body); diff --git a/gcc/cp/semantics.c b/gcc/cp/semantics.c index 494247e7..b48559e 100644 --- a/gcc/cp/semantics.c +++ b/gcc/cp/semantics.c @@ -5522,6 +5522,48 @@ build_data_member_initialization (tree t, VEC(constructor_elt,gc) **vec) return true; } +/* Make sure that there are no statements after LAST in the constructor + body represented by LIST. */ + +bool +check_constexpr_ctor_body (tree last, tree list) +{ + bool ok = true; + if (TREE_CODE (list) == STATEMENT_LIST) + { + tree_stmt_iterator i = tsi_last (list); + for (; !tsi_end_p (i); tsi_prev (&i)) + { + tree t = tsi_stmt (i); + if (t == last) + break; + if (TREE_CODE (t) == BIND_EXPR) + { + if (!check_constexpr_ctor_body (last, BIND_EXPR_BODY (t))) + return false; + else + continue; + } + /* We currently allow typedefs and static_assert. + FIXME allow them in the standard, too. */ + if (TREE_CODE (t) != STATIC_ASSERT) + { + ok = false; + break; + } + } + } + else if (list != last + && TREE_CODE (list) != STATIC_ASSERT) + ok = false; + if (!ok) + { + error ("constexpr constructor does not have empty body"); + DECL_DECLARED_CONSTEXPR_P (current_function_decl) = false; + } + return ok; +} + /* Build compile-time evalable representations of member-initializer list for a constexpr constructor. */ |