From: Nathan Sidwell Date: Mon, 12 Mar 2001 15:43:52 +0000 (+0000) Subject: call.c (convert_like_real): Add extra semantics to INNER parameter. X-Git-Url: https://git.libre-soc.org/?a=commitdiff_plain;h=78fe06c24507d3f70c80477816689dafc2c2c354;p=gcc.git call.c (convert_like_real): Add extra semantics to INNER parameter. cp: * call.c (convert_like_real): Add extra semantics to INNER parameter. Don't convert to temporary if a user conversion gives us an lvalue that we're about to bind to a reference. Set INNER to indicate pending reference binding on recursive calls. testsuite: * g++.old-deja/g++.other/ref4.C: New test. From-SVN: r40415 --- diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog index 3f0fa192607..46871a9c227 100644 --- a/gcc/cp/ChangeLog +++ b/gcc/cp/ChangeLog @@ -1,3 +1,11 @@ +2001-03-12 Nathan Sidwell + + * call.c (convert_like_real): Add extra semantics to INNER + parameter. Don't convert to temporary if a user conversion + gives us an lvalue that we're about to bind to a reference. + Set INNER to indicate pending reference binding on recursive + calls. + 2001-03-10 Neil Booth * cp/lex.c: Delete duplicate pending_lang_change. diff --git a/gcc/cp/call.c b/gcc/cp/call.c index 69c58db2d43..13827c9d1c4 100644 --- a/gcc/cp/call.c +++ b/gcc/cp/call.c @@ -3676,7 +3676,8 @@ enforce_access (basetype_path, decl) /* Perform the conversions in CONVS on the expression EXPR. FN and ARGNUM are used for diagnostics. ARGNUM is zero based, -1 indicates the `this' argument of a method. INNER is non-zero when - being called to continue a conversion chain. */ + being called to continue a conversion chain. It is negative when a + reference binding will be applied, positive otherwise. */ static tree convert_like_real (convs, expr, fn, argnum, inner) @@ -3755,7 +3756,8 @@ convert_like_real (convs, expr, fn, argnum, inner) conversion, but is not considered during overload resolution. If the target is a class, that means call a ctor. */ - if (IS_AGGR_TYPE (totype)) + if (IS_AGGR_TYPE (totype) + && (inner >= 0 || !real_lvalue_p (expr))) { savew = warningcount, savee = errorcount; expr = build_new_method_call @@ -3804,7 +3806,8 @@ convert_like_real (convs, expr, fn, argnum, inner) break; }; - expr = convert_like_real (TREE_OPERAND (convs, 0), expr, fn, argnum, 1); + expr = convert_like_real (TREE_OPERAND (convs, 0), expr, fn, argnum, + TREE_CODE (convs) == REF_BIND ? -1 : 1); if (expr == error_mark_node) return error_mark_node; diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog index ab380c42005..c86f7502246 100644 --- a/gcc/testsuite/ChangeLog +++ b/gcc/testsuite/ChangeLog @@ -1,3 +1,7 @@ +2001-03-12 Nathan Sidwell + + * g++.old-deja/g++.other/ref4.C: New test. + 2001-03-11 Nicola Pero * objc/execute/va_method.m: Added. diff --git a/gcc/testsuite/g++.old-deja/g++.other/ref4.C b/gcc/testsuite/g++.old-deja/g++.other/ref4.C new file mode 100644 index 00000000000..c3535d0d3e0 --- /dev/null +++ b/gcc/testsuite/g++.old-deja/g++.other/ref4.C @@ -0,0 +1,28 @@ +// Build don't link: + +// Copyright (C) 2001 Free Software Foundation, Inc. +// Contributed by Nathan Sidwell 27 Feb 2001 + +// Bug 2117. A conversion op to reference type created a temporary, even +// when bound to another reference. + +struct Abstract +{ + virtual void Foo () = 0; +}; + +struct Proxy +{ + operator Abstract & (); + Abstract &Convert (); +}; + +void Baz (Abstract &); + +void Foo () +{ + Proxy proxy; + + Baz (proxy); + Baz (proxy.Convert ()); +}