2019-08-30 Jakub Jelinek Backported from mainline 2019-01-21 Jakub Jelinek PR c++/88949 * optimize.c (cxx_copy_decl): New function. (clone_body): Use it instead of copy_decl_no_change. * g++.dg/gomp/pr88949.C: New test. --- gcc/cp/optimize.c (revision 268624) +++ gcc/cp/optimize.c (revision 268625) @@ -61,6 +61,25 @@ update_cloned_parm (tree parm, tree clon DECL_GIMPLE_REG_P (cloned_parm) = DECL_GIMPLE_REG_P (parm); } +/* Like copy_decl_no_change, but handle DECL_OMP_PRIVATIZED_MEMBER + properly. */ + +static tree +cxx_copy_decl (tree decl, copy_body_data *id) +{ + tree copy = copy_decl_no_change (decl, id); + if (VAR_P (decl) + && DECL_HAS_VALUE_EXPR_P (decl) + && DECL_ARTIFICIAL (decl) + && DECL_LANG_SPECIFIC (decl) + && DECL_OMP_PRIVATIZED_MEMBER (decl)) + { + tree expr = DECL_VALUE_EXPR (copy); + walk_tree (&expr, copy_tree_body_r, id, NULL); + SET_DECL_VALUE_EXPR (copy, expr); + } + return copy; +} /* FN is a function in High GIMPLE form that has a complete body and no CFG. CLONE is a function whose body is to be set to a copy of FN, @@ -80,7 +99,7 @@ clone_body (tree clone, tree fn, void *a id.src_cfun = DECL_STRUCT_FUNCTION (fn); id.decl_map = static_cast *> (arg_map); - id.copy_decl = copy_decl_no_change; + id.copy_decl = cxx_copy_decl; id.transform_call_graph_edges = CB_CGE_DUPLICATE; id.transform_new_cfg = true; id.transform_return_to_modify = false; --- gcc/testsuite/g++.dg/gomp/pr88949.C (nonexistent) +++ gcc/testsuite/g++.dg/gomp/pr88949.C (revision 268625) @@ -0,0 +1,23 @@ +// PR c++/88949 +// { dg-do compile } + +struct A { + int a; + A (int x) : a (x) { +#pragma omp parallel firstprivate (a) + --a; + } + void foo () { +#pragma omp parallel firstprivate (a) + --a; + } +}; + +int c; + +int +main () +{ + A d(c); + d.foo (); +}