c++: Fix convert_like in template [PR91465, PR93870, PR92031, PR94068]
authorMarek Polacek <polacek@redhat.com>
Fri, 28 Feb 2020 21:57:04 +0000 (16:57 -0500)
committerMarek Polacek <polacek@redhat.com>
Mon, 9 Mar 2020 23:54:51 +0000 (19:54 -0400)
The point of this patch is to fix the recurring problem of trees
generated by convert_like while processing a template that break when
substituting.  For instance, when convert_like creates a CALL_EXPR
while in a template, substituting such a call breaks in finish_call_expr
because we have two 'this' arguments.  Another problem is that we
can create &TARGET_EXPR<> and then fail when substituting because we're
taking the address of an rvalue.  I've analyzed some of the already fixed
PRs and also some of the currently open ones:

In c++/93870 we create EnumWrapper<E>::operator E(&operator~(E)).
In c++/87145 we create S::operator int (&{N}).
In c++/92031 we create &TARGET_EXPR <0>.

The gist of the problem is when convert_like_real creates a call for
a ck_user or wraps a TARGET_EXPR in & in a template.  So in these cases
use IMPLICIT_CONV_EXPR.  In a template we shouldn't need to perform the
actual conversion, we only need it's result type.
perform_direct_initialization_if_possible and
perform_implicit_conversion_flags can also create an IMPLICIT_CONV_EXPR.

Given the change above, build_converted_constant_expr can return an
IMPLICIT_CONV_EXPR so call fold_non_dependent_expr rather than
maybe_constant_value to deal with that.

To avoid the problem of instantiating something twice in a row I'm
removing a call to instantiate_non_dependent_expr_sfinae in
compute_array_index_type_loc.  And the build_converted_constant_expr
pattern can now be simplified.

2020-03-09  Marek Polacek  <polacek@redhat.com>

PR c++/92031 - bogus taking address of rvalue error.
PR c++/91465 - ICE with template codes in check_narrowing.
PR c++/93870 - wrong error when converting template non-type arg.
PR c++/94068 - ICE with template codes in check_narrowing.
* call.c (convert_like_real): Return IMPLICIT_CONV_EXPR
in a template when not ck_identity and we're dealing with a class.
(convert_like_real) <case ck_ref_bind>: Return IMPLICIT_CONV_EXPR
in a template if we need a temporary.
* decl.c (compute_array_index_type_loc): Remove
instantiate_non_dependent_expr_sfinae call.  Call
fold_non_dependent_expr instead of maybe_constant_value.
(build_explicit_specifier): Don't instantiate or create a sentinel
before converting the expression.
* except.c (build_noexcept_spec): Likewise.
* pt.c (convert_nontype_argument): Don't build IMPLICIT_CONV_EXPR.
Set IMPLICIT_CONV_EXPR_NONTYPE_ARG if that's what
build_converted_constant_expr returned.
* typeck2.c (check_narrowing): Call fold_non_dependent_expr instead
of maybe_constant_value.

* g++.dg/cpp0x/conv-tmpl2.C: New test.
* g++.dg/cpp0x/conv-tmpl3.C: New test.
* g++.dg/cpp0x/conv-tmpl4.C: New test.
* g++.dg/cpp0x/conv-tmpl5.C: New test.
* g++.dg/cpp0x/conv-tmpl6.C: New test.
* g++.dg/cpp1z/conv-tmpl1.C: New test.

13 files changed:
gcc/cp/ChangeLog
gcc/cp/call.c
gcc/cp/decl.c
gcc/cp/except.c
gcc/cp/pt.c
gcc/cp/typeck2.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/cpp0x/conv-tmpl2.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp0x/conv-tmpl3.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp0x/conv-tmpl4.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp0x/conv-tmpl5.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp0x/conv-tmpl6.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp1z/conv-tmpl1.C [new file with mode: 0644]

index 7b3b9560571d0ffa4d32dc6b7c1004addd29702f..d9807ad83805b156799231ebfd5316eb4f049478 100644 (file)
@@ -1,3 +1,25 @@
+2020-03-09  Marek Polacek  <polacek@redhat.com>
+
+       PR c++/92031 - bogus taking address of rvalue error.
+       PR c++/91465 - ICE with template codes in check_narrowing.
+       PR c++/93870 - wrong error when converting template non-type arg.
+       PR c++/94068 - ICE with template codes in check_narrowing.
+       * call.c (convert_like_real): Return IMPLICIT_CONV_EXPR
+       in a template when not ck_identity and we're dealing with a class.
+       (convert_like_real) <case ck_ref_bind>: Return IMPLICIT_CONV_EXPR
+       in a template if we need a temporary.
+       * decl.c (compute_array_index_type_loc): Remove
+       instantiate_non_dependent_expr_sfinae call.  Call
+       fold_non_dependent_expr instead of maybe_constant_value.
+       (build_explicit_specifier): Don't instantiate or create a sentinel
+       before converting the expression.
+       * except.c (build_noexcept_spec): Likewise.
+       * pt.c (convert_nontype_argument): Don't build IMPLICIT_CONV_EXPR.
+       Set IMPLICIT_CONV_EXPR_NONTYPE_ARG if that's what
+       build_converted_constant_expr returned.
+       * typeck2.c (check_narrowing): Call fold_non_dependent_expr instead
+       of maybe_constant_value.
+
 2020-03-09  Jakub Jelinek  <jakub@redhat.com>
 
        PR c++/94067
index c0340d96f3c817360b5c66182f355ce81405405f..5767a8b8c4e6b2fc062457ebc3ba2e95d978a662 100644 (file)
@@ -7377,6 +7377,16 @@ convert_like_real (conversion *convs, tree expr, tree fn, int argnum,
   if (issue_conversion_warnings && (complain & tf_warning))
     conversion_null_warnings (totype, expr, fn, argnum);
 
+  /* Creating &TARGET_EXPR<> in a template breaks when substituting,
+     and creating a CALL_EXPR in a template breaks in finish_call_expr
+     so use an IMPLICIT_CONV_EXPR for this conversion.  We would have
+     created such codes e.g. when calling a user-defined conversion
+     function.  */
+  if (processing_template_decl
+      && convs->kind != ck_identity
+      && (CLASS_TYPE_P (totype) || CLASS_TYPE_P (TREE_TYPE (expr))))
+    return build1 (IMPLICIT_CONV_EXPR, totype, expr);
+
   switch (convs->kind)
     {
     case ck_user:
@@ -7763,6 +7773,14 @@ convert_like_real (conversion *convs, tree expr, tree fn, int argnum,
                expr = convert_bitfield_to_declared_type (expr);
                expr = fold_convert (type, expr);
              }
+
+           /* Creating &TARGET_EXPR<> in a template would break when
+              tsubsting the expression, so use an IMPLICIT_CONV_EXPR
+              instead.  This can happen even when there's no class
+              involved, e.g., when converting an integer to a reference
+              type.  */
+           if (processing_template_decl)
+             return build1 (IMPLICIT_CONV_EXPR, totype, expr);
            expr = build_target_expr_with_type (expr, type, complain);
          }
 
index e3f4b435a4905d38e66215a8343f96429af834a0..bb2427430746cf6889be72bd2e47170754ede8d2 100644 (file)
@@ -10276,13 +10276,12 @@ compute_array_index_type_loc (location_t name_loc, tree name, tree size,
           NOP_EXPR with TREE_SIDE_EFFECTS; don't fold in that case.  */;
       else
        {
-         size = instantiate_non_dependent_expr_sfinae (size, complain);
          size = build_converted_constant_expr (size_type_node, size, complain);
          /* Pedantically a constant expression is required here and so
             __builtin_is_constant_evaluated () should fold to true if it
             is successfully folded into a constant.  */
-         size = maybe_constant_value (size, NULL_TREE,
-                                      /*manifestly_const_eval=*/true);
+         size = fold_non_dependent_expr (size, complain,
+                                         /*manifestly_const_eval=*/true);
 
          if (!TREE_CONSTANT (size))
            size = origsize;
@@ -17607,10 +17606,8 @@ build_explicit_specifier (tree expr, tsubst_flags_t complain)
     /* Wait for instantiation, tsubst_function_decl will handle it.  */
     return expr;
 
-  expr = instantiate_non_dependent_expr_sfinae (expr, complain);
-  /* Don't let convert_like_real create more template codes.  */
-  processing_template_decl_sentinel s;
   expr = build_converted_constant_bool_expr (expr, complain);
+  expr = instantiate_non_dependent_expr_sfinae (expr, complain);
   expr = cxx_constant_value (expr);
   return expr;
 }
index 788b96de243e0dc7796283ab49f13eb04d27f283..262ba5d309cfc8abf8b0241bbf7b774429533944 100644 (file)
@@ -1299,10 +1299,8 @@ build_noexcept_spec (tree expr, tsubst_flags_t complain)
   if (TREE_CODE (expr) != DEFERRED_NOEXCEPT
       && !value_dependent_expression_p (expr))
     {
-      expr = instantiate_non_dependent_expr_sfinae (expr, complain);
-      /* Don't let convert_like_real create more template codes.  */
-      processing_template_decl_sentinel s;
       expr = build_converted_constant_bool_expr (expr, complain);
+      expr = instantiate_non_dependent_expr_sfinae (expr, complain);
       expr = cxx_constant_value (expr);
     }
   if (TREE_CODE (expr) == INTEGER_CST)
index 1c721b311764a312068d996315d738d6cfa3308f..49ee3920049e3cf6d2eda5dfd8c76ef47419f470 100644 (file)
@@ -7068,26 +7068,6 @@ convert_nontype_argument (tree type, tree expr, tsubst_flags_t complain)
       else if (INTEGRAL_OR_ENUMERATION_TYPE_P (type)
               || cxx_dialect >= cxx17)
        {
-         /* Calling build_converted_constant_expr might create a call to
-            a conversion function with a value-dependent argument, which
-            could invoke taking the address of a temporary representing
-            the result of the conversion.  */
-         if (!same_type_ignoring_top_level_qualifiers_p (type, expr_type)
-             && ((COMPOUND_LITERAL_P (expr)
-                  && CONSTRUCTOR_IS_DEPENDENT (expr)
-                  && MAYBE_CLASS_TYPE_P (expr_type)
-                  && TYPE_HAS_CONVERSION (expr_type))
-                 /* Similarly, converting e.g. an integer to a class
-                    involves a constructor call.  convert_like would
-                    create a TARGET_EXPR, but in a template we can't
-                    use AGGR_INIT_EXPR, and the TARGET_EXPR would lead
-                    to a bogus error.  */
-                 || (val_dep_p && MAYBE_CLASS_TYPE_P (type))))
-           {
-             expr = build1 (IMPLICIT_CONV_EXPR, type, expr);
-             IMPLICIT_CONV_EXPR_NONTYPE_ARG (expr) = true;
-             return expr;
-           }
          /* C++17: A template-argument for a non-type template-parameter shall
             be a converted constant expression (8.20) of the type of the
             template-parameter.  */
@@ -7096,6 +7076,11 @@ convert_nontype_argument (tree type, tree expr, tsubst_flags_t complain)
            /* Make sure we return NULL_TREE only if we have really issued
               an error, as described above.  */
            return (complain & tf_error) ? NULL_TREE : error_mark_node;
+         else if (TREE_CODE (expr) == IMPLICIT_CONV_EXPR)
+           {
+             IMPLICIT_CONV_EXPR_NONTYPE_ARG (expr) = true;
+             return expr;
+           }
          expr = maybe_constant_value (expr, NULL_TREE,
                                       /*manifestly_const_eval=*/true);
          expr = convert_from_reference (expr);
index 48920894b3baa07161c3d52a51f3104b98928af6..bff4ddbcf81de7930fd300055541186fc85a1c38 100644 (file)
@@ -981,7 +981,11 @@ check_narrowing (tree type, tree init, tsubst_flags_t complain,
       return ok;
     }
 
-  init = maybe_constant_value (init);
+  /* Even non-dependent expressions can still have template
+     codes like CAST_EXPR, so use *_non_dependent_expr to cope.  */
+  init = fold_non_dependent_expr (init, complain);
+  if (init == error_mark_node)
+    return ok;
 
   /* If we were asked to only check constants, return early.  */
   if (const_only && !TREE_CONSTANT (init))
index 44f65ffa2049396c7a685efdd732db57309ce7cd..f91af78a302461df05c33e9b1675f8af2dc0c68e 100644 (file)
@@ -1,3 +1,16 @@
+2020-03-09  Marek Polacek  <polacek@redhat.com>
+
+       PR c++/92031 - bogus taking address of rvalue error.
+       PR c++/91465 - ICE with template codes in check_narrowing.
+       PR c++/93870 - wrong error when converting template non-type arg.
+       PR c++/94068 - ICE with template codes in check_narrowing.
+       * g++.dg/cpp0x/conv-tmpl2.C: New test.
+       * g++.dg/cpp0x/conv-tmpl3.C: New test.
+       * g++.dg/cpp0x/conv-tmpl4.C: New test.
+       * g++.dg/cpp0x/conv-tmpl5.C: New test.
+       * g++.dg/cpp0x/conv-tmpl6.C: New test.
+       * g++.dg/cpp1z/conv-tmpl1.C: New test.
+
 2020-03-09  Jakub Jelinek  <jakub@redhat.com>
 
        PR c++/94067
diff --git a/gcc/testsuite/g++.dg/cpp0x/conv-tmpl2.C b/gcc/testsuite/g++.dg/cpp0x/conv-tmpl2.C
new file mode 100644 (file)
index 0000000..8a50576
--- /dev/null
@@ -0,0 +1,21 @@
+// PR c++/92031 - bogus taking address of rvalue error.
+// { dg-do compile { target c++11 } }
+
+struct x { const int& l; };
+
+void a(const x&) {}
+
+template<class E>                          
+void f() {
+  a(x { 0 });
+}
+
+void g() {
+  a(x { 0 });
+}
+
+void
+test ()
+{
+  f<int>();
+}
diff --git a/gcc/testsuite/g++.dg/cpp0x/conv-tmpl3.C b/gcc/testsuite/g++.dg/cpp0x/conv-tmpl3.C
new file mode 100644 (file)
index 0000000..e2021aa
--- /dev/null
@@ -0,0 +1,16 @@
+// PR c++/91465 - ICE with template codes in check_narrowing.
+// { dg-do compile { target c++11 } }
+
+enum class D { X };
+enum class S { Z };
+
+D foo(S) { return D{}; }
+D foo(double) { return D{}; }
+
+template <typename>
+struct Bar {
+  D baz(S s)
+  {
+    return D{foo(s)};
+  }
+};
diff --git a/gcc/testsuite/g++.dg/cpp0x/conv-tmpl4.C b/gcc/testsuite/g++.dg/cpp0x/conv-tmpl4.C
new file mode 100644 (file)
index 0000000..966a2e1
--- /dev/null
@@ -0,0 +1,33 @@
+// PR c++/93870 - wrong error when converting template non-type arg.
+// { dg-do compile { target c++11 } }
+
+template <typename ENUM> struct EnumWrapper
+{
+       ENUM value;
+
+       constexpr operator ENUM() const
+       {
+               return value;
+       }
+};
+
+enum E : int { V };
+
+constexpr EnumWrapper<E> operator ~(E a)
+{
+    return {E(~int(a))};
+}
+
+template <E X> struct R
+{
+    static void Func();
+};
+
+template <E X> struct S : R<~X>
+{
+};
+
+void Test()
+{
+    S<E::V>::Func();
+}
diff --git a/gcc/testsuite/g++.dg/cpp0x/conv-tmpl5.C b/gcc/testsuite/g++.dg/cpp0x/conv-tmpl5.C
new file mode 100644 (file)
index 0000000..c83e6d8
--- /dev/null
@@ -0,0 +1,13 @@
+// PR c++/94068 - ICE with template codes in check_narrowing.
+// { dg-do compile { target c++11 } }
+
+enum class A { A1, A2 };
+A foo ();
+long foo (int);
+
+template <typename>
+void
+bar ()
+{
+  const auto c{foo ()};
+}
diff --git a/gcc/testsuite/g++.dg/cpp0x/conv-tmpl6.C b/gcc/testsuite/g++.dg/cpp0x/conv-tmpl6.C
new file mode 100644 (file)
index 0000000..2df3a6c
--- /dev/null
@@ -0,0 +1,16 @@
+// { dg-do compile { target c++11 } }
+
+struct A
+{
+  constexpr A(int) { }
+  constexpr operator int() const { return 1; };
+};
+
+template <class T, int N>
+struct B
+{
+  static constexpr A a = A(N);
+  int ar[a];
+};
+
+B<int, 10> b;
diff --git a/gcc/testsuite/g++.dg/cpp1z/conv-tmpl1.C b/gcc/testsuite/g++.dg/cpp1z/conv-tmpl1.C
new file mode 100644 (file)
index 0000000..5b12053
--- /dev/null
@@ -0,0 +1,10 @@
+// PR c++/91465 - ICE with template codes in check_narrowing.
+// { dg-do compile { target c++17 } }
+
+enum class E { Z };
+
+template <typename F>
+void foo(F)
+{
+  E{char(0)};
+}