aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJason Merrill <jason@redhat.com>2023-06-06 15:31:23 -0400
committerJason Merrill <jason@redhat.com>2023-06-06 21:30:31 -0400
commit08cea4e56a094ff9cc7c65fdc9ce8c4d7aff64be (patch)
tree02ed204b5c66f3d0d84d29756d18cdc95990c47b
parent0fa9495553e0e0f4ceb764880b5bdd8ade197382 (diff)
downloadgcc-08cea4e56a094ff9cc7c65fdc9ce8c4d7aff64be.zip
gcc-08cea4e56a094ff9cc7c65fdc9ce8c4d7aff64be.tar.gz
gcc-08cea4e56a094ff9cc7c65fdc9ce8c4d7aff64be.tar.bz2
c++: fix throwing cleanup with label
While looking at PR92407 I noticed that the expectations of maybe_splice_retval_cleanup weren't being met; an sk_cleanup level was confusing its attempt to recognize the outer block of the function. And even if I fixed the detection, it failed to actually wrap the body of the function because the STATEMENT_LIST it got only had the label, not anything after it. So I moved the call after poplevel does pop_stmt_list on all the sk_cleanup levels. PR c++/33799 gcc/cp/ChangeLog: * except.cc (maybe_splice_retval_cleanup): Change recognition of function body and try scopes. * semantics.cc (do_poplevel): Call it after poplevel. (at_try_scope): New. * cp-tree.h (maybe_splice_retval_cleanup): Adjust. gcc/testsuite/ChangeLog: * g++.dg/eh/return1.C: Add label cases.
-rw-r--r--gcc/cp/cp-tree.h2
-rw-r--r--gcc/cp/except.cc25
-rw-r--r--gcc/cp/semantics.cc16
-rw-r--r--gcc/testsuite/g++.dg/eh/return1.C21
4 files changed, 49 insertions, 15 deletions
diff --git a/gcc/cp/cp-tree.h b/gcc/cp/cp-tree.h
index 101da35..834fdd1 100644
--- a/gcc/cp/cp-tree.h
+++ b/gcc/cp/cp-tree.h
@@ -7047,7 +7047,7 @@ extern tree begin_eh_spec_block (void);
extern void finish_eh_spec_block (tree, tree);
extern tree build_eh_type_type (tree);
extern tree cp_protect_cleanup_actions (void);
-extern void maybe_splice_retval_cleanup (tree);
+extern void maybe_splice_retval_cleanup (tree, bool);
extern tree maybe_set_retval_sentinel (void);
extern tree template_parms_to_args (tree);
diff --git a/gcc/cp/except.cc b/gcc/cp/except.cc
index b04eb00..28106da 100644
--- a/gcc/cp/except.cc
+++ b/gcc/cp/except.cc
@@ -1312,21 +1312,20 @@ maybe_set_retval_sentinel ()
on throw. */
void
-maybe_splice_retval_cleanup (tree compound_stmt)
+maybe_splice_retval_cleanup (tree compound_stmt, bool is_try)
{
- /* If we need a cleanup for the return value, add it in at the same level as
+ if (!current_function_decl || !cfun
+ || DECL_CONSTRUCTOR_P (current_function_decl)
+ || DECL_DESTRUCTOR_P (current_function_decl)
+ || !current_retval_sentinel)
+ return;
+
+ /* if we need a cleanup for the return value, add it in at the same level as
pushdecl_outermost_localscope. And also in try blocks. */
- const bool function_body
- = (current_binding_level->level_chain
- && current_binding_level->level_chain->kind == sk_function_parms
- /* When we're processing a default argument, c_f_d may not have been
- set. */
- && current_function_decl);
-
- if ((function_body || current_binding_level->kind == sk_try)
- && !DECL_CONSTRUCTOR_P (current_function_decl)
- && !DECL_DESTRUCTOR_P (current_function_decl)
- && current_retval_sentinel)
+ cp_binding_level *b = current_binding_level;
+ const bool function_body = b->kind == sk_function_parms;
+
+ if (function_body || is_try)
{
location_t loc = DECL_SOURCE_LOCATION (current_function_decl);
tree_stmt_iterator iter = tsi_start (compound_stmt);
diff --git a/gcc/cp/semantics.cc b/gcc/cp/semantics.cc
index c94ea09..a13c16f 100644
--- a/gcc/cp/semantics.cc
+++ b/gcc/cp/semantics.cc
@@ -628,6 +628,17 @@ set_cleanup_locs (tree stmts, location_t loc)
set_cleanup_locs (stmt, loc);
}
+/* True iff the innermost block scope is a try block. */
+
+static bool
+at_try_scope ()
+{
+ cp_binding_level *b = current_binding_level;
+ while (b && b->kind == sk_cleanup)
+ b = b->level_chain;
+ return b && b->kind == sk_try;
+}
+
/* Finish a scope. */
tree
@@ -635,11 +646,14 @@ do_poplevel (tree stmt_list)
{
tree block = NULL;
- maybe_splice_retval_cleanup (stmt_list);
+ bool was_try = at_try_scope ();
if (stmts_are_full_exprs_p ())
block = poplevel (kept_level_p (), 1, 0);
+ /* This needs to come after poplevel merges sk_cleanup statement_lists. */
+ maybe_splice_retval_cleanup (stmt_list, was_try);
+
stmt_list = pop_stmt_list (stmt_list);
/* input_location is the last token of the scope, usually a }. */
diff --git a/gcc/testsuite/g++.dg/eh/return1.C b/gcc/testsuite/g++.dg/eh/return1.C
index ac22254..e22d674 100644
--- a/gcc/testsuite/g++.dg/eh/return1.C
+++ b/gcc/testsuite/g++.dg/eh/return1.C
@@ -33,6 +33,13 @@ X f()
return X(false);
}
+X f2()
+{
+ foo:
+ X x(true);
+ return X(false);
+}
+
X g()
{
return X(true),X(false);
@@ -54,6 +61,16 @@ X i()
return X(false);
}
+X i2()
+{
+ try {
+ foo:
+ X x(true);
+ return X(false);
+ } catch(...) {}
+ return X(false);
+}
+
X j()
{
try {
@@ -84,6 +101,8 @@ int main()
try { f(); }
catch (...) {}
+ try { f2(); } catch (...) {}
+
try { g(); }
catch (...) {}
@@ -93,6 +112,8 @@ int main()
try { i(); }
catch (...) {}
+ try { i2(); } catch (...) {}
+
try { j(); } catch (...) {}
try { k<X>(); } catch (...) {}