Hello, I've fixed the location issue by generating new STATEMENT_LIST_END tree for each compound statement and using it to store the location of closing brace.
I do understand that it's rather an ugly workaround than a solution, but still it resolves the issue for us and doesn't break GCC testsuite. So I'll send it here, maybe someone else needs the fix. Best Regards, Vyacheslav Barinov Eric Botcazou <ebotca...@adacore.com> writes: >> But then, after lowering on the eh pass (gcc/tree-eh.c:1161), the "finally" >> location is set up to the `gimple_location (tf->try_finally_expr)' which >> actually expands to "testcase.cxx:4", a place where the "try" block starts. >> >> The dump testcase.cxx.009t.ehopt shows no location info for destructor: >> std::unique_ptr<char>::~unique_ptr (&lang); >> >> And next testcase.cxx.010t.eh dump shows equal location for constructor and >> destructor: >> [testcase.cxx:4:38] std::unique_ptr<char>::unique_ptr ([testcase.cxx:4:38] >> &lang, D.42272); [testcase.cxx:4:38] std::unique_ptr<char>::~unique_ptr >> (&lang); >> >> I performed several experiments trying to get information about gimple_block >> around the try_finally_expr statement right in `lower_try_finally_onedest', >> but unsuccessfully. And also, as far as I understand, this issue should be >> fixed in all the lowering functions, not only this one. >> >> Where should I dig in order to fix the issue? > > We have been using this patchlet locally for some time: > > Index: tree-eh.c > =================================================================== > --- tree-eh.c (revision 248944) > +++ tree-eh.c (working copy) > @@ -1413,11 +1413,12 @@ lower_try_finally_switch (struct leh_sta > x = gimple_build_assign (finally_tmp, > build_int_cst (integer_type_node, > fallthru_index)); > + gimple_set_location (x, finally_loc); > gimple_seq_add_stmt (&tf->top_p_seq, x); > > tmp = build_int_cst (integer_type_node, fallthru_index); > last_case = build_case_label (tmp, NULL, > - create_artificial_label (tf_loc)); > + create_artificial_label (finally_loc)); > case_label_vec.quick_push (last_case); > last_case_index++; > > @@ -1426,7 +1427,7 @@ lower_try_finally_switch (struct leh_sta > > tmp = lower_try_finally_fallthru_label (tf); > x = gimple_build_goto (tmp); > - gimple_set_location (x, tf_loc); > + gimple_set_location (x, finally_loc); > gimple_seq_add_stmt (&switch_body, x); > }
>From 2e3782efbf39bc26342131eb92b1bf45aeed9e8d Mon Sep 17 00:00:00 2001 From: Slava Barinov <v.bari...@samsung.com> Date: Thu, 6 Jul 2017 16:14:58 +0300 Subject: [PATCH] Fix cleanup location for try_finally_expr. gcc/ * tree.def: Add STATEMENT_LIST_END tree code. * tree.c: Add STATEMENT_LIST_END handling as TS_COMMON. * gimplify.c (gimplify_expr): Use STATEMENT_LIST_END location to provide right information for try_finally_expr. * tree-eh.c (lower_try_finally_onedest): Set finally location * c-family/c-semantics.c (pop_stmt_list): Support single-statement lists extraction with STATEMENT_LIST_END in the end. * fold-const.c (operand_equal_p): Add STATEMENT_LIST_END support. gcc/cp/ * parser.c (cp_parser_compound_statement): Use STATEMENT_LIST_END to keep the location of closing brace. * pt.c: Handle STATEMENT_LIST_END. * constraint.cc (check_function_concept): Handle concept definitions with STATEMENT_LIST_END. * error.c (dump_expr): Add STATEMENT_LIST_END support. gcc/testsuite/ * g++.dg/ext/statement-list-end.C: New. Signed-off-by: Slava Barinov <v.bari...@samsung.com> --- gcc/ChangeLog | 17 ++++++++++++++ gcc/c-family/c-semantics.c | 6 +++++ gcc/cp/ChangeLog | 7 ++++++ gcc/cp/constexpr.c | 7 ++++++ gcc/cp/constraint.cc | 9 ++++++++ gcc/cp/error.c | 1 + gcc/cp/parser.c | 15 ++++++++++--- gcc/cp/pt.c | 3 +++ gcc/fold-const.c | 4 ++++ gcc/gimple.c | 1 + gcc/gimplify.c | 26 ++++++++++++++++++++++ gcc/testsuite/ChangeLog | 4 ++++ .../c-c++-common/Wimplicit-fallthrough-6.c | 16 ++++++------- .../c-c++-common/Wimplicit-fallthrough-7.c | 4 ++-- gcc/testsuite/g++.dg/ext/statement-list-end.C | 11 +++++++++ gcc/testsuite/g++.dg/gcov/gcov-2.C | 4 ++-- gcc/testsuite/g++.dg/parse/error26.C | 4 ++-- gcc/testsuite/g++.dg/tm/inherit2.C | 4 ++-- gcc/testsuite/g++.dg/tm/unsafe1.C | 4 ++-- .../g++.dg/warn/Wimplicit-fallthrough-1.C | 4 ++-- gcc/tree-eh.c | 15 +++++++++---- gcc/tree.c | 2 ++ gcc/tree.def | 4 ++++ 23 files changed, 145 insertions(+), 27 deletions(-) create mode 100644 gcc/testsuite/g++.dg/ext/statement-list-end.C diff --git a/gcc/ChangeLog b/gcc/ChangeLog index 2acf140f407..45df6c6b5a8 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,20 @@ +2017-07-06 Vyacheslav Barinov <v.bari...@samsung.com> + + * tree.def: Add STATEMENT_LIST_END tree code. + * tree.c: Add STATEMENT_LIST_END handling as TS_COMMON. + * gimplify.c (gimplify_expr): Use STATEMENT_LIST_END location to + provide right information for try_finally_expr. + * tree-eh.c (lower_try_finally_onedest): Set finally location + accordingly to location of last statement in cleanup. + * gimple.c (DEFTREECODE): handle STATEMENT_LIST_END as appropriate + GIMPLE_SINGLE_RHS. + * cp/constraint.cc (check_function_concept): Handle concept + definitions with STATEMENT_LIST_END. + * cp/error.c (dump_expr): Add STATEMENT_LIST_END support. + * fold-const.c (operand_equal_p): Add STATEMENT_LIST_END support. + * c-family/c-semantics.c (pop_stmt_list): Support single-statement + lists extraction with STATEMENT_LIST_END in the end. + 2017-07-06 Christophe Lyon <christophe.l...@linaro.org> * doc/sourcebuild.texi (Test Directives, Variants of diff --git a/gcc/c-family/c-semantics.c b/gcc/c-family/c-semantics.c index 3ceb714e4b1..b5cecf35bb1 100644 --- a/gcc/c-family/c-semantics.c +++ b/gcc/c-family/c-semantics.c @@ -66,6 +66,12 @@ pop_stmt_list (tree t) if (TREE_SIDE_EFFECTS (t)) { tree_stmt_iterator i = tsi_start (t); + if (i.ptr->next && i.ptr->next->stmt && + TREE_CODE(i.ptr->next->stmt) == STATEMENT_LIST_END) + { + tree_stmt_iterator i = tsi_last (t); + tsi_delink (&i); + } /* If the statement list contained exactly one statement, then extract it immediately. */ diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog index 4dffee44f17..8a5c929bae3 100644 --- a/gcc/cp/ChangeLog +++ b/gcc/cp/ChangeLog @@ -1,3 +1,10 @@ +2017-07-06 Vyacheslav Barinov <v.bari...@samsung.com> + + * parser.c (cp_parser_compound_statement): Use STATEMENT_LIST_END + to keep the location of closing brace. + * pt.c: Handle STATEMENT_LIST_END. + * constexpr.c (cxx_eval_constant_expression): Likewise. + 2017-07-04 Jakub Jelinek <ja...@redhat.com> * parser.c (cp_parser_decomposition_declaration): Replace diff --git a/gcc/cp/constexpr.c b/gcc/cp/constexpr.c index 32180a74c3f..715d9c8514b 100644 --- a/gcc/cp/constexpr.c +++ b/gcc/cp/constexpr.c @@ -448,6 +448,7 @@ check_constexpr_ctor_body_1 (tree last, tree list) case USING_STMT: case STATIC_ASSERT: + case STATEMENT_LIST_END: return true; default: @@ -562,6 +563,9 @@ build_constexpr_constructor_member_initializers (tree type, tree body) body = BIND_EXPR_BODY (body); goto found; + case STATEMENT_LIST_END: + break; + default: gcc_unreachable (); } @@ -673,6 +677,7 @@ constexpr_fn_retval (tree body) return constexpr_fn_retval (BIND_EXPR_BODY (body)); case USING_STMT: + case STATEMENT_LIST_END: return NULL_TREE; default: @@ -4486,6 +4491,7 @@ cxx_eval_constant_expression (const constexpr_ctx *ctx, tree t, break; case EMPTY_CLASS_EXPR: + case STATEMENT_LIST_END: /* This is good enough for a function argument that might not get used, and they can't do anything with it, so just return it. */ return t; @@ -5777,6 +5783,7 @@ potential_constant_expression_1 (tree t, bool want_rval, bool strict, case TYPE_DECL: case TAG_DEFN: + case STATEMENT_LIST_END: /* We can see these in statement-expressions. */ return true; diff --git a/gcc/cp/constraint.cc b/gcc/cp/constraint.cc index 64a8ea926d2..1c1456162d9 100644 --- a/gcc/cp/constraint.cc +++ b/gcc/cp/constraint.cc @@ -42,6 +42,7 @@ along with GCC; see the file COPYING3. If not see #include "c-family/c-objc.h" #include "cp-objcp-common.h" #include "tree-inline.h" +#include "tree-iterator.h" #include "decl.h" #include "toplev.h" #include "type-utils.h" @@ -2499,6 +2500,14 @@ check_function_concept (tree fn) if (TREE_CODE (body) == CLEANUP_POINT_EXPR) body = TREE_OPERAND (body, 0); + // We need to cut the STATEMENT_LIST_END before the check. + if (TREE_CODE (body) == STATEMENT_LIST && + TREE_CODE (STATEMENT_LIST_TAIL (body)->stmt) == STATEMENT_LIST_END) + { + tree_stmt_iterator i = tsi_last (body); + tsi_delink(&i); + } + /* Check that the definition is written correctly. */ if (TREE_CODE (body) != RETURN_EXPR) { diff --git a/gcc/cp/error.c b/gcc/cp/error.c index 43b5cec0a95..2c89448f07d 100644 --- a/gcc/cp/error.c +++ b/gcc/cp/error.c @@ -2688,6 +2688,7 @@ dump_expr (cxx_pretty_printer *pp, tree t, int flags) case STMT_EXPR: case EXPR_STMT: case STATEMENT_LIST: + case STATEMENT_LIST_END: /* We don't yet have a way of dumping statements in a human-readable format. */ pp_string (pp, "({...})"); diff --git a/gcc/cp/parser.c b/gcc/cp/parser.c index dbe00527924..41658fd581d 100644 --- a/gcc/cp/parser.c +++ b/gcc/cp/parser.c @@ -11142,11 +11142,20 @@ cp_parser_compound_statement (cp_parser *parser, tree in_statement_expr, cp_parser_label_declaration (parser); /* Parse an (optional) statement-seq. */ cp_parser_statement_seq_opt (parser, in_statement_expr); + /* Consume the `}' and keep location, if there is a `}' and compound_stmt is + not empty. */ + cp_token *token = cp_parser_require (parser, CPP_CLOSE_BRACE, RT_CLOSE_BRACE); + if (token && + TREE_CODE(compound_stmt) == STATEMENT_LIST && + STATEMENT_LIST_HEAD (compound_stmt) && + STATEMENT_LIST_TAIL (compound_stmt)) + { + tree brace_stmt = build0(STATEMENT_LIST_END, void_type_node); + SET_EXPR_LOCATION(brace_stmt, token->location); + add_stmt(brace_stmt); + } /* Finish the compound-statement. */ finish_compound_stmt (compound_stmt); - /* Consume the `}'. */ - cp_parser_require (parser, CPP_CLOSE_BRACE, RT_CLOSE_BRACE); - return compound_stmt; } diff --git a/gcc/cp/pt.c b/gcc/cp/pt.c index bd02951cb85..cb2ffd30ba0 100644 --- a/gcc/cp/pt.c +++ b/gcc/cp/pt.c @@ -16516,6 +16516,9 @@ tsubst_expr (tree t, tree args, tsubst_flags_t complain, tree in_decl, RETURN (build2_loc (EXPR_LOCATION (t), ANNOTATE_EXPR, TREE_TYPE (tmp), tmp, RECUR (TREE_OPERAND (t, 1)))); + case STATEMENT_LIST_END: + RETURN (t); + default: gcc_assert (!STATEMENT_CODE_P (TREE_CODE (t))); diff --git a/gcc/fold-const.c b/gcc/fold-const.c index 1bcbbb58154..d573d147cfd 100644 --- a/gcc/fold-const.c +++ b/gcc/fold-const.c @@ -3207,6 +3207,10 @@ operand_equal_p (const_tree arg0, const_tree arg1, unsigned int flags) return OP_SAME (0); return 0; + case STATEMENT_LIST_END: + return 1; + /* All STATEMENT_LIST_END are equal */ + default: return 0; } diff --git a/gcc/gimple.c b/gcc/gimple.c index 488f8c82b82..a5a8b929698 100644 --- a/gcc/gimple.c +++ b/gcc/gimple.c @@ -2060,6 +2060,7 @@ get_gimple_rhs_num_ops (enum tree_code code) || (SYM) == ADDR_EXPR \ || (SYM) == WITH_SIZE_EXPR \ || (SYM) == SSA_NAME) ? GIMPLE_SINGLE_RHS \ + : ((SYM) == STATEMENT_LIST_END) ? GIMPLE_SINGLE_RHS \ : GIMPLE_INVALID_RHS), #define END_OF_BASE_TREE_CODES (unsigned char) GIMPLE_INVALID_RHS, diff --git a/gcc/gimplify.c b/gcc/gimplify.c index 641a8210dad..724520b06a9 100644 --- a/gcc/gimplify.c +++ b/gcc/gimplify.c @@ -1023,6 +1023,11 @@ voidify_wrapper_expr (tree wrapper, tree temp) case STATEMENT_LIST: { tree_stmt_iterator i = tsi_last (*p); + if (TREE_CODE(*tsi_stmt_ptr (i)) == STATEMENT_LIST_END) + { + tsi_delink(&i); + i = tsi_last (*p); + } TREE_SIDE_EFFECTS (*p) = 1; TREE_TYPE (*p) = void_type_node; p = tsi_end_p (i) ? NULL : tsi_stmt_ptr (i); @@ -11671,6 +11676,19 @@ gimplify_expr (tree *expr_p, gimple_seq *pre_p, gimple_seq *post_p, end of gimplify_expr. */ input_location = UNKNOWN_LOCATION; eval = cleanup = NULL; + location_t finally_loc = 0; + /* The cleanup location can be extracted from STATEMENT_LIST_END + location added especially for this purpose. */ + if (TREE_OPERAND (*expr_p, 0) && + TREE_CODE (TREE_OPERAND (*expr_p, 0)) == STATEMENT_LIST) + { + const tree_statement_list_node* last_node = + STATEMENT_LIST_TAIL(TREE_OPERAND (*expr_p, 0)); + if (last_node && + last_node->stmt && + TREE_CODE (last_node->stmt) == STATEMENT_LIST_END) + finally_loc = EXPR_LOCATION(last_node->stmt); + } gimplify_and_add (TREE_OPERAND (*expr_p, 0), &eval); gimplify_and_add (TREE_OPERAND (*expr_p, 1), &cleanup); /* Don't create bogus GIMPLE_TRY with empty cleanup. */ @@ -11691,6 +11709,10 @@ gimplify_expr (tree *expr_p, gimple_seq *pre_p, gimple_seq *post_p, if (TREE_CODE (*expr_p) == TRY_CATCH_EXPR) gimple_try_set_catch_is_cleanup (try_, TRY_CATCH_IS_CLEANUP (*expr_p)); + + gimple *last_in_seq = gimple_seq_last_stmt (cleanup); + gimple_set_location(last_in_seq, finally_loc); + gimplify_seq_add_stmt (pre_p, try_); ret = GS_ALL_DONE; break; @@ -11752,6 +11774,10 @@ gimplify_expr (tree *expr_p, gimple_seq *pre_p, gimple_seq *post_p, ret = gimplify_statement_list (expr_p, pre_p); break; + case STATEMENT_LIST_END: + ret = GS_ALL_DONE; + break; + case WITH_SIZE_EXPR: { gimplify_expr (&TREE_OPERAND (*expr_p, 0), pre_p, diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog index d629c2bd90d..7a8e8dcf1eb 100644 --- a/gcc/testsuite/ChangeLog +++ b/gcc/testsuite/ChangeLog @@ -1,3 +1,7 @@ +2017-07-06 Vyacheslav Barinov <v.bari...@samsung.com> + + * g++.dg/ext/statement-list-end.C: New. + 2017-07-06 Christophe Lyon <christophe.l...@linaro.org> * lib/target-supports-dg.exp (dg-require-stack-check): New. diff --git a/gcc/testsuite/c-c++-common/Wimplicit-fallthrough-6.c b/gcc/testsuite/c-c++-common/Wimplicit-fallthrough-6.c index 32d5febda83..9593f670709 100644 --- a/gcc/testsuite/c-c++-common/Wimplicit-fallthrough-6.c +++ b/gcc/testsuite/c-c++-common/Wimplicit-fallthrough-6.c @@ -121,8 +121,8 @@ L1: int j = 0; bar (j); if (j == 8) - return; /* { dg-warning "statement may fall through" "" { target c++ } } */ - } + return; + } /* { dg-warning "statement may fall through" "" { target c++ } } */ case 2: bar (99); } @@ -151,8 +151,8 @@ L1: if (j == 8) bar (1); else - return; /* { dg-warning "statement may fall through" "" { target c++ } } */ - } + return; + } /* { dg-warning "statement may fall through" "" { target c++ } } */ case 2: bar (99); } @@ -181,8 +181,8 @@ L1: if (j == 8) bar (1); else - bar (2); /* { dg-warning "statement may fall through" "" { target c++ } } */ - } + bar (2); + } /* { dg-warning "statement may fall through" "" { target c++ } } */ case 2: bar (99); } @@ -281,8 +281,8 @@ L1: case 1: { /* { dg-warning "statement may fall through" "" { target c } } */ int j = 9; - switch (j); /* { dg-warning "statement may fall through" "" { target c++ } } */ - } + switch (j); + } /* { dg-warning "statement may fall through" "" { target c++ } } */ case 2: bar (99); } diff --git a/gcc/testsuite/c-c++-common/Wimplicit-fallthrough-7.c b/gcc/testsuite/c-c++-common/Wimplicit-fallthrough-7.c index 24a573b4d19..1145e313352 100644 --- a/gcc/testsuite/c-c++-common/Wimplicit-fallthrough-7.c +++ b/gcc/testsuite/c-c++-common/Wimplicit-fallthrough-7.c @@ -22,8 +22,8 @@ f (int i) { case 1: { /* { dg-warning "statement may fall through" "" { target c } } */ - int a[i]; /* { dg-warning "statement may fall through" "" { target c++ } } */ - } + int a[i]; + } /* { dg-warning "statement may fall through" "" { target c++ } } */ case 2: bar (99); } diff --git a/gcc/testsuite/g++.dg/ext/statement-list-end.C b/gcc/testsuite/g++.dg/ext/statement-list-end.C new file mode 100644 index 00000000000..3381172492a --- /dev/null +++ b/gcc/testsuite/g++.dg/ext/statement-list-end.C @@ -0,0 +1,11 @@ +/* { dg-do compile } */ +/* { dg-options "-fdump-tree-gimple-lineno" } */ +/* { dg-require-effective-target c++11 } */ + +#include <memory> +void i () +{ + std::unique_ptr<char> c(new char); +} + +/* { dg-final { scan-tree-dump "9:1] std::unique_ptr<char>::~unique_ptr" "gimple" } } */ diff --git a/gcc/testsuite/g++.dg/gcov/gcov-2.C b/gcc/testsuite/g++.dg/gcov/gcov-2.C index 6d002f5d2cd..2b4cdd844e9 100644 --- a/gcc/testsuite/g++.dg/gcov/gcov-2.C +++ b/gcc/testsuite/g++.dg/gcov/gcov-2.C @@ -20,9 +20,9 @@ private: void foo() { - C c; /* count(2) */ + C c; /* count(1) */ c.seti (1); /* count(1) */ -} +} /* count(1) */ int main() { diff --git a/gcc/testsuite/g++.dg/parse/error26.C b/gcc/testsuite/g++.dg/parse/error26.C index 95f299163e9..9092bbf2960 100644 --- a/gcc/testsuite/g++.dg/parse/error26.C +++ b/gcc/testsuite/g++.dg/parse/error26.C @@ -4,11 +4,11 @@ void foo() { if (({int c[2];})) ; // { dg-error "7:ISO C.. forbids" "7" } - // { dg-error "17:could not convert" "17" { target *-*-* } .-1 } + // { dg-error "18:could not convert" "18" { target *-*-* } .-1 } } void bar() { if (({})); // { dg-error "7:ISO C.. forbids" "7" } - // { dg-error "11:could not convert" "11" { target *-*-* } .-1 } + // { dg-error "9:could not convert" "9" { target *-*-* } .-1 } } diff --git a/gcc/testsuite/g++.dg/tm/inherit2.C b/gcc/testsuite/g++.dg/tm/inherit2.C index 3b696a9ffb6..366f9b30e90 100644 --- a/gcc/testsuite/g++.dg/tm/inherit2.C +++ b/gcc/testsuite/g++.dg/tm/inherit2.C @@ -26,8 +26,8 @@ int main() B b; // ok D1 d1; // ok B& b1 = d1; - D2 x; // { dg-error "" "destructor of D2 is not transaction-safe" } + D2 x; b1.f(); // ok, calls D1::f() delete b2; // undefined behavior: calls unsafe destructor of D2 - } + } // { dg-error "" "destructor of D2 is not transaction-safe" } } diff --git a/gcc/testsuite/g++.dg/tm/unsafe1.C b/gcc/testsuite/g++.dg/tm/unsafe1.C index 91dd7b110ec..e86b1bd33c4 100644 --- a/gcc/testsuite/g++.dg/tm/unsafe1.C +++ b/gcc/testsuite/g++.dg/tm/unsafe1.C @@ -5,8 +5,8 @@ struct S { virtual ~S(); }; int f() transaction_safe { - S s; // { dg-error "unsafe" "invocation of unsafe destructor" } -} + S s; +} // { dg-error "unsafe" "invocation of unsafe destructor" } int g(int x) { // is transaction-safe if (x <= 0) diff --git a/gcc/testsuite/g++.dg/warn/Wimplicit-fallthrough-1.C b/gcc/testsuite/g++.dg/warn/Wimplicit-fallthrough-1.C index 053ed6885c5..b5ebd3daaad 100644 --- a/gcc/testsuite/g++.dg/warn/Wimplicit-fallthrough-1.C +++ b/gcc/testsuite/g++.dg/warn/Wimplicit-fallthrough-1.C @@ -25,8 +25,8 @@ fn2 () switch (0) { case 0: { - A b; // { dg-warning "statement may fall through" } - } + A b; + } // { dg-warning "statement may fall through" } default: a = 0; } diff --git a/gcc/tree-eh.c b/gcc/tree-eh.c index 79d02adbade..65a61746fcb 100644 --- a/gcc/tree-eh.c +++ b/gcc/tree-eh.c @@ -1135,10 +1135,16 @@ lower_try_finally_onedest (struct leh_state *state, struct leh_tf_state *tf) gimple_stmt_iterator gsi; tree finally_label; location_t loc = gimple_location (tf->try_finally_expr); + location_t finally_loc; finally = gimple_try_cleanup (tf->top_p); tf->top_p_seq = gimple_try_eval (tf->top_p); + /* The location of the finally is either the last stmt in the finally + block or the location of the TRY_FINALLY itself. */ + x = gimple_seq_last_stmt (finally); + finally_loc = x ? gimple_location (x) : loc; + /* Since there's only one destination, and the destination edge can only either be EH or non-EH, that implies that all of our incoming edges are of the same type. Therefore we can lower EH_ELSE immediately. */ @@ -1159,7 +1165,7 @@ lower_try_finally_onedest (struct leh_state *state, struct leh_tf_state *tf) if (LOCATION_LOCUS (gimple_location (stmt)) == UNKNOWN_LOCATION) { tree block = gimple_block (stmt); - gimple_set_location (stmt, gimple_location (tf->try_finally_expr)); + gimple_set_location (stmt, finally_loc); gimple_set_block (stmt, block); } } @@ -1182,7 +1188,7 @@ lower_try_finally_onedest (struct leh_state *state, struct leh_tf_state *tf) return; } - finally_label = create_artificial_label (loc); + finally_label = create_artificial_label (finally_loc); label_stmt = gimple_build_label (finally_label); gimple_seq_add_stmt (&tf->top_p_seq, label_stmt); @@ -1414,11 +1420,12 @@ lower_try_finally_switch (struct leh_state *state, struct leh_tf_state *tf) x = gimple_build_assign (finally_tmp, build_int_cst (integer_type_node, fallthru_index)); + gimple_set_location (x, finally_loc); gimple_seq_add_stmt (&tf->top_p_seq, x); tmp = build_int_cst (integer_type_node, fallthru_index); last_case = build_case_label (tmp, NULL, - create_artificial_label (tf_loc)); + create_artificial_label (finally_loc)); case_label_vec.quick_push (last_case); last_case_index++; @@ -1427,7 +1434,7 @@ lower_try_finally_switch (struct leh_state *state, struct leh_tf_state *tf) tmp = lower_try_finally_fallthru_label (tf); x = gimple_build_goto (tmp); - gimple_set_location (x, tf_loc); + gimple_set_location (x, finally_loc); gimple_seq_add_stmt (&switch_body, x); } diff --git a/gcc/tree.c b/gcc/tree.c index ca28afad0f2..f07758e155d 100644 --- a/gcc/tree.c +++ b/gcc/tree.c @@ -469,6 +469,7 @@ tree_node_structure_for_code (enum tree_code code) case SSA_NAME: return TS_SSA_NAME; case PLACEHOLDER_EXPR: return TS_COMMON; case STATEMENT_LIST: return TS_STATEMENT_LIST; + case STATEMENT_LIST_END: return TS_COMMON; case BLOCK: return TS_BLOCK; case CONSTRUCTOR: return TS_CONSTRUCTOR; case TREE_BINFO: return TS_BINFO; @@ -824,6 +825,7 @@ tree_code_size (enum tree_code code) case TREE_LIST: return sizeof (struct tree_list); case ERROR_MARK: + case STATEMENT_LIST_END: case PLACEHOLDER_EXPR: return sizeof (struct tree_common); case TREE_VEC: diff --git a/gcc/tree.def b/gcc/tree.def index 0ec805903a9..5bab6e4482f 100644 --- a/gcc/tree.def +++ b/gcc/tree.def @@ -991,6 +991,10 @@ DEFTREECODE (POLYNOMIAL_CHREC, "polynomial_chrec", tcc_expression, 3) Use the interface in tree-iterator.h to access this node. */ DEFTREECODE (STATEMENT_LIST, "statement_list", tcc_exceptional, 0) +/* Used to keep STATEMENT_LIST end location. + The only useful field is location, which points to the closing brace. */ +DEFTREECODE (STATEMENT_LIST_END, "statement_list_end", tcc_expression, 0) + /* Predicate assertion. Artificial expression generated by the optimizers to keep track of predicate values. This expression may only appear on the RHS of assignments. -- 2.13.2