From patchwork Fri Nov 30 21:05:04 2012 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit Subject: [C++] PR 54170 Date: Fri, 30 Nov 2012 11:05:04 -0000 From: Paolo Carlini X-Patchwork-Id: 203048 Message-Id: <50B91F80.3080100@oracle.com> To: "gcc-patches@gcc.gnu.org" Cc: Jason Merrill Hi, this wrong code PR is some sort of continuation of PR 52988, where we discard side-effects of expression of nullptr_t. Here too, in cp_convert_to_pointer and in build_ptrmemfunc, we don't check for side-effects and we replace the expression with a plain int_cst or nullptr_node. The testcase in the PR exercises only plain pointers but we have also to handle correctly pointers to members, thus I extended it and the patch became slightly more complex: for the case of pointer to data member in particular I have to call by hand build2 and build a COMPOUND_EXPR to host the side-effects (similarly to what I found used in cp_build_addr_expr_1). Tested x86_64-linux. Thanks, Paolo. ///////////////////////////// /cp 2012-11-30 Paolo Carlini PR c++/54170 * cvt.c (cp_convert_to_pointer): Don't discard side-effects from expressions of nullptr_t. * typeck.c (build_ptrmemfunc): Likewise. /testsuite 2012-11-30 Paolo Carlini PR c++/54170 * g++.dg/cpp0x/lambda/lambda-nullptr.C: New. Index: cp/cvt.c =================================================================== --- cp/cvt.c (revision 193995) +++ cp/cvt.c (working copy) @@ -219,10 +219,15 @@ cp_convert_to_pointer (tree type, tree expr, tsubs { /* A NULL pointer-to-member is represented by -1, not by zero. */ - expr = build_int_cst_type (type, -1); + expr = (TREE_SIDE_EFFECTS (expr) + ? build2 (COMPOUND_EXPR, type, expr, + build_int_cst_type (type, -1)) + : build_int_cst_type (type, -1)); } else - expr = build_int_cst (type, 0); + expr = (TREE_SIDE_EFFECTS (expr) + ? build_nop (type, expr) + : build_int_cst (type, 0)); return expr; } Index: cp/typeck.c =================================================================== --- cp/typeck.c (revision 193995) +++ cp/typeck.c (working copy) @@ -7567,7 +7567,9 @@ build_ptrmemfunc (tree type, tree pfn, int force, /* Handle null pointer to member function conversions. */ if (null_ptr_cst_p (pfn)) { - pfn = build_c_cast (input_location, type, nullptr_node); + pfn = (TREE_SIDE_EFFECTS (pfn) + ? build_nop (type, pfn) + : build_c_cast (input_location, type, nullptr_node)); return build_ptrmemfunc1 (to_type, integer_zero_node, pfn); Index: testsuite/g++.dg/cpp0x/lambda/lambda-nullptr.C =================================================================== --- testsuite/g++.dg/cpp0x/lambda/lambda-nullptr.C (revision 0) +++ testsuite/g++.dg/cpp0x/lambda/lambda-nullptr.C (working copy) @@ -0,0 +1,47 @@ +// PR c++/54170 +// { dg-do run { target c++11 } } + +#include + +struct A; +typedef A* ptr; +typedef int (A::*pmf) (int); +typedef int (A::*pdm); + +int total; + +void add(int n) +{ + total += n; +} + +template +RType Call(Callable native_func, int arg) +{ + return native_func(arg); +} + +template +RType do_test(int delta) +{ + return Call([=](int delta) { add(delta); return nullptr; }, delta); +} + +template +void test() +{ + total = 0; + assert (!do_test(5)); + assert (total == 5); + assert (!do_test(20)); + assert (total == 25); + assert (!do_test(-256)); + assert (total == -231); +} + +int main() +{ + test(); + test(); + test(); +}