https://gcc.gnu.org/bugzilla/show_bug.cgi?id=84920

Lewis Hyatt <lhyatt at gcc dot gnu.org> changed:

           What    |Removed                     |Added
----------------------------------------------------------------------------
                 CC|                            |lhyatt at gcc dot gnu.org

--- Comment #8 from Lewis Hyatt <lhyatt at gcc dot gnu.org> ---
Created attachment 53108
  --> https://gcc.gnu.org/bugzilla/attachment.cgi?id=53108&action=edit
One possible approach

Hello-

What do you all think of this approach?

The new options -foverload-candidates-reduce-level and
-foverload-candidates-stop-level would configure the amount of information that
is output regarding overload candidates. When the number of overloads exceeds
-foverload-candidates-reduce-level (default 3), then the source lines are no
longer printed for each candidate, cutting the output length in half. The
source lines seem rarely useful anyway, since the diagnostic's first line
already prints the overload being considered. When the number of overloads
exceeds -foverload-candidates-stop-level (default 7), none of the overloads is
printed, since this is presumably a case such as a heavily overloaded operator
where the large number of candidates is less likely to be helpful in fixing the
problem.

For a testcase like this common (for me) typo:

=============
#include <ostream>
void f(std::ostream& o, int a) {
    o << a < '\n';
}
=============

GCC then outputs:
=============
iowarn.cpp: In function ‘void f(std::ostream&, int)’:
iowarn.cpp:3:12: error: no match for ‘operator<’ (operand types are
‘std::basic_ostream<char>’ and ‘char’)
    3 |     o << a < '\n';
      |     ~~~~~~ ^ ~~~~
      |       |      |
      |       |      char
      |       std::basic_ostream<char>
iowarn.cpp:3:12: note: declining to output 15 candidates (limit 7); compile
with ‘-foverload-candidates-stop-level=15’ to see them
=============

which at least for this specific case feels like an improvement to me... It's
obvious what is wrong now.

If you follow the suggestion and add the flag to the compile, then you see:
=============
iowarn.cpp: In function ‘void f(std::ostream&, int)’:
iowarn.cpp:3:12: error: no match for ‘operator<’ (operand types are
‘std::basic_ostream<char>’ and ‘char’)
    3 |     o << a < '\n';
      |     ~~~~~~ ^ ~~~~
      |       |      |
      |       |      char
      |       std::basic_ostream<char>
iowarn.cpp:3:12: note: candidate: ‘operator<(int, int)’ (built-in)
iowarn.cpp:3:12: note:   no known conversion for argument 1 from
‘std::basic_ostream<char>’ to ‘int’
In file included from /home/lewis/gccdev/release/include/c++/13.0.0/string:46,
                 from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/locale_classes.h:40,
                 from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/ios_base.h:41,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/ios:42,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/ostream:38,
                 from iowarn.cpp:1:
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:451:5: note:
candidate: ‘template<class _Iterator> constexpr bool std::operator<(const
reverse_iterator<_Iterator>&, const reverse_iterator<_Iterator>&)’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:451:5: note: 
 template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::reverse_iterator<_Iterator>’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:496:5: note:
candidate: ‘template<class _IteratorL, class _IteratorR> constexpr bool
std::operator<(const reverse_iterator<_Iterator>&, const
reverse_iterator<_IteratorR>&)’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:496:5: note: 
 template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::reverse_iterator<_Iterator>’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:1683:5: note:
candidate: ‘template<class _IteratorL, class _IteratorR> constexpr bool
std::operator<(const move_iterator<_IteratorL>&, const
move_iterator<_IteratorR>&)’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:1683:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::move_iterator<_IteratorL>’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:1748:5: note:
candidate: ‘template<class _Iterator> constexpr bool std::operator<(const
move_iterator<_IteratorL>&, const move_iterator<_IteratorL>&)’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:1748:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::move_iterator<_IteratorL>’
In file included from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_algobase.h:64,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/string:49:
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_pair.h:665:5: note:
candidate: ‘template<class _T1, class _T2> constexpr bool std::operator<(const
pair<_T1, _T2>&, const pair<_T1, _T2>&)’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_pair.h:665:5: note:  
template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::pair<_T1, _T2>’
In file included from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:47,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/string:52:
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:594:5: note:
candidate: ‘template<class _CharT, class _Traits> constexpr bool
std::operator<(basic_string_view<_CharT, _Traits>, basic_string_view<_CharT,
_Traits>)’
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:594:5: note:  
template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from
‘std::basic_string_view<_CharT, _Traits>’
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:600:5: note:
candidate: ‘template<class _CharT, class _Traits> constexpr bool
std::operator<(basic_string_view<_CharT, _Traits>,
__type_identity_t<basic_string_view<_CharT, _Traits> >)’
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:600:5: note:  
template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from
‘std::basic_string_view<_CharT, _Traits>’
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:607:5: note:
candidate: ‘template<class _CharT, class _Traits> constexpr bool
std::operator<(__type_identity_t<basic_string_view<_CharT, _Traits> >,
basic_string_view<_CharT, _Traits>)’
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:607:5: note:  
template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   mismatched types ‘std::basic_string_view<_CharT,
_Traits>’ and ‘char’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3690:5: note:
candidate: ‘template<class _CharT, class _Traits, class _Alloc> bool
std::operator<(const __cxx11::basic_string<_CharT, _Traits, _Allocator>&, const
__cxx11::basic_string<_CharT, _Traits, _Allocator>&)’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3690:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::__cxx11::basic_string<_CharT, _Traits, _Allocator>’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3703:5: note:
candidate: ‘template<class _CharT, class _Traits, class _Alloc> bool
std::operator<(const __cxx11::basic_string<_CharT, _Traits, _Allocator>&, const
_CharT*)’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3703:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::__cxx11::basic_string<_CharT, _Traits, _Allocator>’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3715:5: note:
candidate: ‘template<class _CharT, class _Traits, class _Alloc> bool
std::operator<(const _CharT*, const __cxx11::basic_string<_CharT, _Traits,
_Allocator>&)’
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3715:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   mismatched types ‘const _CharT*’ and
‘std::basic_ostream<char>’
In file included from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/memory_resource.h:47,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/string:56:
/home/lewis/gccdev/release/include/c++/13.0.0/tuple:1538:5: note: candidate:
‘template<class ... _TElements, class ... _UElements> constexpr bool
std::operator<(const tuple<_UTypes ...>&, const tuple<_UTypes ...>&)’
/home/lewis/gccdev/release/include/c++/13.0.0/tuple:1538:5: note:   template
argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::tuple<_UTypes ...>’
In file included from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/ios_base.h:46:
/home/lewis/gccdev/release/include/c++/13.0.0/system_error:314:3: note:
candidate: ‘bool std::operator<(const error_code&, const error_code&)’
/home/lewis/gccdev/release/include/c++/13.0.0/system_error:314:31: note:   no
known conversion for argument 1 from ‘std::basic_ostream<char>’ to ‘const
std::error_code&’
/home/lewis/gccdev/release/include/c++/13.0.0/system_error:495:3: note:
candidate: ‘bool std::operator<(const error_condition&, const
error_condition&)’
/home/lewis/gccdev/release/include/c++/13.0.0/system_error:495:36: note:   no
known conversion for argument 1 from ‘std::basic_ostream<char>’ to ‘const
std::error_condition&’
=============

This is still half the length of the current output, because the source lines
are not printed. The current output is:
=============
iowarn.cpp: In function ‘void f(std::ostream&, int)’:
iowarn.cpp:3:12: error: no match for ‘operator<’ (operand types are
‘std::basic_ostream<char>’ and ‘char’)
    3 |     o << a < '\n';
      |     ~~~~~~ ^ ~~~~
      |       |      |
      |       |      char
      |       std::basic_ostream<char>
iowarn.cpp:3:12: note: candidate: ‘operator<(int, int)’ (built-in)
    3 |     o << a < '\n';
      |     ~~~~~~~^~~~~~
iowarn.cpp:3:12: note:   no known conversion for argument 1 from
‘std::basic_ostream<char>’ to ‘int’
In file included from /home/lewis/gccdev/release/include/c++/13.0.0/string:46,
                 from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/locale_classes.h:40,
                 from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/ios_base.h:41,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/ios:42,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/ostream:38,
                 from iowarn.cpp:1:
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:451:5: note:
candidate: ‘template<class _Iterator> constexpr bool std::operator<(const
reverse_iterator<_Iterator>&, const reverse_iterator<_Iterator>&)’
  451 |     operator<(const reverse_iterator<_Iterator>& __x,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:451:5: note: 
 template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::reverse_iterator<_Iterator>’
    3 |     o << a < '\n';
      |              ^~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:496:5: note:
candidate: ‘template<class _IteratorL, class _IteratorR> constexpr bool
std::operator<(const reverse_iterator<_Iterator>&, const
reverse_iterator<_IteratorR>&)’
  496 |     operator<(const reverse_iterator<_IteratorL>& __x,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:496:5: note: 
 template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::reverse_iterator<_Iterator>’
    3 |     o << a < '\n';
      |              ^~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:1683:5: note:
candidate: ‘template<class _IteratorL, class _IteratorR> constexpr bool
std::operator<(const move_iterator<_IteratorL>&, const
move_iterator<_IteratorR>&)’
 1683 |     operator<(const move_iterator<_IteratorL>& __x,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:1683:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::move_iterator<_IteratorL>’
    3 |     o << a < '\n';
      |              ^~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:1748:5: note:
candidate: ‘template<class _Iterator> constexpr bool std::operator<(const
move_iterator<_IteratorL>&, const move_iterator<_IteratorL>&)’
 1748 |     operator<(const move_iterator<_Iterator>& __x,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_iterator.h:1748:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::move_iterator<_IteratorL>’
    3 |     o << a < '\n';
      |              ^~~~
In file included from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_algobase.h:64,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/string:49:
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_pair.h:665:5: note:
candidate: ‘template<class _T1, class _T2> constexpr bool std::operator<(const
pair<_T1, _T2>&, const pair<_T1, _T2>&)’
  665 |     operator<(const pair<_T1, _T2>& __x, const pair<_T1, _T2>& __y)
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/stl_pair.h:665:5: note:  
template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::pair<_T1, _T2>’
    3 |     o << a < '\n';
      |              ^~~~
In file included from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:47,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/string:52:
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:594:5: note:
candidate: ‘template<class _CharT, class _Traits> constexpr bool
std::operator<(basic_string_view<_CharT, _Traits>, basic_string_view<_CharT,
_Traits>)’
  594 |     operator< (basic_string_view<_CharT, _Traits> __x,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:594:5: note:  
template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from
‘std::basic_string_view<_CharT, _Traits>’
    3 |     o << a < '\n';
      |              ^~~~
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:600:5: note:
candidate: ‘template<class _CharT, class _Traits> constexpr bool
std::operator<(basic_string_view<_CharT, _Traits>,
__type_identity_t<basic_string_view<_CharT, _Traits> >)’
  600 |     operator< (basic_string_view<_CharT, _Traits> __x,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:600:5: note:  
template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from
‘std::basic_string_view<_CharT, _Traits>’
    3 |     o << a < '\n';
      |              ^~~~
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:607:5: note:
candidate: ‘template<class _CharT, class _Traits> constexpr bool
std::operator<(__type_identity_t<basic_string_view<_CharT, _Traits> >,
basic_string_view<_CharT, _Traits>)’
  607 |     operator< (__type_identity_t<basic_string_view<_CharT, _Traits>>
__x,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/string_view:607:5: note:  
template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   mismatched types ‘std::basic_string_view<_CharT,
_Traits>’ and ‘char’
    3 |     o << a < '\n';
      |              ^~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3690:5: note:
candidate: ‘template<class _CharT, class _Traits, class _Alloc> bool
std::operator<(const __cxx11::basic_string<_CharT, _Traits, _Allocator>&, const
__cxx11::basic_string<_CharT, _Traits, _Allocator>&)’
 3690 |     operator<(const basic_string<_CharT, _Traits, _Alloc>& __lhs,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3690:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::__cxx11::basic_string<_CharT, _Traits, _Allocator>’
    3 |     o << a < '\n';
      |              ^~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3703:5: note:
candidate: ‘template<class _CharT, class _Traits, class _Alloc> bool
std::operator<(const __cxx11::basic_string<_CharT, _Traits, _Allocator>&, const
_CharT*)’
 3703 |     operator<(const basic_string<_CharT, _Traits, _Alloc>& __lhs,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3703:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::__cxx11::basic_string<_CharT, _Traits, _Allocator>’
    3 |     o << a < '\n';
      |              ^~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3715:5: note:
candidate: ‘template<class _CharT, class _Traits, class _Alloc> bool
std::operator<(const _CharT*, const __cxx11::basic_string<_CharT, _Traits,
_Allocator>&)’
 3715 |     operator<(const _CharT* __lhs,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/bits/basic_string.h:3715:5: note:
  template argument deduction/substitution failed:
iowarn.cpp:3:14: note:   mismatched types ‘const _CharT*’ and
‘std::basic_ostream<char>’
    3 |     o << a < '\n';
      |              ^~~~
In file included from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/memory_resource.h:47,
                 from /home/lewis/gccdev/release/include/c++/13.0.0/string:56:
/home/lewis/gccdev/release/include/c++/13.0.0/tuple:1538:5: note: candidate:
‘template<class ... _TElements, class ... _UElements> constexpr bool
std::operator<(const tuple<_UTypes ...>&, const tuple<_UTypes ...>&)’
 1538 |     operator<(const tuple<_TElements...>& __t,
      |     ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/tuple:1538:5: note:   template
argument deduction/substitution failed:
iowarn.cpp:3:14: note:   ‘std::basic_ostream<char>’ is not derived from ‘const
std::tuple<_UTypes ...>’
    3 |     o << a < '\n';
      |              ^~~~
In file included from
/home/lewis/gccdev/release/include/c++/13.0.0/bits/ios_base.h:46:
/home/lewis/gccdev/release/include/c++/13.0.0/system_error:314:3: note:
candidate: ‘bool std::operator<(const error_code&, const error_code&)’
  314 |   operator<(const error_code& __lhs, const error_code& __rhs) noexcept
      |   ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/system_error:314:31: note:   no
known conversion for argument 1 from ‘std::basic_ostream<char>’ to ‘const
std::error_code&’
  314 |   operator<(const error_code& __lhs, const error_code& __rhs) noexcept
      |             ~~~~~~~~~~~~~~~~~~^~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/system_error:495:3: note:
candidate: ‘bool std::operator<(const error_condition&, const
error_condition&)’
  495 |   operator<(const error_condition& __lhs,
      |   ^~~~~~~~
/home/lewis/gccdev/release/include/c++/13.0.0/system_error:495:36: note:   no
known conversion for argument 1 from ‘std::basic_ostream<char>’ to ‘const
std::error_condition&’
  495 |   operator<(const error_condition& __lhs,
      |             ~~~~~~~~~~~~~~~~~~~~~~~^~~~~
=============

Perhaps the default thresholds of 3 and 7 could be tuned better based on
experience, but I hope maybe it's at least a useful framework to set up? If so
then the patch is ready to go and I could send it to gcc-patches. Thanks for
taking a look!

-Lewis

Reply via email to