From: Jason Merrill Date: Wed, 3 Mar 2021 04:59:00 +0000 (-0500) Subject: c++: C++17 and decltype of multi-operator expression [PR95675] X-Git-Tag: releases/gcc-10.3.0~250 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=371c7a82833df5d84392838cf2bf95b4da88655c;p=thirdparty%2Fgcc.git c++: C++17 and decltype of multi-operator expression [PR95675] A call that is the immediate operand of decltype has special semantics: no temporary is produced, so it's OK for the return type to be e.g. incomplete. But we were treating (e | f) the same way, which confused overload resolution when we then tried to evaluate ... | g. Fixed by making build_temp do what its name says, and force the C++17 temporary materialization conversion. gcc/cp/ChangeLog: PR c++/95675 * call.c (build_temp): Wrap a CALL_EXPR in a TARGET_EXPR if it didn't get one before. gcc/testsuite/ChangeLog: PR c++/95675 * g++.dg/cpp0x/decltype-call5.C: New test. * g++.dg/cpp0x/decltype-call6.C: New test. --- diff --git a/gcc/cp/call.c b/gcc/cp/call.c index d2c810038eae..054a31d3ff3a 100644 --- a/gcc/cp/call.c +++ b/gcc/cp/call.c @@ -7123,6 +7123,14 @@ build_temp (tree expr, tree type, int flags, && !type_has_nontrivial_copy_init (TREE_TYPE (expr))) return get_target_expr_sfinae (expr, complain); + /* In decltype, we might have decided not to wrap this call in a TARGET_EXPR. + But it turns out to be a subexpression, so perform temporary + materialization now. */ + if (TREE_CODE (expr) == CALL_EXPR + && CLASS_TYPE_P (type) + && same_type_ignoring_top_level_qualifiers_p (type, TREE_TYPE (expr))) + expr = build_cplus_new (type, expr, complain); + savew = warningcount + werrorcount, savee = errorcount; releasing_vec args (make_tree_vector_single (expr)); expr = build_special_member_call (NULL_TREE, complete_ctor_identifier, diff --git a/gcc/testsuite/g++.dg/cpp0x/decltype-call5.C b/gcc/testsuite/g++.dg/cpp0x/decltype-call5.C new file mode 100644 index 000000000000..81ef6e6f9c94 --- /dev/null +++ b/gcc/testsuite/g++.dg/cpp0x/decltype-call5.C @@ -0,0 +1,7 @@ +// PR c++/95675 +// { dg-do compile { target c++11 } } + +struct b {}; +b operator|(b, b) { return {}; } +b e, f, g; +using h = decltype(e | f | g); diff --git a/gcc/testsuite/g++.dg/cpp0x/decltype-call6.C b/gcc/testsuite/g++.dg/cpp0x/decltype-call6.C new file mode 100644 index 000000000000..4173b6076895 --- /dev/null +++ b/gcc/testsuite/g++.dg/cpp0x/decltype-call6.C @@ -0,0 +1,12 @@ +// PR c++/95675 +// { dg-do compile { target c++11 } } + +struct a {}; +template struct b; +template struct b { + decltype(bq()(br())) c; +}; +struct e { + operator a(); +}; +b d;