From 81ae598bbd9f3faa1f0d7e16acfba9e6c914f94c Mon Sep 17 00:00:00 2001 From: Mark Mitchell Date: Sat, 28 May 2005 02:21:30 +0000 Subject: [PATCH] re PR c++/21614 (wrong code when calling member function of undefined class) PR c++/21614 * typeck.c (get_member_function_from_ptrfunc): Do not attempt conversions to base classes of incomplete types. PR c++/21614 * g++.dg/expr/ptrmem6.C: New test. * g++.dg/expr/ptrmem6a.C: Likewise. From-SVN: r100291 --- gcc/cp/ChangeLog | 6 ++++++ gcc/cp/typeck.c | 23 ++++++++++++++++------- gcc/testsuite/ChangeLog | 6 ++++++ gcc/testsuite/g++.dg/expr/ptrmem6.C | 12 ++++++++++++ gcc/testsuite/g++.dg/expr/ptrmem6a.C | 9 +++++++++ 5 files changed, 49 insertions(+), 7 deletions(-) create mode 100644 gcc/testsuite/g++.dg/expr/ptrmem6.C create mode 100644 gcc/testsuite/g++.dg/expr/ptrmem6a.C diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog index 2a9b21681226..24dc95819b6f 100644 --- a/gcc/cp/ChangeLog +++ b/gcc/cp/ChangeLog @@ -1,3 +1,9 @@ +2005-05-27 Mark Mitchell + + PR c++/21614 + * typeck.c (get_member_function_from_ptrfunc): Do not attempt + conversions to base classes of incomplete types. + 2005-05-27 Ian Lance Taylor * semantics.c (add_stmt): Add C++ frontend specific version. diff --git a/gcc/cp/typeck.c b/gcc/cp/typeck.c index 464b8efd3fa2..73bb514f4992 100644 --- a/gcc/cp/typeck.c +++ b/gcc/cp/typeck.c @@ -2346,14 +2346,23 @@ get_member_function_from_ptrfunc (tree *instance_ptrptr, tree function) gcc_unreachable (); } - /* Convert down to the right base before using the instance. First - use the type... */ + /* Convert down to the right base before using the instance. A + special case is that in a pointer to member of class C, C may + be incomplete. In that case, the function will of course be + a member of C, and no conversion is required. In fact, + lookup_base will fail in that case, because incomplete + classes do not have BINFOs. */ basetype = TYPE_METHOD_BASETYPE (TREE_TYPE (fntype)); - basetype = lookup_base (TREE_TYPE (TREE_TYPE (instance_ptr)), - basetype, ba_check, NULL); - instance_ptr = build_base_path (PLUS_EXPR, instance_ptr, basetype, 1); - if (instance_ptr == error_mark_node) - return error_mark_node; + if (!same_type_ignoring_top_level_qualifiers_p + (basetype, TREE_TYPE (TREE_TYPE (instance_ptr)))) + { + basetype = lookup_base (TREE_TYPE (TREE_TYPE (instance_ptr)), + basetype, ba_check, NULL); + instance_ptr = build_base_path (PLUS_EXPR, instance_ptr, basetype, + 1); + if (instance_ptr == error_mark_node) + return error_mark_node; + } /* ...and then the delta in the PMF. */ instance_ptr = build2 (PLUS_EXPR, TREE_TYPE (instance_ptr), instance_ptr, delta); diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog index efeae615a679..a6d54b3a4fca 100644 --- a/gcc/testsuite/ChangeLog +++ b/gcc/testsuite/ChangeLog @@ -1,3 +1,9 @@ +2005-05-27 Mark Mitchell + + PR c++/21614 + * g++.dg/expr/ptrmem6.C: New test. + * g++.dg/expr/ptrmem6a.C: Likewise. + 2005-05-27 Kazu Hirata PR tree-optimization/21658 diff --git a/gcc/testsuite/g++.dg/expr/ptrmem6.C b/gcc/testsuite/g++.dg/expr/ptrmem6.C new file mode 100644 index 000000000000..0c75385fbd0f --- /dev/null +++ b/gcc/testsuite/g++.dg/expr/ptrmem6.C @@ -0,0 +1,12 @@ +// PR C++/21614 +// { dg-additional-sources "ptrmem6a.C" } +// { dg-do run } + +extern struct Z *p; +extern int (Z::*m) (); + +int main () { + if ((p->*m)() == 7) + return 0; + return 1; +} diff --git a/gcc/testsuite/g++.dg/expr/ptrmem6a.C b/gcc/testsuite/g++.dg/expr/ptrmem6a.C new file mode 100644 index 000000000000..8dad81c5889a --- /dev/null +++ b/gcc/testsuite/g++.dg/expr/ptrmem6a.C @@ -0,0 +1,9 @@ +struct Z { + int f(); +}; + +int Z::f() { return 7; } + +struct Z z; +int (Z::*m)() = &Z::f; +struct Z*p = &z;