Fix handling of namespace-scope undeduced auto decls.
authorJason Merrill <jason@redhat.com>
Fri, 23 Aug 2019 23:29:16 +0000 (19:29 -0400)
committerJason Merrill <jason@gcc.gnu.org>
Fri, 23 Aug 2019 23:29:16 +0000 (19:29 -0400)
* decl2.c (decl_dependent_p): New.
(mark_used): Check it instead of just processing_template_decl.

From-SVN: r274894

gcc/cp/ChangeLog
gcc/cp/decl2.c
gcc/testsuite/g++.dg/cpp1z/constexpr-if30.C [new file with mode: 0644]

index 06437559480c1754a28fabfc7abd38b445d77bb8..86d1849acc4b31bfd5e1a7c442c61c91b406de3a 100644 (file)
@@ -1,3 +1,8 @@
+2019-08-22  Jason Merrill  <jason@redhat.com>
+
+       * decl2.c (decl_dependent_p): New.
+       (mark_used): Check it instead of just processing_template_decl.
+
 2019-08-22  Jason Merrill  <jason@redhat.com>
 
        * parser.c (cp_parser_nested_name_specifier_opt): Avoid redundant
index aca37a28f494e8b6ac6188084ac6ef8189e9fcf3..36c6f4ce06ab28169f3d84268c9db49b1bc7e94d 100644 (file)
@@ -5425,6 +5425,25 @@ cp_warn_deprecated_use_scopes (tree scope)
     }
 }
 
+/* True if DECL or its enclosing scope have unbound template parameters.  */
+
+bool
+decl_dependent_p (tree decl)
+{
+  if (DECL_FUNCTION_SCOPE_P (decl)
+      || TREE_CODE (decl) == CONST_DECL
+      || TREE_CODE (decl) == USING_DECL
+      || TREE_CODE (decl) == FIELD_DECL)
+    decl = CP_DECL_CONTEXT (decl);
+  if (tree tinfo = get_template_info (decl))
+    if (any_dependent_template_arguments_p (TI_ARGS (tinfo)))
+      return true;
+  if (LAMBDA_FUNCTION_P (decl)
+      && dependent_type_p (DECL_CONTEXT (decl)))
+    return true;
+  return false;
+}
+
 /* Mark DECL (either a _DECL or a BASELINK) as "used" in the program.
    If DECL is a specialization or implicitly declared class member,
    generate the actual definition.  Return false if something goes
@@ -5451,6 +5470,9 @@ mark_used (tree decl, tsubst_flags_t complain)
       decl = OVL_FIRST (decl);
     }
 
+  if (!DECL_P (decl))
+    return true;
+
   /* Set TREE_USED for the benefit of -Wunused.  */
   TREE_USED (decl) = 1;
   /* And for structured bindings also the underlying decl.  */
@@ -5498,7 +5520,7 @@ mark_used (tree decl, tsubst_flags_t complain)
       || DECL_LANG_SPECIFIC (decl) == NULL
       || DECL_THUNK_P (decl))
     {
-      if (!processing_template_decl
+      if (!decl_dependent_p (decl)
          && !require_deduced_type (decl, complain))
        return false;
       return true;
diff --git a/gcc/testsuite/g++.dg/cpp1z/constexpr-if30.C b/gcc/testsuite/g++.dg/cpp1z/constexpr-if30.C
new file mode 100644 (file)
index 0000000..1e3d15d
--- /dev/null
@@ -0,0 +1,10 @@
+// { dg-do compile { target c++17 } }
+
+auto fn = [](auto i) {
+    if constexpr (sizeof(i) == 1)
+        return fn(123);                // { dg-error "auto" }
+};
+
+int main() {
+    fn('!');
+}