This patch uses the nested diagnostics capabilities added in the earlier
patch in the C++ frontend.

With this, and enabling the non-standard text formatting via:
  -fdiagnostics-set-output=text:experimental-nesting=yes
and using:
  -std=c++20 -fconcepts-diagnostics-depth=2
then the output for the example in SG15's P3358R0 ("SARIF for Structured
Diagnostics") is:

P3358R0.C: In function ‘int main()’:
P3358R0.C:26:6: error: no matching function for call to ‘pet(lizard)’
   26 |   pet(lizard{});
      |   ~~~^~~~~~~~~~
  • note: candidate: ‘template<class auto:1>  requires  pettable<auto:1> void 
pet(auto:1)’
    P3358R0.C:21:6:
       21 | void pet(pettable auto t);
          |      ^~~
    • note: template argument deduction/substitution failed:
      • note: constraints not satisfied
        • P3358R0.C: In substitution of ‘template<class auto:1>  requires  
pettable<auto:1> void pet(auto:1) [with auto:1 = lizard]’:
        • required from here
          P3358R0.C:26:6:
             26 |   pet(lizard{});
                |   ~~~^~~~~~~~~~
        • required for the satisfaction of ‘pettable<auto:1>’ [with auto:1 = 
lizard]
          P3358R0.C:19:9:
             19 | concept pettable = has_member_pet<T> or has_default_pet<T>;
                |         ^~~~~~~~
        • note: no operand of the disjunction is satisfied
          P3358R0.C:19:38:
             19 | concept pettable = has_member_pet<T> or has_default_pet<T>;
                |                    ~~~~~~~~~~~~~~~~~~^~~~~~~~~~~~~~~~~~~~~
          • note: the operand ‘has_member_pet<T>’ is unsatisfied because
            P3358R0.C:19:20:
               19 | concept pettable = has_member_pet<T> or has_default_pet<T>;
                  |                    ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
            • required for the satisfaction of ‘has_member_pet<T>’ [with T = 
lizard]
              P3358R0.C:13:9:
                 13 | concept has_member_pet = requires(T t) { t.pet(); };
                    |         ^~~~~~~~~~~~~~
            • required for the satisfaction of ‘pettable<auto:1>’ [with auto:1 
= lizard]
              P3358R0.C:19:9:
                 19 | concept pettable = has_member_pet<T> or 
has_default_pet<T>;
                    |         ^~~~~~~~
            • in requirements with ‘T t’ [with T = lizard]
              P3358R0.C:13:26:
                 13 | concept has_member_pet = requires(T t) { t.pet(); };
                    |                          ^~~~~~~~~~~~~~~~~~~~~~~~~~
            • note: the required expression ‘t.pet()’ is invalid, because
              P3358R0.C:13:47:
                 13 | concept has_member_pet = requires(T t) { t.pet(); };
                    |                                          ~~~~~^~
              • error: ‘struct lizard’ has no member named ‘pet’
                P3358R0.C:13:44:
                   13 | concept has_member_pet = requires(T t) { t.pet(); };
                      |                                          ~~^~~
          • note: the operand ‘has_default_pet<T>’ is unsatisfied because
            P3358R0.C:19:41:
               19 | concept pettable = has_member_pet<T> or has_default_pet<T>;
                  |                    ~~~~~~~~~~~~~~~~~~~~~^~~~~~~~~~~~~~~~~~
            • required for the satisfaction of ‘has_default_pet<T>’ [with T = 
lizard]
              P3358R0.C:16:9:
                 16 | concept has_default_pet = T::is_pettable;
                    |         ^~~~~~~~~~~~~~~
            • required for the satisfaction of ‘pettable<auto:1>’ [with auto:1 
= lizard]
              P3358R0.C:19:9:
                 19 | concept pettable = has_member_pet<T> or 
has_default_pet<T>;
                    |         ^~~~~~~~
            • error: ‘is_pettable’ is not a member of ‘lizard’
              P3358R0.C:16:30:
                 16 | concept has_default_pet = T::is_pettable;
                    |                              ^~~~~~~~~~~
  • note: candidate: ‘void pet(dog)’
    P3358R0.C:9:6:
        9 | void pet(dog);
          |      ^~~
    • note: no known conversion for argument 1 from ‘lizard’ to ‘dog’
      P3358R0.C:9:10:
          9 | void pet(dog);
            |          ^~~
  • note: candidate: ‘void pet(cat)’
    P3358R0.C:10:6:
       10 | void pet(cat);
          |      ^~~
    • note: no known conversion for argument 1 from ‘lizard’ to ‘cat’
      P3358R0.C:10:10:
         10 | void pet(cat);
            |          ^~~

showing the hierarchical structure of the messages; ideally there
would be a UI here allowing the user to expand/collapse the messages
to drill out into the detail they are interested in.

The structure is also captured in SARIF output (via the "nestingLevel"
property).

gcc/cp/ChangeLog:
        PR other/116253
        * call.cc (print_conversion_rejection): Remove leading space from
        diagnostic messages.
        (print_conversion_rejection): Likewise.
        (print_arity_information): Likewise.
        (print_z_candidate): Likewise.  Add auto_diagnostic_nesting_level
        before calls to fn_type_unification and diagnose_constraints.
        (print_z_candidates): Add auto_diagnostic_nesting_level before
        looping over candidates.
        (conversion_null_warnings): Remove leading space from
        diagnostic messages.
        (maybe_inform_about_fndecl_for_bogus_argument_init): Likewise.
        * constraint.cc (tsubst_valid_expression_requirement): Add
        auto_diagnostic_nesting_level when showing why the expression is
        invalid.
        (satisfy_disjunction): Likewise when showing operans, and again
        when replaying each branch of the disjunction.
        (diagnose_constraints): Likewise when replaying satisfaction.
        * error.cc (cp_diagnostic_text_starter): Set prefix.
        (print_instantiation_full_context): Only show the file
        if we're not showing nesting or the user has opted in to
        showing location information in nested diagnostics.
        (class auto_context_line): New.
        (print_instantiation_partial_context_line): Replace calls to
        print_location and to diagnostic_show_locus with an
        auto_context_line.
        (print_instantiation_partial_context): Replace calls to
        print_location with an auto_context_line.
        (maybe_print_constexpr_context): Likewise.
        (print_constrained_decl_info): Likewise.
        (print_concept_check_info): Likewise.
        (print_constraint_context_head): Likewise.
        (print_requires_expression_info): Likewise.

gcc/testsuite/ChangeLog:
        PR other/116253
        * g++.dg/concepts/nested-diagnostics-1-truncated.C: New test.
        * g++.dg/concepts/nested-diagnostics-1.C: New test.
        * g++.dg/concepts/nested-diagnostics-2.C: New test.

Signed-off-by: David Malcolm <dmalc...@redhat.com>
---
 gcc/cp/call.cc                                | 69 +++++++------
 gcc/cp/constraint.cc                          |  5 +
 gcc/cp/error.cc                               | 99 ++++++++++++++++---
 .../concepts/nested-diagnostics-1-truncated.C | 41 ++++++++
 .../g++.dg/concepts/nested-diagnostics-1.C    | 51 ++++++++++
 .../g++.dg/concepts/nested-diagnostics-2.C    | 37 +++++++
 6 files changed, 257 insertions(+), 45 deletions(-)
 create mode 100644 
gcc/testsuite/g++.dg/concepts/nested-diagnostics-1-truncated.C
 create mode 100644 gcc/testsuite/g++.dg/concepts/nested-diagnostics-1.C
 create mode 100644 gcc/testsuite/g++.dg/concepts/nested-diagnostics-2.C

diff --git a/gcc/cp/call.cc b/gcc/cp/call.cc
index 220ac130b0b4..db9eb1a55cfc 100644
--- a/gcc/cp/call.cc
+++ b/gcc/cp/call.cc
@@ -3887,18 +3887,18 @@ print_conversion_rejection (location_t loc, struct 
conversion_info *info,
       /* Conversion of implicit `this' argument failed.  */
       if (!TYPE_P (info->from))
        /* A bad conversion for 'this' must be discarding cv-quals.  */
-       inform (loc, "  passing %qT as %<this%> "
+       inform (loc, "passing %qT as %<this%> "
                "argument discards qualifiers",
                from);
       else
-       inform (loc, "  no known conversion for implicit "
+       inform (loc, "no known conversion for implicit "
                "%<this%> parameter from %qH to %qI",
                from, info->to_type);
     }
   else if (!TYPE_P (info->from))
     {
       if (info->n_arg >= 0)
-       inform (loc, "  conversion of argument %d would be ill-formed:",
+       inform (loc, "conversion of argument %d would be ill-formed:",
                info->n_arg + 1);
       iloc_sentinel ils = loc;
       perform_implicit_conversion (info->to_type, info->from,
@@ -3906,13 +3906,13 @@ print_conversion_rejection (location_t loc, struct 
conversion_info *info,
     }
   else if (info->n_arg == -2)
     /* Conversion of conversion function return value failed.  */
-    inform (loc, "  no known conversion from %qH to %qI",
+    inform (loc, "no known conversion from %qH to %qI",
            from, info->to_type);
   else
     {
       if (TREE_CODE (fn) == FUNCTION_DECL)
        loc = get_fndecl_argument_location (fn, info->n_arg);
-      inform (loc, "  no known conversion for argument %d from %qH to %qI",
+      inform (loc, "no known conversion for argument %d from %qH to %qI",
              info->n_arg + 1, from, info->to_type);
     }
 }
@@ -3926,13 +3926,13 @@ print_arity_information (location_t loc, unsigned int 
have, unsigned int want,
 {
   if (least_p)
     inform_n (loc, want,
-             "  candidate expects at least %d argument, %d provided",
-             "  candidate expects at least %d arguments, %d provided",
+             "candidate expects at least %d argument, %d provided",
+             "candidate expects at least %d arguments, %d provided",
              want, have);
   else
     inform_n (loc, want,
-             "  candidate expects %d argument, %d provided",
-             "  candidate expects %d arguments, %d provided",
+             "candidate expects %d argument, %d provided",
+             "candidate expects %d arguments, %d provided",
              want, have);
 }
 
@@ -3988,11 +3988,12 @@ print_z_candidate (location_t loc, const char *msgstr,
   if (fn != candidate->fn)
     {
       cloc = location_of (candidate->fn);
-      inform (cloc, "  inherited here");
+      inform (cloc, "inherited here");
     }
   /* Give the user some information about why this candidate failed.  */
   if (candidate->reason != NULL)
     {
+      auto_diagnostic_nesting_level sentinel;
       struct rejection_reason *r = candidate->reason;
 
       switch (r->code)
@@ -4009,13 +4010,13 @@ print_z_candidate (location_t loc, const char *msgstr,
          print_conversion_rejection (cloc, &r->u.bad_conversion, fn);
          break;
        case rr_explicit_conversion:
-         inform (cloc, "  return type %qT of explicit conversion function "
+         inform (cloc, "return type %qT of explicit conversion function "
                  "cannot be converted to %qT with a qualification "
                  "conversion", r->u.conversion.from,
                  r->u.conversion.to_type);
          break;
        case rr_template_conversion:
-         inform (cloc, "  conversion from return type %qT of template "
+         inform (cloc, "conversion from return type %qT of template "
                  "conversion function specialization to %qT is not an "
                  "exact match", r->u.conversion.from,
                  r->u.conversion.to_type);
@@ -4026,33 +4027,39 @@ print_z_candidate (location_t loc, const char *msgstr,
             them here.  */
          if (r->u.template_unification.tmpl == NULL_TREE)
            {
-             inform (cloc, "  substitution of deduced template arguments "
+             inform (cloc, "substitution of deduced template arguments "
                      "resulted in errors seen above");
              break;
            }
          /* Re-run template unification with diagnostics.  */
-         inform (cloc, "  template argument deduction/substitution failed:");
-         fn_type_unification (r->u.template_unification.tmpl,
-                              r->u.template_unification.explicit_targs,
-                              (make_tree_vec
-                               (r->u.template_unification.num_targs)),
-                              r->u.template_unification.args,
-                              r->u.template_unification.nargs,
-                              r->u.template_unification.return_type,
-                              r->u.template_unification.strict,
-                              r->u.template_unification.flags,
-                              NULL, true, false);
+         inform (cloc, "template argument deduction/substitution failed:");
+         {
+           auto_diagnostic_nesting_level sentinel;
+           fn_type_unification (r->u.template_unification.tmpl,
+                                r->u.template_unification.explicit_targs,
+                                (make_tree_vec
+                                 (r->u.template_unification.num_targs)),
+                                r->u.template_unification.args,
+                                r->u.template_unification.nargs,
+                                r->u.template_unification.return_type,
+                                r->u.template_unification.strict,
+                                r->u.template_unification.flags,
+                                NULL, true, false);
+         }
          break;
        case rr_invalid_copy:
          inform (cloc,
-                 "  a constructor taking a single argument of its own "
+                 "a constructor taking a single argument of its own "
                  "class type is invalid");
          break;
        case rr_constraint_failure:
-         diagnose_constraints (cloc, fn, NULL_TREE);
+         {
+           auto_diagnostic_nesting_level sentinel;
+           diagnose_constraints (cloc, fn, NULL_TREE);
+         }
          break;
        case rr_inherited_ctor:
-         inform (cloc, "  an inherited constructor is not a candidate for "
+         inform (cloc, "an inherited constructor is not a candidate for "
                  "initialization from an expression of the same or derived "
                  "type");
          break;
@@ -4125,6 +4132,8 @@ print_z_candidates (location_t loc, struct z_candidate 
*candidates,
   if (only_viable_p.is_unknown ())
     only_viable_p = candidates->viable == 1;
 
+  auto_diagnostic_nesting_level sentinel;
+
   for (; candidates; candidates = candidates->next)
     {
       if (only_viable_p.is_true () && candidates->viable != 1)
@@ -8337,7 +8346,7 @@ conversion_null_warnings (tree totype, tree expr, tree 
fn, int argnum)
                          "passing NULL to non-pointer argument %P of %qD",
                          argnum, fn))
            inform (get_fndecl_argument_location (fn, argnum),
-                   "  declared here");
+                   "declared here");
        }
       else
        warning_at (loc, OPT_Wconversion_null,
@@ -8356,7 +8365,7 @@ conversion_null_warnings (tree totype, tree expr, tree 
fn, int argnum)
                          "converting %<false%> to pointer type for argument "
                          "%P of %qD", argnum, fn))
            inform (get_fndecl_argument_location (fn, argnum),
-                   "  declared here");
+                   "declared here");
        }
       else
        warning_at (loc, OPT_Wconversion_null,
@@ -8431,7 +8440,7 @@ maybe_inform_about_fndecl_for_bogus_argument_init (tree 
fn, int argnum,
       gcc_rich_location richloc (get_fndecl_argument_location (fn, argnum));
       richloc.set_highlight_color (highlight_color);
       inform (&richloc,
-             "  initializing argument %P of %qD", argnum, fn);
+             "initializing argument %P of %qD", argnum, fn);
     }
 }
 
diff --git a/gcc/cp/constraint.cc b/gcc/cp/constraint.cc
index 8a36b9c88c4d..c25ba7893552 100644
--- a/gcc/cp/constraint.cc
+++ b/gcc/cp/constraint.cc
@@ -1452,6 +1452,7 @@ tsubst_valid_expression_requirement (tree t, tree args, 
sat_info info)
       if (diagnosing_failed_constraint::replay_errors_p ())
        {
          inform (loc, "the required expression %qE is invalid, because", t);
+         auto_diagnostic_nesting_level sentinel;
          if (r == error_mark_node)
            tsubst_expr (t, args, info.complain, info.in_decl);
          else
@@ -2360,6 +2361,7 @@ satisfy_disjunction (tree t, tree args, sat_info info)
              "no operand of the disjunction is satisfied");
       if (diagnosing_failed_constraint::replay_errors_p ())
        {
+         auto_diagnostic_nesting_level sentinel;
          /* Replay the error in each branch of the disjunction.  */
          auto_vec<tree_pair> operands;
          collect_operands_of_disjunction (t, &operands);
@@ -2371,6 +2373,7 @@ satisfy_disjunction (tree t, tree args, sat_info info)
                                              disj_expr.get_start (),
                                              disj_expr.get_finish ());
              inform (loc, "the operand %qE is unsatisfied because", op);
+             auto_diagnostic_nesting_level sentinel;
              satisfy_constraint_r (norm_op, args, info);
            }
        }
@@ -3421,6 +3424,8 @@ diagnose_constraints (location_t loc, tree t, tree args)
   if (concepts_diagnostics_max_depth == 0)
     return;
 
+  auto_diagnostic_nesting_level sentinel;
+
   /* Replay satisfaction, but diagnose unsatisfaction.  */
   sat_info noisy (tf_warning_or_error, NULL_TREE, /*diag_unsat=*/true);
   constraint_satisfaction_value (t, args, noisy);
diff --git a/gcc/cp/error.cc b/gcc/cp/error.cc
index 23cfee4405ed..b917be2de955 100644
--- a/gcc/cp/error.cc
+++ b/gcc/cp/error.cc
@@ -3637,6 +3637,8 @@ static void
 cp_diagnostic_text_starter (diagnostic_text_output_format &text_output,
                            const diagnostic_info *diagnostic)
 {
+  pp_set_prefix (text_output.get_printer (),
+                text_output.build_indent_prefix (true));
   text_output.report_current_module (diagnostic_location (diagnostic));
   cp_print_error_function (text_output, diagnostic);
   maybe_print_instantiation_context (text_output);
@@ -3804,13 +3806,19 @@ print_instantiation_full_context 
(diagnostic_text_output_format &text_output)
 
   if (p)
     {
+      bool show_file
+       = ((!text_output.show_nesting_p ())
+          || text_output.show_locations_in_nesting_p ());
+      char *indent = text_output.build_indent_prefix (true);
       pp_verbatim (text_output.get_printer (),
                   p->list_p ()
-                  ? _("%s: In substitution of %qS:\n")
-                  : _("%s: In instantiation of %q#D:\n"),
-                  LOCATION_FILE (location),
+                  ? _("%s%s%sIn substitution of %qS:\n")
+                  : _("%s%s%sIn instantiation of %q#D:\n"),
+                  indent,
+                  show_file ? LOCATION_FILE (location) : "",
+                  show_file ? ": " : "",
                   p->get_node ());
-
+      free (indent);
       location = p->locus;
       p = p->next;
     }
@@ -3832,6 +3840,71 @@ print_location (diagnostic_text_output_format 
&text_output,
                 "locus", xloc.file, xloc.line);
 }
 
+/* A RAII class for use when emitting a line of contextual information
+   via pp_verbatim to a diagnostic_text_output_format to add before/after
+   behaviors to the pp_verbatim calls.
+
+   If the text output has show_nesting_p (), then the ctor prints
+   leading indentation and a bullet point, and the dtor prints
+   the location on a new line, and calls diagnostic_show_locus, both
+   with indentation (and no bullet point).
+
+   Otherwise (when the text output has !show_nesting_p), the ctor prints
+   the location as leading information on the same line, and the
+   dtor optionally calls diagnostic_show_locus.  */
+
+class auto_context_line
+{
+public:
+  auto_context_line (diagnostic_text_output_format &text_output,
+                    location_t loc,
+                    bool show_locus = false)
+  : m_text_output (text_output),
+    m_loc (loc),
+    m_show_locus (show_locus)
+  {
+    char *indent = m_text_output.build_indent_prefix (true);
+    pp_verbatim (m_text_output.get_printer (), indent);
+    free (indent);
+    if (!m_text_output.show_nesting_p ())
+      print_location (m_text_output, m_loc);
+  }
+  ~auto_context_line ()
+  {
+    pretty_printer *const pp = m_text_output.get_printer ();
+    if (m_text_output.show_nesting_p ())
+      {
+       if (m_text_output.show_locations_in_nesting_p ())
+         {
+           char *indent = m_text_output.build_indent_prefix (false);
+           pp_verbatim (pp, indent);
+           print_location (m_text_output, m_loc);
+           pp_newline (pp);
+
+           char *saved_prefix = pp_take_prefix (pp);
+           pp_set_prefix (pp, indent);
+           gcc_rich_location rich_loc (m_loc);
+           diagnostic_show_locus (&m_text_output.get_context (), &rich_loc,
+                                  DK_NOTE, pp);
+           pp_set_prefix (pp, saved_prefix);
+         }
+      }
+    else if (m_show_locus)
+      {
+       char *saved_prefix = pp_take_prefix (pp);
+       pp_set_prefix (pp, nullptr);
+       gcc_rich_location rich_loc (m_loc);
+       diagnostic_show_locus (&m_text_output.get_context (), &rich_loc,
+                              DK_NOTE, pp);
+       pp_set_prefix (pp, saved_prefix);
+      }
+  }
+private:
+  diagnostic_text_output_format &m_text_output;
+  location_t m_loc;
+  bool m_show_locus;
+};
+
 /* Helper function of print_instantiation_partial_context() that
    prints a single line of instantiation context.  */
 
@@ -3843,7 +3916,7 @@ print_instantiation_partial_context_line 
(diagnostic_text_output_format &text_ou
   if (loc == UNKNOWN_LOCATION)
     return;
 
-  print_location (text_output, loc);
+  auto_context_line sentinel (text_output, loc, true);
 
   pretty_printer *const pp = text_output.get_printer ();
 
@@ -3869,10 +3942,6 @@ print_instantiation_partial_context_line 
(diagnostic_text_output_format &text_ou
                   ? _("recursively required from here\n")
                   : _("required from here\n"));
     }
-  gcc_rich_location rich_loc (loc);
-  char *saved_prefix = pp_take_prefix (pp);
-  diagnostic_show_locus (&text_output.get_context (), &rich_loc, DK_NOTE, pp);
-  pp_set_prefix (pp, saved_prefix);
 }
 
 /* Same as print_instantiation_full_context but less verbose.  */
@@ -3919,7 +3988,7 @@ print_instantiation_partial_context 
(diagnostic_text_output_format &text_output,
        }
       if (t != NULL && skip > 0)
        {
-         print_location (text_output, loc);
+         auto_context_line sentinel (text_output, loc);
          pp_verbatim (text_output.get_printer (),
                       _("[ skipping %d instantiation contexts,"
                         " use -ftemplate-backtrace-limit=0 to disable ]\n"),
@@ -3971,7 +4040,7 @@ maybe_print_constexpr_context 
(diagnostic_text_output_format &text_output)
     {
       const char *s = expr_as_string (t, 0);
       pretty_printer *const pp = text_output.get_printer ();
-      print_location (text_output, EXPR_LOCATION (t));
+      auto_context_line sentinel (text_output, EXPR_LOCATION (t));
       pp_verbatim (pp,
                   _("in %<constexpr%> expansion of %qs"),
                   s);
@@ -3984,7 +4053,7 @@ static void
 print_constrained_decl_info (diagnostic_text_output_format &text_output,
                             tree decl)
 {
-  print_location (text_output, DECL_SOURCE_LOCATION (decl));
+  auto_context_line sentinel (text_output, DECL_SOURCE_LOCATION (decl));
   pretty_printer *const pp = text_output.get_printer ();
   pp_verbatim (pp, "required by the constraints of %q#D\n", decl);
 }
@@ -3997,7 +4066,7 @@ print_concept_check_info (diagnostic_text_output_format 
&text_output,
 
   tree tmpl = TREE_OPERAND (expr, 0);
 
-  print_location (text_output, DECL_SOURCE_LOCATION (tmpl));
+  auto_context_line sentinel (text_output, DECL_SOURCE_LOCATION (tmpl));
 
   cxx_pretty_printer *const pp
     = (cxx_pretty_printer *)text_output.get_printer ();
@@ -4021,7 +4090,7 @@ print_constraint_context_head 
(diagnostic_text_output_format &text_output,
   tree src = TREE_VALUE (cxt);
   if (!src)
     {
-      print_location (text_output, input_location);
+      auto_context_line sentinel (text_output, input_location);
       pretty_printer *const pp = text_output.get_printer ();
       pp_verbatim (pp, "required for constraint satisfaction\n");
       return NULL_TREE;
@@ -4049,7 +4118,7 @@ print_requires_expression_info 
(diagnostic_text_output_format &text_output,
   if (map == error_mark_node)
     return;
 
-  print_location (text_output, cp_expr_loc_or_input_loc (expr));
+  auto_context_line sentinel (text_output, cp_expr_loc_or_input_loc (expr));
   cxx_pretty_printer *const pp
     = static_cast <cxx_pretty_printer *> (text_output.get_printer ());
   pp_verbatim (pp, "in requirements ");
diff --git a/gcc/testsuite/g++.dg/concepts/nested-diagnostics-1-truncated.C 
b/gcc/testsuite/g++.dg/concepts/nested-diagnostics-1-truncated.C
new file mode 100644
index 000000000000..5965e25f2563
--- /dev/null
+++ b/gcc/testsuite/g++.dg/concepts/nested-diagnostics-1-truncated.C
@@ -0,0 +1,41 @@
+// { dg-do compile { target c++17 } }
+// { dg-options "-fconcepts" }
+// { dg-additional-options 
"-fdiagnostics-set-output=text:experimental-nesting=yes,experimental-nesting-show-locations=no"
 }
+
+struct dog {};
+struct cat {};
+
+void pet(dog);
+void pet(cat);
+
+template <class T>
+concept has_member_pet = requires(T t) { t.pet(); };
+
+template <class T>
+concept has_default_pet = T::is_pettable;
+
+template <class T>
+concept pettable = has_member_pet<T> or has_default_pet<T>;
+
+void pet(pettable auto t);
+
+struct lizard {};
+
+int main() {
+  pet(lizard{}); // { dg-error "no matching function for call to 
'pet\\\(lizard\\\)'" }
+}
+
+/* { dg-begin-multiline-output "" }
+  * note: candidate: 'template<class auto:1>  requires  pettable<auto:1> void 
pet(auto:1)'
+    * note: template argument deduction/substitution failed:
+      * note: constraints not satisfied
+        * In substitution of 'template<class auto:1>  requires  
pettable<auto:1> void pet(auto:1) [with auto:1 = lizard]':
+        * required from here
+        * required for the satisfaction of 'pettable<auto:1>' [with auto:1 = 
lizard]
+        * note: no operand of the disjunction is satisfied
+        * note: set '-fconcepts-diagnostics-depth=' to at least 2 for more 
detail
+  * note: candidate: 'void pet(dog)'
+    * note: no known conversion for argument 1 from 'lizard' to 'dog'
+  * note: candidate: 'void pet(cat)'
+    * note: no known conversion for argument 1 from 'lizard' to 'cat'
+   { dg-end-multiline-output "" } */
diff --git a/gcc/testsuite/g++.dg/concepts/nested-diagnostics-1.C 
b/gcc/testsuite/g++.dg/concepts/nested-diagnostics-1.C
new file mode 100644
index 000000000000..a4a2f4eb139f
--- /dev/null
+++ b/gcc/testsuite/g++.dg/concepts/nested-diagnostics-1.C
@@ -0,0 +1,51 @@
+// { dg-do compile { target c++17 } }
+// { dg-options "-fconcepts" }
+// { dg-additional-options 
"-fdiagnostics-set-output=text:experimental-nesting=yes,experimental-nesting-show-locations=no"
 }
+// { dg-additional-options "-fconcepts-diagnostics-depth=3" }
+
+struct dog {};
+struct cat {};
+
+void pet(dog);
+void pet(cat);
+
+template <class T>
+concept has_member_pet = requires(T t) { t.pet(); };
+
+template <class T>
+concept has_default_pet = T::is_pettable;
+
+template <class T>
+concept pettable = has_member_pet<T> or has_default_pet<T>;
+
+void pet(pettable auto t);
+
+struct lizard {};
+
+int main() {
+  pet(lizard{}); // { dg-error "no matching function for call to 
'pet\\\(lizard\\\)'" }
+}
+
+/* { dg-begin-multiline-output "" }
+  * note: candidate: 'template<class auto:1>  requires  pettable<auto:1> void 
pet(auto:1)'
+    * note: template argument deduction/substitution failed:
+      * note: constraints not satisfied
+        * In substitution of 'template<class auto:1>  requires  
pettable<auto:1> void pet(auto:1) [with auto:1 = lizard]':
+        * required from here
+        * required for the satisfaction of 'pettable<auto:1>' [with auto:1 = 
lizard]
+        * note: no operand of the disjunction is satisfied
+          * note: the operand 'has_member_pet<T>' is unsatisfied because
+            * required for the satisfaction of 'has_member_pet<T>' [with T = 
lizard]
+            * required for the satisfaction of 'pettable<auto:1>' [with auto:1 
= lizard]
+            * in requirements with 'T t' [with T = lizard]
+            * note: the required expression 't.pet()' is invalid, because
+              * error: 'struct lizard' has no member named 'pet'
+          * note: the operand 'has_default_pet<T>' is unsatisfied because
+            * required for the satisfaction of 'has_default_pet<T>' [with T = 
lizard]
+            * required for the satisfaction of 'pettable<auto:1>' [with auto:1 
= lizard]
+            * error: 'is_pettable' is not a member of 'lizard'
+  * note: candidate: 'void pet(dog)'
+    * note: no known conversion for argument 1 from 'lizard' to 'dog'
+  * note: candidate: 'void pet(cat)'
+    * note: no known conversion for argument 1 from 'lizard' to 'cat'
+   { dg-end-multiline-output "" } */
diff --git a/gcc/testsuite/g++.dg/concepts/nested-diagnostics-2.C 
b/gcc/testsuite/g++.dg/concepts/nested-diagnostics-2.C
new file mode 100644
index 000000000000..0c5cc0152979
--- /dev/null
+++ b/gcc/testsuite/g++.dg/concepts/nested-diagnostics-2.C
@@ -0,0 +1,37 @@
+// { dg-do compile { target c++17 } }
+// { dg-options "-fconcepts" }
+// { dg-additional-options 
"-fdiagnostics-set-output=text:experimental-nesting=yes,experimental-nesting-show-locations=no"
 }
+
+struct dog{};
+struct cat{};
+
+void pet(dog);
+void pet(cat);
+
+template <class T>
+concept pettable = requires(T t) { t.pet(); };
+
+template <pettable T>
+void pet(T);
+
+struct donkey {};
+
+int main() {
+  pet(donkey{}); // { dg-error "no matching function for call to 
'pet\\\(donkey\\\)'" }
+}
+
+/* { dg-begin-multiline-output "" }
+  * note: candidate: 'template<class T>  requires  pettable<T> void pet(T)'
+    * note: template argument deduction/substitution failed:
+      * note: constraints not satisfied
+        * In substitution of 'template<class T>  requires  pettable<T> void 
pet(T) [with T = donkey]':
+        * required from here
+        * required for the satisfaction of 'pettable<T>' [with T = donkey]
+        * in requirements with 'T t' [with T = donkey]
+        * note: the required expression 't.pet()' is invalid
+        * note: set '-fconcepts-diagnostics-depth=' to at least 2 for more 
detail
+  * note: candidate: 'void pet(dog)'
+    * note: no known conversion for argument 1 from 'donkey' to 'dog'
+  * note: candidate: 'void pet(cat)'
+    * note: no known conversion for argument 1 from 'donkey' to 'cat'
+   { dg-end-multiline-output "" } */
-- 
2.26.3

Reply via email to