diff options
author | David Mitchell <davem@iabyn.com> | 2010-07-12 20:53:04 +0100 |
---|---|---|
committer | David Mitchell <davem@iabyn.com> | 2010-07-14 23:06:18 +0100 |
commit | 803f274831f937654d48f8cf0468521cbf8f5dff (patch) | |
tree | 297f701cf0a8ef3af29be3017402207f1fa62707 /t | |
parent | 96bafef935f82644670a19c8ca57886c240cd969 (diff) | |
download | perl-803f274831f937654d48f8cf0468521cbf8f5dff.tar.gz |
protect CvGV weakref with backref
Each CV usually has a pointer, CvGV(cv), back to the GV that corresponds
to the CV's name (or to *foo::__ANON__ for anon CVs). This pointer wasn't
reference counted, to avoid loops. This could leave it dangling if the GV
is deleted.
We fix this by:
For named subs, adding backref magic to the GV, so that when the GV is
freed, it can trigger processing the CV's CvGV field. This processing
consists of: if it looks like the freeing of the GV is about to trigger
freeing of the CV too, set it to NULL; otherwise make it point to
*foo::__ANON__ (and set CvAONON(cv)).
For anon subs, make CvGV a strong reference, i.e. increment the refcnt of
*foo::__ANON__. This doesn't cause a loop, since in this case the
__ANON__ glob doesn't point to the CV. This also avoids dangling pointers
if someone does an explicit 'delete $foo::{__ANON__}'.
Note that there was already some partial protection for CvGV with
commit f1c32fec87699aee2eeb638f44135f21217d2127. This worked by
anonymising any corresponding CV when freeing a stash or stash entry.
This had two drawbacks. First it didn't fix CVs that were anonmous or that
weren't currently pointed to by the GV (e.g. after local *foo), and
second, it caused *all* CVs to get anonymised during cleanup, even the
ones that would have been deleted shortly afterwards anyway. This commit
effectively removes that former commit, while reusing a bit of the
actual anonymising code.
Diffstat (limited to 't')
-rw-r--r-- | t/op/caller.t | 8 | ||||
-rw-r--r-- | t/op/stash.t | 122 |
2 files changed, 74 insertions, 56 deletions
diff --git a/t/op/caller.t b/t/op/caller.t index 67992f1af7..27a55a8312 100644 --- a/t/op/caller.t +++ b/t/op/caller.t @@ -31,8 +31,8 @@ ok( $c[4], "hasargs true with anon sub" ); sub foo { @c = caller(0) } my $fooref = delete $::{foo}; $fooref -> (); -is( $c[3], "(unknown)", "unknown subroutine name" ); -ok( $c[4], "hasargs true with unknown sub" ); +is( $c[3], "main::__ANON__", "deleted subroutine name" ); +ok( $c[4], "hasargs true with deleted sub" ); print "# Tests with caller(1)\n"; @@ -60,8 +60,8 @@ ok( $c[4], "hasargs true with anon sub" ); sub foo2 { f() } my $fooref2 = delete $::{foo2}; $fooref2 -> (); -is( $c[3], "(unknown)", "unknown subroutine name" ); -ok( $c[4], "hasargs true with unknown sub" ); +is( $c[3], "main::__ANON__", "deleted subroutine name" ); +ok( $c[4], "hasargs true with deleted sub" ); # See if caller() returns the correct warning mask diff --git a/t/op/stash.t b/t/op/stash.t index 676c26c8c2..81ca233b42 100644 --- a/t/op/stash.t +++ b/t/op/stash.t @@ -7,7 +7,7 @@ BEGIN { BEGIN { require "./test.pl"; } -plan( tests => 32 ); +plan( tests => 37 ); # Used to segfault (bug #15479) fresh_perl_like( @@ -110,56 +110,34 @@ SKIP: { is( eval { $gv->NAME }, "__ANON__", "...and an __ANON__ name"); is( eval { $gv->STASH->NAME }, "__ANON__", "...and an __ANON__ stash"); - TODO: { - local $TODO = "anon CVs not accounted for yet"; - - my @results = split "\n", runperl( - switches => [ "-MB", "-l" ], - prog => q{ - my $sub = do { - package four; - sub { 1 }; - }; - %four:: = (); - - my $gv = B::svref_2object($sub)->GV; - print $gv->isa(q/B::GV/) ? q/ok/ : q/not ok/; - - my $st = eval { $gv->STASH->NAME }; - print $st eq q/__ANON__/ ? q/ok/ : q/not ok/; - - my $sub = do { - package five; - sub { 1 }; - }; - undef %five::; - - $gv = B::svref_2object($sub)->GV; - print $gv->isa(q/B::GV/) ? q/ok/ : q/not ok/; - - $st = eval { $gv->STASH->NAME }; - print $st eq q/__ANON__/ ? q/ok/ : q/not ok/; - - print q/done/; - }, - ($^O eq 'VMS') ? (stderr => 1) : () - ); - - ok( @results == 5 && $results[4] eq "done", - "anon CVs in undefed stash don't segfault" ) - or todo_skip $TODO, 4; - - ok( $results[0] eq "ok", - "cleared stash leaves anon CV with valid GV"); - ok( $results[1] eq "ok", - "...and an __ANON__ stash"); - - ok( $results[2] eq "ok", - "undefed stash leaves anon CV with valid GV"); - ok( $results[3] eq "ok", - "...and an __ANON__ stash"); + my $sub = do { + package four; + sub { 1 }; + }; + %four:: = (); + + my $gv = B::svref_2object($sub)->GV; + ok($gv->isa(q/B::GV/), "cleared stash leaves anon CV with valid GV"); + + my $st = eval { $gv->STASH->NAME }; + { local $TODO = 'STASHES not anonymized'; + is($st, q/__ANON__/, "...and an __ANON__ stash"); + } + + my $sub = do { + package five; + sub { 1 }; + }; + undef %five::; + + $gv = B::svref_2object($sub)->GV; + ok($gv->isa(q/B::GV/), "undefed stash leaves anon CV with valid GV"); + + $st = eval { $gv->STASH->NAME }; + { local $TODO = 'STASHES not anonymized'; + is($st, q/__ANON__/, "...and an __ANON__ stash"); } - + # [perl #58530] fresh_perl_is( 'sub foo { 1 }; use overload q/""/ => \&foo;' . @@ -169,7 +147,7 @@ SKIP: { "no segfault with overload/deleted stash entry [#58530]", ); - # CvSTASH should be null on a nmed sub if the stash has been deleted + # CvSTASH should be null on a named sub if the stash has been deleted { package FOO; sub foo {} @@ -177,8 +155,48 @@ SKIP: { package main; delete $::{'FOO::'}; my $cv = B::svref_2object($rfoo); - # XXX is there a better way of testing for NULL ? + # (is there a better way of testing for NULL ?) my $stash = $cv->STASH; like($stash, qr/B::SPECIAL/, "NULL CvSTASH on named sub"); } + + # on glob reassignment, orphaned CV should have anon CvGV + + { + my $r; + eval q[ + package FOO2; + sub f{}; + $r = \&f; + *f = sub {}; + ]; + delete $FOO2::{f}; + my $cv = B::svref_2object($r); + my $gv = $cv->GV; + ok($gv->isa(q/B::GV/), "orphaned CV has valid GV"); + is($gv->NAME, '__ANON__', "orphaned CV has anon GV"); + } + + # deleting __ANON__ glob shouldn't break things + + { + package FOO3; + sub named {}; + my $anon = sub {}; + my $named = eval q[\&named]; + package main; + delete $FOO3::{named}; # make named anonymous + + delete $FOO3::{__ANON__}; # whoops! + my ($cv,$gv); + $cv = B::svref_2object($named); + $gv = $cv->GV; + ok($gv->isa(q/B::GV/), "ex-named CV has valid GV"); + is($gv->NAME, '__ANON__', "ex-named CV has anon GV"); + + $cv = B::svref_2object($anon); + $gv = $cv->GV; + ok($gv->isa(q/B::GV/), "anon CV has valid GV"); + is($gv->NAME, '__ANON__', "anon CV has anon GV"); + } } |