This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
Fix blead segfault on Cygwin for t/op/stash.t
[perl5.git] / t / op / stash.t
1 #!./perl
2
3 BEGIN {
4     chdir 't' if -d 't';
5     @INC = qw(../lib);
6 }
7
8 BEGIN { require "./test.pl"; }
9
10 plan( tests => 31 );
11
12 # Used to segfault (bug #15479)
13 fresh_perl_is(
14     '%:: = ""',
15     'Odd number of elements in hash assignment at - line 1.',
16     { switches => [ '-w' ] },
17     'delete $::{STDERR} and print a warning',
18 );
19
20 # Used to segfault
21 fresh_perl_is(
22     'BEGIN { $::{"X::"} = 2 }',
23     '',
24     { switches => [ '-w' ] },
25     q(Insert a non-GV in a stash, under warnings 'once'),
26 );
27
28 ok( !defined %oedipa::maas::, q(stashes aren't defined if not used) );
29 ok( !defined %{"oedipa::maas::"}, q(- work with hard refs too) );
30
31 ok( defined %tyrone::slothrop::, q(stashes are defined if seen at compile time) );
32 ok( defined %{"tyrone::slothrop::"}, q(- work with hard refs too) );
33
34 ok( defined %bongo::shaftsbury::, q(stashes are defined if a var is seen at compile time) );
35 ok( defined %{"bongo::shaftsbury::"}, q(- work with hard refs too) );
36
37 package tyrone::slothrop;
38 $bongo::shaftsbury::scalar = 1;
39
40 package main;
41
42 # Used to warn
43 # Unbalanced string table refcount: (1) for "A::" during global destruction.
44 # for ithreads.
45 {
46     local $ENV{PERL_DESTRUCT_LEVEL} = 2;
47     fresh_perl_is(
48                   'package A; sub a { // }; %::=""',
49                   '',
50                   '',
51                   );
52 }
53
54 # now tests in eval
55
56 ok( !eval  { defined %achtfaden:: },   'works in eval{}' );
57 ok( !eval q{ defined %schoenmaker:: }, 'works in eval("")' );
58
59 # now tests with strictures
60
61 {
62     use strict;
63     ok( !defined %pig::, q(referencing a non-existent stash doesn't produce stricture errors) );
64     ok( !exists $pig::{bodine}, q(referencing a non-existent stash element doesn't produce stricture errors) );
65 }
66
67 SKIP: {
68     eval { require B; 1 } or skip "no B", 12;
69
70     *b = \&B::svref_2object;
71     my $CVf_ANON = B::CVf_ANON();
72
73     my $sub = do {
74         package one;
75         \&{"one"};
76     };
77     delete $one::{one};
78     my $gv = b($sub)->GV;
79
80     isa_ok( $gv, "B::GV", "deleted stash entry leaves CV with valid GV");
81     is( b($sub)->CvFLAGS & $CVf_ANON, $CVf_ANON, "...and CVf_ANON set");
82     is( eval { $gv->NAME }, "__ANON__", "...and an __ANON__ name");
83     is( eval { $gv->STASH->NAME }, "one", "...but leaves stash intact");
84
85     $sub = do {
86         package two;
87         \&{"two"};
88     };
89     %two:: = ();
90     $gv = b($sub)->GV;
91
92     isa_ok( $gv, "B::GV", "cleared stash leaves CV with valid GV");
93     is( b($sub)->CvFLAGS & $CVf_ANON, $CVf_ANON, "...and CVf_ANON set");
94     is( eval { $gv->NAME }, "__ANON__", "...and an __ANON__ name");
95     is( eval { $gv->STASH->NAME }, "__ANON__", "...and an __ANON__ stash");
96
97     $sub = do {
98         package three;
99         \&{"three"};
100     };
101     undef %three::;
102     $gv = b($sub)->GV;
103
104     isa_ok( $gv, "B::GV", "undefed stash leaves CV with valid GV");
105     is( b($sub)->CvFLAGS & $CVf_ANON, $CVf_ANON, "...and CVf_ANON set");
106     is( eval { $gv->NAME }, "__ANON__", "...and an __ANON__ name");
107     is( eval { $gv->STASH->NAME }, "__ANON__", "...and an __ANON__ stash");
108
109     TODO: {
110         local $TODO = "anon CVs not accounted for yet";
111
112         my @results = split "\n", runperl
113             switches    => [ "-MB", "-l" ],
114             prog        => q{
115                 my $sub = do {
116                     package four;
117                     sub { 1 };
118                 };
119                 %four:: = ();
120
121                 my $gv = B::svref_2object($sub)->GV;
122                 print $gv->isa(q/B::GV/) ? q/ok/ : q/not ok/;
123
124                 my $st = eval { $gv->STASH->NAME };
125                 print $st eq q/__ANON__/ ? q/ok/ : q/not ok/;
126
127                 my $sub = do {
128                     package five;
129                     sub { 1 };
130                 };
131                 undef %five::;
132
133                 $gv = B::svref_2object($sub)->GV;
134                 print $gv->isa(q/B::GV/) ? q/ok/ : q/not ok/;
135
136                 $st = eval { $gv->STASH->NAME };
137                 print $st eq q/__ANON__/ ? q/ok/ : q/not ok/;
138
139                 print q/done/;
140             };
141
142         ok( @results == 5 && $results[4] eq "done",
143             "anon CVs in undefed stash don't segfault" )
144             or todo_skip $TODO, 4;
145
146         ok( $results[0] eq "ok", 
147             "cleared stash leaves anon CV with valid GV");
148         ok( $results[1] eq "ok",
149             "...and an __ANON__ stash");
150             
151         ok( $results[2] eq "ok", 
152             "undefed stash leaves anon CV with valid GV");
153         ok( $results[3] eq "ok",
154             "...and an __ANON__ stash");
155     }
156     
157     # [perl #58530]
158     fresh_perl_is(
159         'sub foo { 1 }; use overload q/""/ => \&foo;' .
160             'delete $main::{foo}; bless []',
161         "",
162         {},
163         "no segfault with overload/deleted stash entry [#58530]",
164     );
165 }