diff --git a/gcc/cp/call.c b/gcc/cp/call.c index feb3004517f193f5b0b3cc03964d4b59263312c5..4eae7338c322947267c6c4b907b0a5929b5cd3d7 100644 --- a/gcc/cp/call.c +++ b/gcc/cp/call.c @@ -4072,8 +4072,20 @@ build_new_op (enum tree_code code, int flags, tree arg1, tree arg2, tree arg3, default: if ((flags & LOOKUP_COMPLAIN) && (complain & tf_error)) { - op_error (code, code2, arg1, arg2, arg3, "no match"); - print_z_candidates (candidates); + /* If one of the arguments of the operator represents + an invalid use of member function pointer, try to report + a meaningful error ... */ + if (invalid_nonstatic_memfn_p (arg1, tf_error) + || invalid_nonstatic_memfn_p (arg2, tf_error) + || invalid_nonstatic_memfn_p (arg3, tf_error)) + /* We displayed the error message. */; + else + { + /* ... Otherwise, report the more generic + "no matching operator found" error */ + op_error (code, code2, arg1, arg2, arg3, "no match"); + print_z_candidates (candidates); + } } result = error_mark_node; break; diff --git a/gcc/cp/cvt.c b/gcc/cp/cvt.c index 5032f1c039169488b6d4426554e00da34c769dfb..6d35245e823362aaff2085534734e49b30d3e1a8 100644 --- a/gcc/cp/cvt.c +++ b/gcc/cp/cvt.c @@ -760,8 +760,15 @@ ocp_convert (tree type, tree expr, int convtype, int flags) } if (flags & LOOKUP_COMPLAIN) - error ("conversion from %qT to non-scalar type %qT requested", - TREE_TYPE (expr), type); + { + /* If the conversion failed and expr was an invalid use of pointer to + member function, try to report a meaningful error. */ + if (invalid_nonstatic_memfn_p (expr, tf_warning_or_error)) + /* We displayed the error message. */; + else + error ("conversion from %qT to non-scalar type %qT requested", + TREE_TYPE (expr), type); + } return error_mark_node; } diff --git a/gcc/cp/pt.c b/gcc/cp/pt.c index a4a5c10af7d78c39a242bd0cea52ece5ac9c89be..4844333770d5a5db71b9fac1d4770801616eda0b 100644 --- a/gcc/cp/pt.c +++ b/gcc/cp/pt.c @@ -13581,6 +13581,13 @@ unify (tree tparms, tree targs, tree parm, tree arg, int strict) && !template_parameter_pack_p (parm)) return 1; + /* If the argument deduction results is a METHOD_TYPE, + then there is a problem. + METHOD_TYPE doesn't map to any real C++ type the result of + the deduction can not be of that type. */ + if (TREE_CODE (arg) == METHOD_TYPE) + return 1; + TREE_VEC_ELT (INNERMOST_TEMPLATE_ARGS (targs), idx) = arg; return 0; diff --git a/gcc/cp/typeck.c b/gcc/cp/typeck.c index 34a20b8af2a7b1b83125ab114134c2306d909901..d9222dcbd06130bdcbacf357fd653dc3c335f05d 100644 --- a/gcc/cp/typeck.c +++ b/gcc/cp/typeck.c @@ -1508,7 +1508,7 @@ cxx_sizeof_or_alignof_expr (tree e, enum tree_code op, bool complain) bool invalid_nonstatic_memfn_p (const_tree expr, tsubst_flags_t complain) { - if (DECL_NONSTATIC_MEMBER_FUNCTION_P (expr)) + if (expr && DECL_NONSTATIC_MEMBER_FUNCTION_P (expr)) { if (complain & tf_error) error ("invalid use of non-static member function"); diff --git a/gcc/testsuite/g++.dg/expr/bound-mem-fun.C b/gcc/testsuite/g++.dg/expr/bound-mem-fun.C new file mode 100644 index 0000000000000000000000000000000000000000..9e699b638e36311f988ba68a062764d0cdfcd458 --- /dev/null +++ b/gcc/testsuite/g++.dg/expr/bound-mem-fun.C @@ -0,0 +1,18 @@ +// Contributed by Dodji Seketeli <dodji@redhat.com> +// Origin PR c++/38228 +// { dg-do "compile" } + +struct A +{ + A (); + template<typename T> A(T); +}; + +struct B +{ + int foo(); +}; + +A a = B().*(&B::foo); // { dg-error "invalid use of non-static member function" } + +