diff options
author | Father Chrysostomos <sprout@cpan.org> | 2012-03-27 22:38:01 -0700 |
---|---|---|
committer | Father Chrysostomos <sprout@cpan.org> | 2012-03-27 23:19:52 -0700 |
commit | 63429d50b5b6c112d030f925ddce05a84d042758 (patch) | |
tree | af2c8d38a574b302144cedd1d4b2c4bcb0315e14 /t/op | |
parent | d03ce4f5412fd49bf3d119e68c5e987e3221dacf (diff) | |
download | perl-63429d50b5b6c112d030f925ddce05a84d042758.tar.gz |
[perl #111462] Don’t leak eval "" op tree when croaking
This patch only fixes the problem for croaks that occur in the peep-
hole optimiser or in Perl_finalize_optree.
It does this by doing SAVEFREEOP first and then restoring the
savestack index to its previous value afterwards (to void the effect
of SAVEFREEOP).
A more correct fix might be to do op_free in die_unwind before
POPEVAL, but I would have to do a lot more digging through the code
to tell whether that is safe. I don’t feel comfortable with doing
that for 5.16.
This leak causes this warning on non-threaded debugging builds:
$ PERL_DESTRUCT_LEVEL=1 ./perl -Ilib -e 'BEGIN { $^H{foo} = bar } our %FIELDS; my main $x; eval q[$x->{foo}]'
Unbalanced string table refcount: (1) for "foo" during global destruction.
This problem does not affect the main program, because perl_destruct
frees PL_main_root. It does not affect subroutines, as the op tree is
attached to the CV first, so freeing the CV frees the op tree.
Diffstat (limited to 't/op')
-rw-r--r-- | t/op/eval.t | 15 |
1 files changed, 14 insertions, 1 deletions
diff --git a/t/op/eval.t b/t/op/eval.t index 5cd7f4c125..e9a6996db8 100644 --- a/t/op/eval.t +++ b/t/op/eval.t @@ -6,7 +6,7 @@ BEGIN { require './test.pl'; } -plan(tests => 125); +plan(tests => 126); eval 'pass();'; @@ -596,3 +596,16 @@ EOP # SNMP::Trapinfo, when toke.c finds no syntax errors but perly.y fails. eval(q|""!=!~//|); pass("phew! dodged the assertion after a parsing (not lexing) error"); + +# [perl #111462] +{ + local $ENV{PERL_DESTRUCT_LEVEL} = 1; + unlike + runperl( + prog => 'BEGIN { $^H{foo} = bar }' + .'our %FIELDS; my main $x; eval q[$x->{foo}]', + stderr => 1, + ), + qr/Unbalanced string table/, + 'Errors in finalize_optree do not leak string eval op tree'; +} |