SvPVutf8_nolen crashes [PATCH]
[p5sagit/p5-mst-13.2.git] / t / op / local.t
CommitLineData
a687059c 1#!./perl
2
d441d3db 3BEGIN {
4 chdir 't' if -d 't';
5 require './test.pl';
6}
d60c5a05 7plan tests => 85;
a687059c 8
9sub foo {
10 local($a, $b) = @_;
11 local($c, $d);
d441d3db 12 $c = "c 3";
13 $d = "d 4";
14 { local($a,$c) = ("a 9", "c 10"); ($x, $y) = ($a, $c); }
15 is($a, "a 1");
16 is($b, "b 2");
17 $c, $d;
a687059c 18}
19
d441d3db 20$a = "a 5";
21$b = "b 6";
22$c = "c 7";
23$d = "d 8";
a687059c 24
d441d3db 25my @res;
26@res = &foo("a 1","b 2");
27is($res[0], "c 3");
28is($res[1], "d 4");
a687059c 29
d441d3db 30is($a, "a 5");
31is($b, "b 6");
32is($c, "c 7");
33is($d, "d 8");
34is($x, "a 9");
35is($y, "c 10");
a687059c 36
37# same thing, only with arrays and associative arrays
38
39sub foo2 {
40 local($a, @b) = @_;
41 local(@c, %d);
d441d3db 42 @c = "c 3";
43 $d{''} = "d 4";
44 { local($a,@c) = ("a 19", "c 20"); ($x, $y) = ($a, @c); }
45 is($a, "a 1");
46 is("@b", "b 2");
47 $c[0], $d{''};
a687059c 48}
49
d441d3db 50$a = "a 5";
51@b = "b 6";
52@c = "c 7";
53$d{''} = "d 8";
54
55@res = &foo2("a 1","b 2");
56is($res[0], "c 3");
57is($res[1], "d 4");
a687059c 58
d441d3db 59is($a, "a 5");
60is("@b", "b 6");
61is($c[0], "c 7");
62is($d{''}, "d 8");
63is($x, "a 19");
64is($y, "c 20");
a687059c 65
706a304b 66
67eval 'local($$e)';
d441d3db 68like($@, qr/Can't localize through a reference/);
706a304b 69
82d03984 70eval '$e = []; local(@$e)';
d441d3db 71like($@, qr/Can't localize through a reference/);
706a304b 72
82d03984 73eval '$e = {}; local(%$e)';
d441d3db 74like($@, qr/Can't localize through a reference/);
85aff577 75
161b7d16 76# Array and hash elements
77
78@a = ('a', 'b', 'c');
79{
80 local($a[1]) = 'foo';
81 local($a[2]) = $a[2];
d441d3db 82 is($a[1], 'foo');
83 is($a[2], 'c');
161b7d16 84 undef @a;
85}
d441d3db 86is($a[1], 'b');
87is($a[2], 'c');
88ok(!defined $a[0]);
161b7d16 89
90@a = ('a', 'b', 'c');
91{
92 local($a[1]) = "X";
93 shift @a;
94}
d441d3db 95is($a[0].$a[1], "Xb");
d60c5a05 96{
97 my $d = "@a";
98 local @a = @a;
99 is("@a", $d);
100}
161b7d16 101
102%h = ('a' => 1, 'b' => 2, 'c' => 3);
103{
104 local($h{'a'}) = 'foo';
105 local($h{'b'}) = $h{'b'};
d441d3db 106 is($h{'a'}, 'foo');
107 is($h{'b'}, 2);
161b7d16 108 local($h{'c'});
109 delete $h{'c'};
110}
d441d3db 111is($h{'a'}, 1);
112is($h{'b'}, 2);
d60c5a05 113{
114 my $d = join("\n", map { "$_=>$h{$_}" } sort keys %h);
115 local %h = %h;
116 is(join("\n", map { "$_=>$h{$_}" } sort keys %h), $d);
117}
d441d3db 118is($h{'c'}, 3);
2bb40b7f 119
120# check for scope leakage
121$a = 'outer';
122if (1) { local $a = 'inner' }
d441d3db 123is($a, 'outer');
2bb40b7f 124
125# see if localization works when scope unwinds
126local $m = 5;
127eval {
128 for $m (6) {
129 local $m = 7;
130 die "bye";
131 }
132};
d441d3db 133is($m, 5);
4e4c362e 134
135# see if localization works on tied arrays
136{
137 package TA;
138 sub TIEARRAY { bless [], $_[0] }
139 sub STORE { print "# STORE [@_]\n"; $_[0]->[$_[1]] = $_[2] }
140 sub FETCH { my $v = $_[0]->[$_[1]]; print "# FETCH [@_=$v]\n"; $v }
141 sub CLEAR { print "# CLEAR [@_]\n"; @{$_[0]} = (); }
142 sub FETCHSIZE { scalar(@{$_[0]}) }
143 sub SHIFT { shift (@{$_[0]}) }
144 sub EXTEND {}
145}
146
147tie @a, 'TA';
148@a = ('a', 'b', 'c');
149{
150 local($a[1]) = 'foo';
be6c24e0 151 local($a[2]) = $a[2];
d441d3db 152 is($a[1], 'foo');
153 is($a[2], 'c');
4e4c362e 154 @a = ();
155}
d441d3db 156is($a[1], 'b');
157is($a[2], 'c');
158ok(!defined $a[0]);
d60c5a05 159{
160 my $d = "@a";
161 local @a = @a;
162 is("@a", $d);
163}
4e4c362e 164
165{
166 package TH;
167 sub TIEHASH { bless {}, $_[0] }
168 sub STORE { print "# STORE [@_]\n"; $_[0]->{$_[1]} = $_[2] }
169 sub FETCH { my $v = $_[0]->{$_[1]}; print "# FETCH [@_=$v]\n"; $v }
c39e6ab0 170 sub EXISTS { print "# EXISTS [@_]\n"; exists $_[0]->{$_[1]}; }
4e4c362e 171 sub DELETE { print "# DELETE [@_]\n"; delete $_[0]->{$_[1]}; }
172 sub CLEAR { print "# CLEAR [@_]\n"; %{$_[0]} = (); }
d60c5a05 173 sub FIRSTKEY { print "# FIRSTKEY [@_]\n"; keys %{$_[0]}; each %{$_[0]} }
174 sub NEXTKEY { print "# NEXTKEY [@_]\n"; each %{$_[0]} }
4e4c362e 175}
176
177# see if localization works on tied hashes
178tie %h, 'TH';
179%h = ('a' => 1, 'b' => 2, 'c' => 3);
180
181{
182 local($h{'a'}) = 'foo';
be6c24e0 183 local($h{'b'}) = $h{'b'};
159ad915 184 local($h{'y'});
185 local($h{'z'}) = 33;
d441d3db 186 is($h{'a'}, 'foo');
187 is($h{'b'}, 2);
4e4c362e 188 local($h{'c'});
189 delete $h{'c'};
190}
d441d3db 191is($h{'a'}, 1);
192is($h{'b'}, 2);
193is($h{'c'}, 3);
194# local() should preserve the existenceness of tied hash elements
195ok(! exists $h{'y'});
196ok(! exists $h{'z'});
d60c5a05 197TODO: {
198 todo_skip("Localize entire tied hash");
199 my $d = join("\n", map { "$_=>$h{$_}" } sort keys %h);
200 local %h = %h;
201 is(join("\n", map { "$_=>$h{$_}" } sort keys %h), $d);
202}
4e4c362e 203
204@a = ('a', 'b', 'c');
205{
206 local($a[1]) = "X";
207 shift @a;
208}
d441d3db 209is($a[0].$a[1], "Xb");
4e4c362e 210
be6c24e0 211# now try the same for %SIG
212
213$SIG{TERM} = 'foo';
214$SIG{INT} = \&foo;
215$SIG{__WARN__} = $SIG{INT};
216{
217 local($SIG{TERM}) = $SIG{TERM};
218 local($SIG{INT}) = $SIG{INT};
219 local($SIG{__WARN__}) = $SIG{__WARN__};
d441d3db 220 is($SIG{TERM}, 'main::foo');
221 is($SIG{INT}, \&foo);
222 is($SIG{__WARN__}, \&foo);
be6c24e0 223 local($SIG{INT});
224 delete $SIG{__WARN__};
225}
d441d3db 226is($SIG{TERM}, 'main::foo');
227is($SIG{INT}, \&foo);
228is($SIG{__WARN__}, \&foo);
d60c5a05 229{
230 my $d = join("\n", map { "$_=>$SIG{$_}" } sort keys %SIG);
231 local %SIG = %SIG;
232 is(join("\n", map { "$_=>$SIG{$_}" } sort keys %SIG), $d);
233}
be6c24e0 234
235# and for %ENV
236
237$ENV{_X_} = 'a';
238$ENV{_Y_} = 'b';
239$ENV{_Z_} = 'c';
240{
159ad915 241 local($ENV{_A_});
242 local($ENV{_B_}) = 'foo';
be6c24e0 243 local($ENV{_X_}) = 'foo';
244 local($ENV{_Y_}) = $ENV{_Y_};
d441d3db 245 is($ENV{_X_}, 'foo');
246 is($ENV{_Y_}, 'b');
be6c24e0 247 local($ENV{_Z_});
248 delete $ENV{_Z_};
249}
d441d3db 250is($ENV{_X_}, 'a');
251is($ENV{_Y_}, 'b');
252is($ENV{_Z_}, 'c');
253# local() should preserve the existenceness of %ENV elements
254ok(! exists $ENV{_A_});
255ok(! exists $ENV{_B_});
d60c5a05 256{
257 my $d = join("\n", map { "$_=>$ENV{$_}" } sort keys %ENV);
258 local %ENV = %ENV;
259 is(join("\n", map { "$_=>$ENV{$_}" } sort keys %ENV), $d);
260}
be6c24e0 261
0214ae40 262# does implicit localization in foreach skip magic?
263
d441d3db 264$_ = "o 0,o 1,";
0214ae40 265my $iter = 0;
266while (/(o.+?),/gc) {
d441d3db 267 is($1, "o $iter");
0214ae40 268 foreach (1..1) { $iter++ }
d441d3db 269 if ($iter > 2) { fail("endless loop"); last; }
0214ae40 270}
271
272{
273 package UnderScore;
274 sub TIESCALAR { bless \my $self, shift }
275 sub FETCH { die "read \$_ forbidden" }
276 sub STORE { die "write \$_ forbidden" }
277 tie $_, __PACKAGE__;
0214ae40 278 my @tests = (
279 "Nesting" => sub { print '#'; for (1..3) { print }
280 print "\n" }, 1,
281 "Reading" => sub { print }, 0,
282 "Matching" => sub { $x = /badness/ }, 0,
283 "Concat" => sub { $_ .= "a" }, 0,
284 "Chop" => sub { chop }, 0,
285 "Filetest" => sub { -x }, 0,
286 "Assignment" => sub { $_ = "Bad" }, 0,
287 # XXX whether next one should fail is debatable
288 "Local \$_" => sub { local $_ = 'ok?'; print }, 0,
289 "for local" => sub { for("#ok?\n"){ print } }, 1,
290 );
291 while ( ($name, $code, $ok) = splice(@tests, 0, 3) ) {
0214ae40 292 eval { &$code };
d441d3db 293 main::ok(($ok xor $@), "Underscore '$name'");
0214ae40 294 }
295 untie $_;
296}
297
1f5346dc 298{
299 # BUG 20001205.22
300 my %x;
301 $x{a} = 1;
302 { local $x{b} = 1; }
d441d3db 303 ok(! exists $x{b});
1f5346dc 304 { local @x{c,d,e}; }
d441d3db 305 ok(! exists $x{c});
1f5346dc 306}
159ad915 307
33f3c7b8 308# local() and readonly magic variables
309
310eval { local $1 = 1 };
d441d3db 311like($@, qr/Modification of a read-only value attempted/);
33f3c7b8 312
313eval { for ($1) { local $_ = 1 } };
d441d3db 314like($@, qr/Modification of a read-only value attempted/);
33f3c7b8 315
0cbee0a4 316# make sure $1 is still read-only
33f3c7b8 317eval { for ($1) { local $_ = 1 } };
d441d3db 318like($@, qr/Modification of a read-only value attempted/);
ac117f44 319
320# The s/// adds 'g' magic to $_, but it should remain non-readonly
321eval { for("a") { for $x (1,2) { local $_="b"; s/(.*)/+$1/ } } };
d441d3db 322is($@, "");