[PR c++/71965] silence multi-dim array init sorry without tf_error

We shouldn't substitute templates into short-circuited-out concepts
constraints, but we do, and to add insult to injury, we issue a
sorry() error when a concept that shouldn't even have been substituted
attempts to perform a multi-dimensional array initialization with a
new{} expression.

Although fixing the requirements short-circuiting is probably too
risky at this point, we can get closer to the intended effect by
silencing that sorry just as we silence other errors.

for  gcc/cp/ChangeLog

	PR c++/71965
	* init.c (build_vec_init): Silence error, former sorry,
	without tf_error.

for  gcc/testsuite/ChangeLog

	PR c++/71965
	* g++.dg/concepts/pr71965.C: New.

From-SVN: r258749
This commit is contained in:
Alexandre Oliva 2018-03-21 22:08:34 +00:00 committed by Alexandre Oliva
parent 7446b353b5
commit 0186f68450
4 changed files with 46 additions and 7 deletions

View File

@ -1,5 +1,9 @@
2018-03-21 Alexandre Oliva <aoliva@redhat.com> 2018-03-21 Alexandre Oliva <aoliva@redhat.com>
PR c++/71965
* init.c (build_vec_init): Silence error, former sorry,
without tf_error.
PR c++/84610 PR c++/84610
PR c++/84642 PR c++/84642
* parser.c (abort_fully_implicit_template_p): New. * parser.c (abort_fully_implicit_template_p): New.

View File

@ -4384,8 +4384,13 @@ build_vec_init (tree base, tree maxindex, tree init,
else if (TREE_CODE (type) == ARRAY_TYPE) else if (TREE_CODE (type) == ARRAY_TYPE)
{ {
if (init && !BRACE_ENCLOSED_INITIALIZER_P (init)) if (init && !BRACE_ENCLOSED_INITIALIZER_P (init))
sorry {
("cannot initialize multi-dimensional array with initializer"); if ((complain & tf_error))
error_at (loc, "array must be initialized "
"with a brace-enclosed initializer");
elt_init = error_mark_node;
}
else
elt_init = build_vec_init (build1 (INDIRECT_REF, type, base), elt_init = build_vec_init (build1 (INDIRECT_REF, type, base),
0, init, 0, init,
explicit_value_init_p, explicit_value_init_p,
@ -4449,7 +4454,7 @@ build_vec_init (tree base, tree maxindex, tree init,
} }
current_stmt_tree ()->stmts_are_full_exprs_p = 1; current_stmt_tree ()->stmts_are_full_exprs_p = 1;
if (elt_init) if (elt_init && !errors)
finish_expr_stmt (elt_init); finish_expr_stmt (elt_init);
current_stmt_tree ()->stmts_are_full_exprs_p = 0; current_stmt_tree ()->stmts_are_full_exprs_p = 0;

View File

@ -1,5 +1,8 @@
2018-03-21 Alexandre Oliva <aoliva@redhat.com> 2018-03-21 Alexandre Oliva <aoliva@redhat.com>
PR c++/71965
* g++.dg/concepts/pr71965.C: New.
PR c++/84610 PR c++/84610
PR c++/84642 PR c++/84642
* g++.dg/cpp0x/pr84610.C: New. * g++.dg/cpp0x/pr84610.C: New.

View File

@ -0,0 +1,27 @@
// { dg-do compile { target c++14 } }
// { dg-options "-fconcepts" }
template <class T>
concept bool Destructible() {
return false;
}
template <class T, class...Args>
concept bool ConstructibleObject =
// Concept evaluation should short-circuit even the template
// substitution, so we shouldn't even substitute into the requires
// constraint and the unimplemented multi-dimensional new T{...}
// initialization. ATM we do, but as long as we don't output the
// sorry() message we used to for such constructs when asked not
// to issue errors, this shouldn't be a problem for this and
// similar cases.
Destructible<T>() && requires (Args&&...args) {
new T{ (Args&&)args... };
};
int main() {
using T = int[2][2];
// GCC has not implemented initialization of multi-dimensional
// arrays with new{} expressions.
static_assert(!ConstructibleObject<T, T>);
}